[Lldb-commits] [lldb] LLDB Statusline (PR #121860)

Jonas Devlieghere via lldb-commits lldb-commits at lists.llvm.org
Mon Jan 6 16:15:26 PST 2025


https://github.com/JDevlieghere created https://github.com/llvm/llvm-project/pull/121860

This is a draft PR with my prototype implementation of https://discourse.llvm.org/t/rfc-lldb-statusline/83948

>From 24fea391893de79fc0455a81c1ce6cd77b6106fc Mon Sep 17 00:00:00 2001
From: Jonas Devlieghere <jonas at devlieghere.com>
Date: Fri, 3 Jan 2025 18:13:16 -0800
Subject: [PATCH 1/2] Add OptionValueFormatEntityList

---
 lldb/include/lldb/Interpreter/OptionValue.h   |  13 +-
 .../Interpreter/OptionValueFormatEntityList.h |  62 +++++
 lldb/include/lldb/Interpreter/OptionValues.h  |   1 +
 lldb/include/lldb/lldb-forward.h              |   1 +
 lldb/source/Interpreter/CMakeLists.txt        |   1 +
 lldb/source/Interpreter/OptionValue.cpp       |  24 ++
 .../OptionValueFormatEntityList.cpp           | 223 ++++++++++++++++++
 lldb/source/Interpreter/Property.cpp          |   9 +
 8 files changed, 331 insertions(+), 3 deletions(-)
 create mode 100644 lldb/include/lldb/Interpreter/OptionValueFormatEntityList.h
 create mode 100644 lldb/source/Interpreter/OptionValueFormatEntityList.cpp

diff --git a/lldb/include/lldb/Interpreter/OptionValue.h b/lldb/include/lldb/Interpreter/OptionValue.h
index d19c8b8fab6222..73944c133d3b4e 100644
--- a/lldb/include/lldb/Interpreter/OptionValue.h
+++ b/lldb/include/lldb/Interpreter/OptionValue.h
@@ -51,7 +51,8 @@ class OptionValue {
     eTypeString,
     eTypeUInt64,
     eTypeUUID,
-    eTypeFormatEntity
+    eTypeFormatEntity,
+    eTypeFormatEntityList,
   };
 
   enum {
@@ -72,7 +73,7 @@ class OptionValue {
   virtual ~OptionValue() = default;
 
   OptionValue(const OptionValue &other);
-  
+
   OptionValue& operator=(const OptionValue &other);
 
   // Subclasses should override these functions
@@ -249,6 +250,9 @@ class OptionValue {
   OptionValueFormatEntity *GetAsFormatEntity();
   const OptionValueFormatEntity *GetAsFormatEntity() const;
 
+  OptionValueFormatEntityList *GetAsFormatEntityList();
+  const OptionValueFormatEntityList *GetAsFormatEntityList() const;
+
   bool AppendFileSpecValue(FileSpec file_spec);
 
   bool OptionWasSet() const { return m_value_was_set; }
@@ -286,6 +290,8 @@ class OptionValue {
       return GetFileSpecValue();
     if constexpr (std::is_same_v<T, FileSpecList>)
       return GetFileSpecListValue();
+    if constexpr (std::is_same_v<T, std::vector<FormatEntity::Entry>>)
+      return GetFormatEntityList();
     if constexpr (std::is_same_v<T, lldb::LanguageType>)
       return GetLanguageValue();
     if constexpr (std::is_same_v<T, llvm::StringRef>)
@@ -387,8 +393,9 @@ class OptionValue {
   bool SetUUIDValue(const UUID &uuid);
 
   const FormatEntity::Entry *GetFormatEntity() const;
+  std::vector<FormatEntity::Entry> GetFormatEntityList() const;
   const RegularExpression *GetRegexValue() const;
-  
+
   mutable std::mutex m_mutex;
 };
 
diff --git a/lldb/include/lldb/Interpreter/OptionValueFormatEntityList.h b/lldb/include/lldb/Interpreter/OptionValueFormatEntityList.h
new file mode 100644
index 00000000000000..4270bf26b54c9f
--- /dev/null
+++ b/lldb/include/lldb/Interpreter/OptionValueFormatEntityList.h
@@ -0,0 +1,62 @@
+//===-- OptionValueFormatEntityList.h --------------------------------*-
+// C++-*-===//
+//
+// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
+// See https://llvm.org/LICENSE.txt for license information.
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
+//
+//===----------------------------------------------------------------------===//
+
+#ifndef LLDB_INTERPRETER_OPTIONVALUEFORMATENTITYLIST_H
+#define LLDB_INTERPRETER_OPTIONVALUEFORMATENTITYLIST_H
+
+#include "lldb/Core/FormatEntity.h"
+#include "lldb/Interpreter/OptionValue.h"
+
+namespace lldb_private {
+
+class OptionValueFormatEntityList
+    : public Cloneable<OptionValueFormatEntityList, OptionValue> {
+public:
+  OptionValueFormatEntityList();
+
+  OptionValueFormatEntityList(const OptionValueFormatEntityList &other)
+      : Cloneable(other), m_current_entries(other.m_current_entries),
+        m_current_formats(other.m_current_formats) {}
+
+  ~OptionValueFormatEntityList() override = default;
+
+  // Virtual subclass pure virtual overrides
+
+  OptionValue::Type GetType() const override { return eTypeFormatEntityList; }
+
+  void DumpValue(const ExecutionContext *exe_ctx, Stream &strm,
+                 uint32_t dump_mask) override;
+
+  Status
+  SetValueFromString(llvm::StringRef value,
+                     VarSetOperationType op = eVarSetOperationAssign) override;
+
+  void Clear() override;
+
+  std::vector<FormatEntity::Entry> GetCurrentValue() const {
+    std::lock_guard<std::recursive_mutex> lock(m_mutex);
+    return m_current_entries;
+  }
+
+protected:
+  llvm::Error Append(llvm::StringRef str);
+  llvm::Error Insert(size_t idx, llvm::StringRef str);
+  llvm::Error Replace(size_t idx, llvm::StringRef str);
+  llvm::Error Remove(size_t idx);
+
+  lldb::OptionValueSP Clone() const override;
+
+  std::vector<FormatEntity::Entry> m_current_entries;
+  std::vector<std::string> m_current_formats;
+  mutable std::recursive_mutex m_mutex;
+};
+
+} // namespace lldb_private
+
+#endif // LLDB_INTERPRETER_OPTIONVALUEFORMATENTITYLIST_H
diff --git a/lldb/include/lldb/Interpreter/OptionValues.h b/lldb/include/lldb/Interpreter/OptionValues.h
index 6efc9e1ad064ce..399119e290401b 100644
--- a/lldb/include/lldb/Interpreter/OptionValues.h
+++ b/lldb/include/lldb/Interpreter/OptionValues.h
@@ -22,6 +22,7 @@
 #include "lldb/Interpreter/OptionValueFileSpecList.h"
 #include "lldb/Interpreter/OptionValueFormat.h"
 #include "lldb/Interpreter/OptionValueFormatEntity.h"
+#include "lldb/Interpreter/OptionValueFormatEntityList.h"
 #include "lldb/Interpreter/OptionValueLanguage.h"
 #include "lldb/Interpreter/OptionValuePathMappings.h"
 #include "lldb/Interpreter/OptionValueProperties.h"
diff --git a/lldb/include/lldb/lldb-forward.h b/lldb/include/lldb/lldb-forward.h
index d09edeeccaff1a..5831d62f5e9338 100644
--- a/lldb/include/lldb/lldb-forward.h
+++ b/lldb/include/lldb/lldb-forward.h
@@ -150,6 +150,7 @@ class OptionValueFileSpec;
 class OptionValueFileSpecList;
 class OptionValueFormat;
 class OptionValueFormatEntity;
+class OptionValueFormatEntityList;
 class OptionValueLanguage;
 class OptionValuePathMappings;
 class OptionValueProperties;
diff --git a/lldb/source/Interpreter/CMakeLists.txt b/lldb/source/Interpreter/CMakeLists.txt
index 642263a8bda7fa..4fca862de9cfcc 100644
--- a/lldb/source/Interpreter/CMakeLists.txt
+++ b/lldb/source/Interpreter/CMakeLists.txt
@@ -41,6 +41,7 @@ add_lldb_library(lldbInterpreter NO_PLUGIN_DEPENDENCIES
   OptionValueFileSpecList.cpp
   OptionValueFormat.cpp
   OptionValueFormatEntity.cpp
+  OptionValueFormatEntityList.cpp
   OptionValueLanguage.cpp
   OptionValuePathMappings.cpp
   OptionValueProperties.cpp
diff --git a/lldb/source/Interpreter/OptionValue.cpp b/lldb/source/Interpreter/OptionValue.cpp
index b95f4fec339499..d7c1996b24a0cc 100644
--- a/lldb/source/Interpreter/OptionValue.cpp
+++ b/lldb/source/Interpreter/OptionValue.cpp
@@ -184,6 +184,18 @@ const OptionValueFormatEntity *OptionValue::GetAsFormatEntity() const {
   return nullptr;
 }
 
+OptionValueFormatEntityList *OptionValue::GetAsFormatEntityList() {
+  if (GetType() == OptionValue::eTypeFormatEntityList)
+    return static_cast<OptionValueFormatEntityList *>(this);
+  return nullptr;
+}
+
+const OptionValueFormatEntityList *OptionValue::GetAsFormatEntityList() const {
+  if (GetType() == OptionValue::eTypeFormatEntityList)
+    return static_cast<const OptionValueFormatEntityList *>(this);
+  return nullptr;
+}
+
 OptionValuePathMappings *OptionValue::GetAsPathMappings() {
   if (GetType() == OptionValue::eTypePathMap)
     return static_cast<OptionValuePathMappings *>(this);
@@ -380,6 +392,13 @@ bool OptionValue::SetLanguageValue(lldb::LanguageType new_language) {
   return false;
 }
 
+std::vector<FormatEntity::Entry> OptionValue::GetFormatEntityList() const {
+  std::lock_guard<std::mutex> lock(m_mutex);
+  if (const OptionValueFormatEntityList *option_value = GetAsFormatEntityList())
+    return option_value->GetCurrentValue();
+  return {};
+}
+
 const FormatEntity::Entry *OptionValue::GetFormatEntity() const {
   std::lock_guard<std::mutex> lock(m_mutex);
   if (const OptionValueFormatEntity *option_value = GetAsFormatEntity())
@@ -502,6 +521,8 @@ const char *OptionValue::GetBuiltinTypeAsCString(Type t) {
     return "format";
   case eTypeFormatEntity:
     return "format-string";
+  case eTypeFormatEntityList:
+    return "format-string-list";
   case eTypeLanguage:
     return "language";
   case eTypePathMap:
@@ -546,6 +567,9 @@ lldb::OptionValueSP OptionValue::CreateValueFromCStringForTypeMask(
   case 1u << eTypeFormatEntity:
     value_sp = std::make_shared<OptionValueFormatEntity>(nullptr);
     break;
+  case 1u << eTypeFormatEntityList:
+    value_sp = std::make_shared<OptionValueFormatEntityList>();
+    break;
   case 1u << eTypeLanguage:
     value_sp = std::make_shared<OptionValueLanguage>(eLanguageTypeUnknown);
     break;
diff --git a/lldb/source/Interpreter/OptionValueFormatEntityList.cpp b/lldb/source/Interpreter/OptionValueFormatEntityList.cpp
new file mode 100644
index 00000000000000..6f7d46fa68618c
--- /dev/null
+++ b/lldb/source/Interpreter/OptionValueFormatEntityList.cpp
@@ -0,0 +1,223 @@
+//===-- OptionValueFormatEntityList.cpp
+//---------------------------------------===//
+//
+// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
+// See https://llvm.org/LICENSE.txt for license information.
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
+//
+//===----------------------------------------------------------------------===//
+
+#include "lldb/Interpreter/OptionValueFormatEntityList.h"
+
+#include "lldb/Core/Module.h"
+#include "lldb/Interpreter/CommandInterpreter.h"
+#include "lldb/Utility/Stream.h"
+#include "lldb/Utility/StringList.h"
+using namespace lldb;
+using namespace lldb_private;
+
+OptionValueFormatEntityList::OptionValueFormatEntityList() {}
+
+void OptionValueFormatEntityList::Clear() {
+  m_current_entries.clear();
+  m_current_formats.clear();
+  m_value_was_set = false;
+}
+
+static void EscapeBackticks(llvm::StringRef str, std::string &dst) {
+  dst.clear();
+  dst.reserve(str.size());
+
+  for (size_t i = 0, e = str.size(); i != e; ++i) {
+    char c = str[i];
+    if (c == '`') {
+      if (i == 0 || str[i - 1] != '\\')
+        dst += '\\';
+    }
+    dst += c;
+  }
+}
+
+void OptionValueFormatEntityList::DumpValue(const ExecutionContext *exe_ctx,
+                                            Stream &strm, uint32_t dump_mask) {
+  if (dump_mask & eDumpOptionType)
+    strm.Printf("(%s)", GetTypeAsCString());
+  if (dump_mask & eDumpOptionValue) {
+    const bool one_line = dump_mask & eDumpOptionCommand;
+    const size_t size = m_current_formats.size();
+    if (dump_mask & eDumpOptionType)
+      strm.Printf(" =%s", (size > 0 && !one_line) ? "\n" : "");
+    if (!one_line)
+      strm.IndentMore();
+    for (uint32_t i = 0; i < size; ++i) {
+      if (!one_line) {
+        strm.Indent();
+        strm.Printf("[%u]: ", i);
+      }
+      strm << m_current_formats[i];
+      if (one_line)
+        strm << ' ';
+    }
+    if (!one_line)
+      strm.IndentLess();
+  }
+}
+
+static llvm::Expected<FormatEntity::Entry> Parse(llvm::StringRef str) {
+  FormatEntity::Entry entry;
+  Status error = FormatEntity::Parse(str, entry);
+  if (error.Fail())
+    return error.ToError();
+  return entry;
+}
+
+llvm::Error OptionValueFormatEntityList::Append(llvm::StringRef str) {
+  auto maybe_entry = Parse(str);
+  if (!maybe_entry)
+    return maybe_entry.takeError();
+
+  m_current_entries.emplace_back(*maybe_entry);
+  m_current_formats.emplace_back(str);
+
+  return llvm::Error::success();
+}
+
+llvm::Error OptionValueFormatEntityList::Insert(size_t idx,
+                                                llvm::StringRef str) {
+  if (idx >= m_current_formats.size())
+    return llvm::createStringError(
+        "invalid file list index %s, index must be 0 through %u", idx,
+        m_current_formats.size());
+
+  auto maybe_entry = Parse(str);
+  if (!maybe_entry)
+    return maybe_entry.takeError();
+
+  m_current_entries.insert(m_current_entries.begin() + idx, *maybe_entry);
+  m_current_formats.insert(m_current_formats.begin() + idx, std::string(str));
+
+  return llvm::Error::success();
+  return llvm::Error::success();
+}
+
+llvm::Error OptionValueFormatEntityList::Replace(size_t idx,
+                                                 llvm::StringRef str) {
+  if (idx >= m_current_formats.size())
+    return llvm::createStringError(
+        "invalid file list index %s, index must be 0 through %u", idx,
+        m_current_formats.size());
+
+  auto maybe_entry = Parse(str);
+  if (!maybe_entry)
+    return maybe_entry.takeError();
+
+  m_current_entries[idx] = *maybe_entry;
+  m_current_formats[idx] = str;
+
+  return llvm::Error::success();
+}
+
+llvm::Error OptionValueFormatEntityList::Remove(size_t idx) {
+  if (idx >= m_current_formats.size())
+    return llvm::createStringError(
+        "invalid fromat entry list index %s, index must be 0 through %u", idx,
+        m_current_formats.size());
+
+  m_current_formats.erase(m_current_formats.begin() + idx);
+  m_current_entries.erase(m_current_entries.begin() + idx);
+
+  return llvm::Error::success();
+}
+
+Status
+OptionValueFormatEntityList::SetValueFromString(llvm::StringRef value_str,
+                                                VarSetOperationType op) {
+  std::lock_guard<std::recursive_mutex> lock(m_mutex);
+
+  Args args(value_str.str());
+  const size_t argc = args.GetArgumentCount();
+
+  switch (op) {
+  case eVarSetOperationClear:
+    Clear();
+    NotifyValueChanged();
+    break;
+
+  case eVarSetOperationReplace:
+
+  case eVarSetOperationAssign:
+    Clear();
+    // Fall through to append case
+    [[fallthrough]];
+  case eVarSetOperationAppend:
+    if (argc > 0) {
+      m_value_was_set = true;
+      for (size_t i = 0; i < argc; ++i) {
+        if (llvm::Error err = Append(args.GetArgumentAtIndex(i)))
+          return Status::FromError(std::move(err));
+      }
+      NotifyValueChanged();
+    } else {
+      return Status::FromErrorString(
+          "assign operation takes at least one file path argument");
+    }
+    break;
+  case eVarSetOperationInsertBefore:
+  case eVarSetOperationInsertAfter:
+    if (argc > 1) {
+      uint32_t idx;
+      if (!llvm::to_integer(args.GetArgumentAtIndex(0), idx))
+        return Status::FromErrorStringWithFormat("invalid index %s",
+                                                 args.GetArgumentAtIndex(0));
+      if (op == eVarSetOperationInsertAfter)
+        ++idx;
+      for (size_t i = 1; i < argc; ++i, ++idx) {
+        if (llvm::Error err = Append(args.GetArgumentAtIndex(i)))
+          return Status::FromError(std::move(err));
+      }
+      NotifyValueChanged();
+    } else {
+      return Status::FromErrorString(
+          "insert operation takes an array index followed by "
+          "one or more values");
+    }
+    break;
+  case eVarSetOperationRemove:
+    if (argc > 0) {
+      std::vector<int> indexes;
+      for (size_t i = 0; i < argc; ++i) {
+        int idx;
+        if (!llvm::to_integer(args.GetArgumentAtIndex(i), idx))
+          return Status::FromErrorStringWithFormat(
+              "invalid array index '%s', aborting remove operation",
+              args.GetArgumentAtIndex(i));
+        indexes.push_back(idx);
+      }
+
+      if (indexes.empty())
+        return Status::FromErrorString(
+            "remove operation takes one or more array index");
+
+      // Sort and then erase in reverse so indexes are always valid.
+      llvm::sort(indexes);
+      for (auto i : llvm::reverse(indexes)) {
+        if (auto err = Remove(i))
+          return Status::FromError(std::move(err));
+      }
+    } else {
+      return Status::FromErrorString(
+          "remove operation takes one or more array index");
+    }
+    break;
+  case eVarSetOperationInvalid:
+    return OptionValue::SetValueFromString(value_str, op);
+    break;
+  }
+
+  return {};
+}
+
+OptionValueSP OptionValueFormatEntityList::Clone() const {
+  std::lock_guard<std::recursive_mutex> lock(m_mutex);
+  return Cloneable::Clone();
+}
diff --git a/lldb/source/Interpreter/Property.cpp b/lldb/source/Interpreter/Property.cpp
index 56e45363be89a1..331735f167385a 100644
--- a/lldb/source/Interpreter/Property.cpp
+++ b/lldb/source/Interpreter/Property.cpp
@@ -161,6 +161,15 @@ Property::Property(const PropertyDefinition &definition)
         definition.default_cstr_value);
     break;
 
+  case OptionValue::eTypeFormatEntityList:
+    // "definition.default_uint_value" is not used for a
+    // OptionValue::eTypeFormatEntityList
+    m_value_sp = std::make_shared<OptionValueFormatEntityList>();
+    if (definition.default_cstr_value) {
+      m_value_sp->SetValueFromString(definition.default_cstr_value);
+    }
+    break;
+
   case OptionValue::eTypePathMap:
     // "definition.default_uint_value" tells us if notifications should occur
     // for path mappings

>From 338883256bc5e832bef2b3595904206e50a589cc Mon Sep 17 00:00:00 2001
From: Jonas Devlieghere <jonas at devlieghere.com>
Date: Wed, 18 Dec 2024 16:05:48 -0800
Subject: [PATCH 2/2] Add LLDB statusline

---
 lldb/include/lldb/Core/Debugger.h             |  10 +-
 lldb/include/lldb/Core/Statusline.h           |  73 ++++++
 lldb/source/Core/CMakeLists.txt               |   1 +
 lldb/source/Core/CoreProperties.td            |   8 +
 lldb/source/Core/Debugger.cpp                 | 110 +++-----
 lldb/source/Core/Statusline.cpp               | 247 ++++++++++++++++++
 .../source/Interpreter/CommandInterpreter.cpp |   8 +-
 7 files changed, 376 insertions(+), 81 deletions(-)
 create mode 100644 lldb/include/lldb/Core/Statusline.h
 create mode 100644 lldb/source/Core/Statusline.cpp

diff --git a/lldb/include/lldb/Core/Debugger.h b/lldb/include/lldb/Core/Debugger.h
index 70f4c4216221c6..764275105ee494 100644
--- a/lldb/include/lldb/Core/Debugger.h
+++ b/lldb/include/lldb/Core/Debugger.h
@@ -19,6 +19,7 @@
 #include "lldb/Core/FormatEntity.h"
 #include "lldb/Core/IOHandler.h"
 #include "lldb/Core/SourceManager.h"
+#include "lldb/Core/Statusline.h"
 #include "lldb/Core/UserSettingsController.h"
 #include "lldb/Host/HostThread.h"
 #include "lldb/Host/StreamFile.h"
@@ -308,6 +309,10 @@ class Debugger : public std::enable_shared_from_this<Debugger>,
 
   bool SetShowProgress(bool show_progress);
 
+  bool GetShowStatusline() const;
+
+  std::vector<FormatEntity::Entry> GetStatuslineFormat() const;
+
   llvm::StringRef GetShowProgressAnsiPrefix() const;
 
   llvm::StringRef GetShowProgressAnsiSuffix() const;
@@ -595,6 +600,9 @@ class Debugger : public std::enable_shared_from_this<Debugger>,
   /// Manually stop the debugger's default event handler.
   void StopEventHandlerThread();
 
+  void ShowStatusline();
+  void HideStatusline();
+
   /// Force flushing the process's pending stdout and stderr to the debugger's
   /// asynchronous stdout and stderr streams.
   void FlushProcessOutput(Process &process, bool flush_stdout,
@@ -728,7 +736,7 @@ class Debugger : public std::enable_shared_from_this<Debugger>,
   IOHandlerStack m_io_handler_stack;
   std::recursive_mutex m_io_handler_synchronous_mutex;
 
-  std::optional<uint64_t> m_current_event_id;
+  std::optional<Statusline> m_statusline;
 
   llvm::StringMap<std::weak_ptr<LogHandler>> m_stream_handlers;
   std::shared_ptr<CallbackLogHandler> m_callback_handler_sp;
diff --git a/lldb/include/lldb/Core/Statusline.h b/lldb/include/lldb/Core/Statusline.h
new file mode 100644
index 00000000000000..ad5c0e47c2803d
--- /dev/null
+++ b/lldb/include/lldb/Core/Statusline.h
@@ -0,0 +1,73 @@
+//===-- Statusline.h -----------------------------------------------------===//
+//
+// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
+// See https://llvm.org/LICENSE.txt for license information.
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
+//
+//===----------------------------------------------------------------------===//
+#include "lldb/Core/Debugger.h"
+#include "llvm/ADT/SmallVector.h"
+#include <string>
+#ifndef LLDB_CORE_STATUSBAR_H
+#define LLDB_CORE_STATUSBAR_H
+
+namespace lldb_private {
+class Statusline {
+public:
+  Statusline(Debugger &debugger);
+  ~Statusline();
+
+  void Enable();
+  void Disable();
+
+  void Clear();
+  void Draw(llvm::StringRef msg);
+
+  bool StartStatuslineThread();
+  void StopStatuslineThread();
+
+  void ReportProgress(const ProgressEventData &data);
+
+  void TerminalSizeChanged() { m_terminal_size_has_changed = 1; }
+
+private:
+  // Update terminal dimensions.
+  void UpdateTerminalProperties();
+
+  // Set the scroll window to the given height.
+  void SetScrollWindow(uint64_t height);
+
+  // Write at the given column.
+  void AddAtPosition(uint64_t col, llvm::StringRef str);
+
+  // Clear the statusline (without redrawing the background).
+  void Reset();
+
+  bool IsSupported() const;
+
+  lldb::thread_result_t StatuslineThread();
+
+  Debugger &m_debugger;
+  HostThread m_statusline_thread;
+
+  struct ProgressReport {
+    uint64_t id;
+    std::string message;
+  };
+  llvm::SmallVector<ProgressReport, 8> m_progress_reports;
+
+  volatile std::sig_atomic_t m_terminal_size_has_changed = 1;
+  uint64_t m_terminal_width = 0;
+  uint64_t m_terminal_height = 0;
+  uint64_t m_scroll_height = 0;
+
+  std::string m_ansi_prefix = "${ansi.bg.cyan}${ansi.fg.black}";
+  std::string m_ansi_suffix = "${ansi.normal}";
+  llvm::SmallVector<FormatEntity::Entry> m_components;
+
+  bool m_statusline_thread_exit = false;
+  std::mutex m_statusline_mutex;
+  std::condition_variable m_statusline_cv;
+};
+} // namespace lldb_private
+#endif // LLDB_CORE_STATUSBAR_H
diff --git a/lldb/source/Core/CMakeLists.txt b/lldb/source/Core/CMakeLists.txt
index 6d14f7a87764e0..5d4576837dbe61 100644
--- a/lldb/source/Core/CMakeLists.txt
+++ b/lldb/source/Core/CMakeLists.txt
@@ -46,6 +46,7 @@ add_lldb_library(lldbCore
   Opcode.cpp
   PluginManager.cpp
   Progress.cpp
+  Statusline.cpp
   RichManglingContext.cpp
   SearchFilter.cpp
   Section.cpp
diff --git a/lldb/source/Core/CoreProperties.td b/lldb/source/Core/CoreProperties.td
index d3816c3070bbc5..37b0ec231d4d48 100644
--- a/lldb/source/Core/CoreProperties.td
+++ b/lldb/source/Core/CoreProperties.td
@@ -172,6 +172,14 @@ let Definition = "debugger" in {
     Global,
     DefaultStringValue<"${ansi.normal}">,
     Desc<"When displaying progress in a color-enabled terminal, use the ANSI terminal code specified in this format immediately after the progress message.">;
+  def ShowStatusline: Property<"show-statusline", "Boolean">,
+    Global,
+    DefaultTrue,
+    Desc<"Whether to show a statusline at the bottom of the terminal.">;
+  def StatuslineFormat: Property<"statusline-format", "FormatEntityList">,
+    Global,
+    DefaultStringValue<"${module.file.basename} ${line.file.basename}:${line.number}:${line.column} ${thread.stop-reason}">,
+    Desc<"List of statusline format entities.">;
   def UseSourceCache: Property<"use-source-cache", "Boolean">,
     Global,
     DefaultTrue,
diff --git a/lldb/source/Core/Debugger.cpp b/lldb/source/Core/Debugger.cpp
index 6ceb209269c9e7..3e83536544a4c6 100644
--- a/lldb/source/Core/Debugger.cpp
+++ b/lldb/source/Core/Debugger.cpp
@@ -376,6 +376,8 @@ bool Debugger::SetTerminalWidth(uint64_t term_width) {
 
   if (auto handler_sp = m_io_handler_stack.Top())
     handler_sp->TerminalSizeChanged();
+  if (m_statusline)
+    m_statusline->TerminalSizeChanged();
 
   return success;
 }
@@ -392,6 +394,8 @@ bool Debugger::SetTerminalHeight(uint64_t term_height) {
 
   if (auto handler_sp = m_io_handler_stack.Top())
     handler_sp->TerminalSizeChanged();
+  if (m_statusline)
+    m_statusline->TerminalSizeChanged();
 
   return success;
 }
@@ -454,6 +458,17 @@ llvm::StringRef Debugger::GetShowProgressAnsiSuffix() const {
       idx, g_debugger_properties[idx].default_cstr_value);
 }
 
+bool Debugger::GetShowStatusline() const {
+  const uint32_t idx = ePropertyShowStatusline;
+  return GetPropertyAtIndexAs<bool>(
+      idx, g_debugger_properties[idx].default_uint_value != 0);
+}
+
+std::vector<FormatEntity::Entry> Debugger::GetStatuslineFormat() const {
+  const uint32_t idx = ePropertyStatuslineFormat;
+  return GetPropertyAtIndexAs<std::vector<FormatEntity::Entry>>(idx, {});
+}
+
 bool Debugger::GetUseAutosuggestion() const {
   const uint32_t idx = ePropertyShowAutosuggestion;
   return GetPropertyAtIndexAs<bool>(
@@ -989,6 +1004,7 @@ void Debugger::Clear() {
     ClearIOHandlers();
     StopIOHandlerThread();
     StopEventHandlerThread();
+    HideStatusline();
     m_listener_sp->Clear();
     for (TargetSP target_sp : m_target_list.Targets()) {
       if (target_sp) {
@@ -2080,84 +2096,8 @@ void Debugger::HandleProgressEvent(const lldb::EventSP &event_sp) {
   if (!data)
     return;
 
-  // Do some bookkeeping for the current event, regardless of whether we're
-  // going to show the progress.
-  const uint64_t id = data->GetID();
-  if (m_current_event_id) {
-    Log *log = GetLog(LLDBLog::Events);
-    if (log && log->GetVerbose()) {
-      StreamString log_stream;
-      log_stream.AsRawOstream()
-          << static_cast<void *>(this) << " Debugger(" << GetID()
-          << ")::HandleProgressEvent( m_current_event_id = "
-          << *m_current_event_id << ", data = { ";
-      data->Dump(&log_stream);
-      log_stream << " } )";
-      log->PutString(log_stream.GetString());
-    }
-    if (id != *m_current_event_id)
-      return;
-    if (data->GetCompleted() == data->GetTotal())
-      m_current_event_id.reset();
-  } else {
-    m_current_event_id = id;
-  }
-
-  // Decide whether we actually are going to show the progress. This decision
-  // can change between iterations so check it inside the loop.
-  if (!GetShowProgress())
-    return;
-
-  // Determine whether the current output file is an interactive terminal with
-  // color support. We assume that if we support ANSI escape codes we support
-  // vt100 escape codes.
-  File &file = GetOutputFile();
-  if (!file.GetIsInteractive() || !file.GetIsTerminalWithColors())
-    return;
-
-  StreamSP output = GetAsyncOutputStream();
-
-  // Print over previous line, if any.
-  output->Printf("\r");
-
-  if (data->GetCompleted() == data->GetTotal()) {
-    // Clear the current line.
-    output->Printf("\x1B[2K");
-    output->Flush();
-    return;
-  }
-
-  // Trim the progress message if it exceeds the window's width and print it.
-  std::string message = data->GetMessage();
-  if (data->IsFinite())
-    message = llvm::formatv("[{0}/{1}] {2}", data->GetCompleted(),
-                            data->GetTotal(), message)
-                  .str();
-
-  // Trim the progress message if it exceeds the window's width and print it.
-  const uint32_t term_width = GetTerminalWidth();
-  const uint32_t ellipsis = 3;
-  if (message.size() + ellipsis >= term_width)
-    message.resize(term_width - ellipsis);
-
-  const bool use_color = GetUseColor();
-  llvm::StringRef ansi_prefix = GetShowProgressAnsiPrefix();
-  if (!ansi_prefix.empty())
-    output->Printf(
-        "%s", ansi::FormatAnsiTerminalCodes(ansi_prefix, use_color).c_str());
-
-  output->Printf("%s...", message.c_str());
-
-  llvm::StringRef ansi_suffix = GetShowProgressAnsiSuffix();
-  if (!ansi_suffix.empty())
-    output->Printf(
-        "%s", ansi::FormatAnsiTerminalCodes(ansi_suffix, use_color).c_str());
-
-  // Clear until the end of the line.
-  output->Printf("\x1B[K\r");
-
-  // Flush the output.
-  output->Flush();
+  if (m_statusline)
+    m_statusline->ReportProgress(*data);
 }
 
 void Debugger::HandleDiagnosticEvent(const lldb::EventSP &event_sp) {
@@ -2169,6 +2109,20 @@ void Debugger::HandleDiagnosticEvent(const lldb::EventSP &event_sp) {
   data->Dump(stream.get());
 }
 
+void Debugger::ShowStatusline() {
+  if (!m_statusline && GetShowStatusline()) {
+    m_statusline.emplace(*this);
+    m_statusline->StartStatuslineThread();
+  }
+}
+
+void Debugger::HideStatusline() {
+  if (m_statusline) {
+    m_statusline->StopStatuslineThread();
+    m_statusline.reset();
+  }
+}
+
 bool Debugger::HasIOHandlerThread() const {
   return m_io_handler_thread.IsJoinable();
 }
diff --git a/lldb/source/Core/Statusline.cpp b/lldb/source/Core/Statusline.cpp
new file mode 100644
index 00000000000000..018742ac33b584
--- /dev/null
+++ b/lldb/source/Core/Statusline.cpp
@@ -0,0 +1,247 @@
+//===-- Statusline.cpp ---------------------------------------------------===//
+//
+// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
+// See https://llvm.org/LICENSE.txt for license information.
+// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
+//
+//===----------------------------------------------------------------------===//
+
+#include "lldb/Core/Statusline.h"
+#include "lldb/Core/Debugger.h"
+#include "lldb/Core/FormatEntity.h"
+#include "lldb/Host/ThreadLauncher.h"
+#include "lldb/Interpreter/CommandInterpreter.h"
+#include "lldb/Symbol/SymbolContext.h"
+#include "lldb/Target/StackFrame.h"
+#include "lldb/Utility/AnsiTerminal.h"
+#include "lldb/Utility/LLDBLog.h"
+#include "lldb/Utility/Log.h"
+#include "lldb/Utility/StreamString.h"
+
+#include <sys/ioctl.h>
+#include <termios.h>
+
+#define ESCAPE "\x1b"
+#define ANSI_SAVE_CURSOR ESCAPE "7"
+#define ANSI_RESTORE_CURSOR ESCAPE "8"
+#define ANSI_CLEAR_BELOW ESCAPE "[J"
+#define ANSI_CLEAR_LINE "\r\x1B[2K"
+#define ANSI_SET_SCROLL_ROWS ESCAPE "[0;%ur"
+#define ANSI_TO_START_OF_ROW ESCAPE "[%u;0f"
+#define ANSI_UP_ROWS ESCAPE "[%dA"
+#define ANSI_DOWN_ROWS ESCAPE "[%dB"
+#define ANSI_FORWARD_COLS ESCAPE "\033[%dC"
+#define ANSI_BACKWARD_COLS ESCAPE "\033[%dD"
+
+using namespace lldb;
+using namespace lldb_private;
+
+Statusline::Statusline(Debugger &debugger) : m_debugger(debugger) {}
+
+Statusline::~Statusline() { StopStatuslineThread(); }
+
+bool Statusline::IsSupported() const {
+  File &file = m_debugger.GetOutputFile();
+  return file.GetIsInteractive() && file.GetIsTerminalWithColors();
+}
+
+void Statusline::Enable() {
+  if (!IsSupported())
+    return;
+
+  UpdateTerminalProperties();
+  // Reduce the scroll window to make space for the status bar below.
+  SetScrollWindow(m_terminal_height - 1);
+}
+
+void Statusline::Disable() {
+  UpdateTerminalProperties();
+  // Clear the previous status bar if any.
+  Clear();
+  // Extend the scroll window to cover the status bar.
+  SetScrollWindow(m_terminal_height);
+}
+
+void Statusline::Draw(llvm::StringRef str) {
+  UpdateTerminalProperties();
+
+  const uint32_t ellipsis = 3;
+  if (str.size() + ellipsis >= m_terminal_width)
+    str = str.substr(0, m_terminal_width - ellipsis);
+
+  StreamFile &out = m_debugger.GetOutputStream();
+  out << ANSI_SAVE_CURSOR;
+  out.Printf(ANSI_TO_START_OF_ROW, static_cast<unsigned>(m_terminal_height));
+  out << ANSI_CLEAR_LINE;
+  out << ansi::FormatAnsiTerminalCodes(m_ansi_prefix);
+  out << str;
+  out << std::string(m_terminal_width - str.size(), ' ');
+  out << ansi::FormatAnsiTerminalCodes(m_ansi_suffix);
+  out << ANSI_RESTORE_CURSOR;
+}
+
+void Statusline::Reset() {
+  StreamFile &out = m_debugger.GetOutputStream();
+  out << ANSI_SAVE_CURSOR;
+  out.Printf(ANSI_TO_START_OF_ROW, static_cast<unsigned>(m_terminal_height));
+  out << ANSI_CLEAR_LINE;
+  out << ANSI_RESTORE_CURSOR;
+}
+
+void Statusline::Clear() { Draw(""); }
+
+void Statusline::UpdateTerminalProperties() {
+  if (m_terminal_size_has_changed == 0)
+    return;
+
+  // Clear the previous statusline.
+  Reset();
+
+  // Purposely ignore the terminal settings. If the setting doesn't match
+  // reality and we draw the status bar over existing text, we have no way to
+  // recover.
+  struct winsize window_size;
+  if ((isatty(STDIN_FILENO) != 0) &&
+      ::ioctl(STDIN_FILENO, TIOCGWINSZ, &window_size) == 0) {
+    m_terminal_width = window_size.ws_col;
+    m_terminal_height = window_size.ws_row;
+  }
+
+  // Set the scroll window based on the new terminal height.
+  SetScrollWindow(m_terminal_height - 1);
+
+  // Clear the flag.
+  m_terminal_size_has_changed = 0;
+}
+
+void Statusline::SetScrollWindow(uint64_t height) {
+  StreamFile &out = m_debugger.GetOutputStream();
+  out << '\n';
+  out << ANSI_SAVE_CURSOR;
+  out.Printf(ANSI_SET_SCROLL_ROWS, static_cast<unsigned>(height));
+  out << ANSI_RESTORE_CURSOR;
+  out.Printf(ANSI_UP_ROWS, 1);
+  out << ANSI_CLEAR_BELOW;
+  out.Flush();
+
+  m_scroll_height = height;
+}
+
+lldb::thread_result_t Statusline::StatuslineThread() {
+  using namespace std::chrono_literals;
+  static constexpr const std::chrono::milliseconds g_refresh_rate = 100ms;
+
+  bool exit = false;
+  std::optional<ProgressReport> progress_report;
+
+  while (!exit) {
+    std::unique_lock<std::mutex> lock(m_statusline_mutex);
+    if (!m_statusline_cv.wait_for(lock, g_refresh_rate,
+                                  [&]() { return m_statusline_thread_exit; })) {
+      // We hit the timeout. First check if we're asked to exit.
+      if (m_statusline_thread_exit) {
+        exit = true;
+        continue;
+      }
+
+      StreamString stream;
+      ExecutionContext exe_ctx =
+          m_debugger.GetCommandInterpreter().GetExecutionContext();
+      SymbolContext symbol_ctx;
+      if (auto frame_sp = exe_ctx.GetFrameSP())
+        symbol_ctx = frame_sp->GetSymbolContext(eSymbolContextEverything);
+
+      // Add the user-configured components.
+      bool add_separator = false;
+      for (const FormatEntity::Entry& entry : m_debugger.GetStatuslineFormat()) {
+        if (add_separator)
+          stream << " | ";
+        add_separator =
+            FormatEntity::Format(entry, stream, &symbol_ctx, &exe_ctx, nullptr,
+                                 nullptr, false, false);
+      }
+
+      // Add progress reports at the end, if enabled.
+      if (m_debugger.GetShowProgress()) {
+        if (m_progress_reports.empty()) {
+          progress_report.reset();
+        } else {
+          progress_report.emplace(m_progress_reports.back());
+        }
+        if (progress_report) {
+          if (add_separator)
+            stream << " | ";
+          stream << progress_report->message;
+        }
+      }
+
+      Draw(stream.GetString());
+    } else {
+      // We got notified and the predicate passed. First check if we're asked to
+      // exit.
+      if (m_statusline_thread_exit) {
+        exit = true;
+        continue;
+      }
+    }
+  }
+
+  return {};
+}
+
+bool Statusline::StartStatuslineThread() {
+  Enable();
+  if (!m_statusline_thread.IsJoinable()) {
+    m_statusline_thread_exit = false;
+    llvm::Expected<HostThread> statusline_thread = ThreadLauncher::LaunchThread(
+        "lldb.debugger.statusline", [this] { return StatuslineThread(); });
+
+    if (statusline_thread) {
+      m_statusline_thread = *statusline_thread;
+    } else {
+      LLDB_LOG_ERROR(GetLog(LLDBLog::Host), statusline_thread.takeError(),
+                     "failed to launch host thread: {0}");
+    }
+  }
+  return m_statusline_thread.IsJoinable();
+}
+
+void Statusline::StopStatuslineThread() {
+  if (m_statusline_thread.IsJoinable()) {
+    {
+      std::lock_guard<std::mutex> guard(m_statusline_mutex);
+      m_statusline_thread_exit = true;
+    }
+    m_statusline_cv.notify_one();
+    m_statusline_thread.Join(nullptr);
+  }
+  Disable();
+}
+
+void Statusline::ReportProgress(const ProgressEventData &data) {
+  // Make a local copy of the incoming progress report, which might get modified
+  // below.
+  ProgressReport progress_report{
+      data.GetID(), data.IsFinite()
+                        ? llvm::formatv("[{0}/{1}] {2}", data.GetCompleted(),
+                                        data.GetTotal(), data.GetMessage())
+                              .str()
+                        : data.GetMessage()};
+
+  std::lock_guard<std::mutex> guard(m_statusline_mutex);
+
+  // Do some bookkeeping regardless of whether we're going to display
+  // progress reports.
+  auto it = std::find_if(
+      m_progress_reports.begin(), m_progress_reports.end(),
+      [&](const auto &report) { return report.id == progress_report.id; });
+  if (it != m_progress_reports.end()) {
+    const bool complete = data.GetCompleted() == data.GetTotal();
+    if (complete)
+      m_progress_reports.erase(it);
+    else
+      *it = progress_report;
+  } else {
+    m_progress_reports.push_back(progress_report);
+  }
+}
diff --git a/lldb/source/Interpreter/CommandInterpreter.cpp b/lldb/source/Interpreter/CommandInterpreter.cpp
index 764dcfd1903b19..57b7efa0dc705f 100644
--- a/lldb/source/Interpreter/CommandInterpreter.cpp
+++ b/lldb/source/Interpreter/CommandInterpreter.cpp
@@ -3447,8 +3447,10 @@ CommandInterpreterRunResult CommandInterpreter::RunCommandInterpreter(
   m_debugger.RunIOHandlerAsync(GetIOHandler(force_create, &options));
   m_result = CommandInterpreterRunResult();
 
-  if (options.GetAutoHandleEvents())
+  if (options.GetAutoHandleEvents()) {
     m_debugger.StartEventHandlerThread();
+    m_debugger.ShowStatusline();
+  }
 
   if (options.GetSpawnThread()) {
     m_debugger.StartIOHandlerThread();
@@ -3461,8 +3463,10 @@ CommandInterpreterRunResult CommandInterpreter::RunCommandInterpreter(
     m_debugger.RunIOHandlers();
     m_debugger.SetIOHandlerThread(old_io_handler_thread);
 
-    if (options.GetAutoHandleEvents())
+    if (options.GetAutoHandleEvents()) {
       m_debugger.StopEventHandlerThread();
+      m_debugger.HideStatusline();
+    }
   }
 
   return m_result;



More information about the lldb-commits mailing list