144d93782SGreg Clayton //===-- IOHandler.cpp -------------------------------------------*- C++ -*-===//
244d93782SGreg Clayton //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
644d93782SGreg Clayton //
744d93782SGreg Clayton //===----------------------------------------------------------------------===//
844d93782SGreg Clayton 
92f3df613SZachary Turner #include "lldb/Core/IOHandler.h"
102f3df613SZachary Turner 
11315b6884SEugene Zelenko #ifndef LLDB_DISABLE_CURSES
1227801f4fSBruce Mitchener #include <curses.h>
13315b6884SEugene Zelenko #include <panel.h>
14315b6884SEugene Zelenko #endif
1544d93782SGreg Clayton 
167c9aa073STodd Fiala #if defined(__APPLE__)
177c9aa073STodd Fiala #include <deque>
187c9aa073STodd Fiala #endif
1944d93782SGreg Clayton #include <string>
2044d93782SGreg Clayton 
2144d93782SGreg Clayton #include "lldb/Core/Debugger.h"
2244d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
23672d2c12SJonas Devlieghere #include "lldb/Host/File.h"
24672d2c12SJonas Devlieghere #include "lldb/Utility/Predicate.h"
25672d2c12SJonas Devlieghere #include "lldb/Utility/Status.h"
26672d2c12SJonas Devlieghere #include "lldb/Utility/StreamString.h"
27672d2c12SJonas Devlieghere #include "lldb/Utility/StringList.h"
28672d2c12SJonas Devlieghere #include "lldb/lldb-forward.h"
292f3df613SZachary Turner 
30cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
3144d93782SGreg Clayton #include "lldb/Host/Editline.h"
32cacde7dfSTodd Fiala #endif
3344d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
3444d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
352f3df613SZachary Turner #ifndef LLDB_DISABLE_CURSES
362f3df613SZachary Turner #include "lldb/Breakpoint/BreakpointLocation.h"
372f3df613SZachary Turner #include "lldb/Core/Module.h"
382f3df613SZachary Turner #include "lldb/Core/ValueObject.h"
392f3df613SZachary Turner #include "lldb/Core/ValueObjectRegister.h"
4044d93782SGreg Clayton #include "lldb/Symbol/Block.h"
4144d93782SGreg Clayton #include "lldb/Symbol/Function.h"
4244d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
43c5dac77aSEugene Zelenko #include "lldb/Symbol/VariableList.h"
44c5dac77aSEugene Zelenko #include "lldb/Target/Process.h"
452f3df613SZachary Turner #include "lldb/Target/RegisterContext.h"
46c5dac77aSEugene Zelenko #include "lldb/Target/StackFrame.h"
472f3df613SZachary Turner #include "lldb/Target/StopInfo.h"
48b9c1b51eSKate Stone #include "lldb/Target/Target.h"
49b9c1b51eSKate Stone #include "lldb/Target/Thread.h"
50d821c997SPavel Labath #include "lldb/Utility/State.h"
51c5dac77aSEugene Zelenko #endif
527c9aa073STodd Fiala 
53672d2c12SJonas Devlieghere #include "llvm/ADT/StringRef.h"
542f3df613SZachary Turner 
55fab31220STed Woodward #ifdef _MSC_VER
56aaea8ee6SZachary Turner #include "lldb/Host/windows/windows.h"
57fab31220STed Woodward #endif
58fab31220STed Woodward 
59672d2c12SJonas Devlieghere #include <memory>
60672d2c12SJonas Devlieghere #include <mutex>
612f3df613SZachary Turner 
62672d2c12SJonas Devlieghere #include <assert.h>
63672d2c12SJonas Devlieghere #include <ctype.h>
64672d2c12SJonas Devlieghere #include <errno.h>
65672d2c12SJonas Devlieghere #include <locale.h>
66672d2c12SJonas Devlieghere #include <stdint.h>
67672d2c12SJonas Devlieghere #include <stdio.h>
68672d2c12SJonas Devlieghere #include <string.h>
69672d2c12SJonas Devlieghere #include <type_traits>
702f3df613SZachary Turner 
7144d93782SGreg Clayton using namespace lldb;
7244d93782SGreg Clayton using namespace lldb_private;
7344d93782SGreg Clayton 
74b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type)
75b9c1b51eSKate Stone     : IOHandler(debugger, type,
7644d93782SGreg Clayton                 StreamFileSP(), // Adopt STDIN from top input reader
7744d93782SGreg Clayton                 StreamFileSP(), // Adopt STDOUT from top input reader
78340b0309SGreg Clayton                 StreamFileSP(), // Adopt STDERR from top input reader
79d77c2e09SJonas Devlieghere                 0,              // Flags
80d77c2e09SJonas Devlieghere                 nullptr         // Shadow file recorder
81d77c2e09SJonas Devlieghere       ) {}
8244d93782SGreg Clayton 
83b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type,
8444d93782SGreg Clayton                      const lldb::StreamFileSP &input_sp,
8544d93782SGreg Clayton                      const lldb::StreamFileSP &output_sp,
86d77c2e09SJonas Devlieghere                      const lldb::StreamFileSP &error_sp, uint32_t flags,
87d77c2e09SJonas Devlieghere                      repro::DataRecorder *data_recorder)
88b9c1b51eSKate Stone     : m_debugger(debugger), m_input_sp(input_sp), m_output_sp(output_sp),
89d77c2e09SJonas Devlieghere       m_error_sp(error_sp), m_data_recorder(data_recorder), m_popped(false),
90d77c2e09SJonas Devlieghere       m_flags(flags), m_type(type), m_user_data(nullptr), m_done(false),
91d77c2e09SJonas Devlieghere       m_active(false) {
9244d93782SGreg Clayton   // If any files are not specified, then adopt them from the top input reader.
9344d93782SGreg Clayton   if (!m_input_sp || !m_output_sp || !m_error_sp)
94b9c1b51eSKate Stone     debugger.AdoptTopIOHandlerFilesIfInvalid(m_input_sp, m_output_sp,
9544d93782SGreg Clayton                                              m_error_sp);
9644d93782SGreg Clayton }
9744d93782SGreg Clayton 
98315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
9944d93782SGreg Clayton 
100b9c1b51eSKate Stone int IOHandler::GetInputFD() {
101c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetDescriptor() : -1);
10244d93782SGreg Clayton }
10344d93782SGreg Clayton 
104b9c1b51eSKate Stone int IOHandler::GetOutputFD() {
105c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetDescriptor() : -1);
10644d93782SGreg Clayton }
10744d93782SGreg Clayton 
108b9c1b51eSKate Stone int IOHandler::GetErrorFD() {
109c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetDescriptor() : -1);
11044d93782SGreg Clayton }
11144d93782SGreg Clayton 
112b9c1b51eSKate Stone FILE *IOHandler::GetInputFILE() {
113c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetStream() : nullptr);
11444d93782SGreg Clayton }
11544d93782SGreg Clayton 
116b9c1b51eSKate Stone FILE *IOHandler::GetOutputFILE() {
117c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetStream() : nullptr);
11844d93782SGreg Clayton }
11944d93782SGreg Clayton 
120b9c1b51eSKate Stone FILE *IOHandler::GetErrorFILE() {
121c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetStream() : nullptr);
12244d93782SGreg Clayton }
12344d93782SGreg Clayton 
124b9c1b51eSKate Stone StreamFileSP &IOHandler::GetInputStreamFile() { return m_input_sp; }
12544d93782SGreg Clayton 
126b9c1b51eSKate Stone StreamFileSP &IOHandler::GetOutputStreamFile() { return m_output_sp; }
12744d93782SGreg Clayton 
128b9c1b51eSKate Stone StreamFileSP &IOHandler::GetErrorStreamFile() { return m_error_sp; }
12944d93782SGreg Clayton 
130b9c1b51eSKate Stone bool IOHandler::GetIsInteractive() {
131340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsInteractive();
132340b0309SGreg Clayton }
133340b0309SGreg Clayton 
134b9c1b51eSKate Stone bool IOHandler::GetIsRealTerminal() {
135340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsRealTerminal();
136340b0309SGreg Clayton }
13744d93782SGreg Clayton 
138b9c1b51eSKate Stone void IOHandler::SetPopped(bool b) { m_popped.SetValue(b, eBroadcastOnChange); }
139e30f11d9SKate Stone 
140b9c1b51eSKate Stone void IOHandler::WaitForPop() { m_popped.WaitForValueEqualTo(true); }
141e30f11d9SKate Stone 
142b9c1b51eSKate Stone void IOHandlerStack::PrintAsync(Stream *stream, const char *s, size_t len) {
143b9c1b51eSKate Stone   if (stream) {
14416ff8604SSaleem Abdulrasool     std::lock_guard<std::recursive_mutex> guard(m_mutex);
1454446487dSPavel Labath     if (m_top)
1464446487dSPavel Labath       m_top->PrintAsync(stream, s, len);
1474446487dSPavel Labath   }
1484446487dSPavel Labath }
1494446487dSPavel Labath 
1507a120c8bSZachary Turner IOHandlerConfirm::IOHandlerConfirm(Debugger &debugger, llvm::StringRef prompt,
151b9c1b51eSKate Stone                                    bool default_response)
152b9c1b51eSKate Stone     : IOHandlerEditline(
153b9c1b51eSKate Stone           debugger, IOHandler::Type::Confirm,
154c5dac77aSEugene Zelenko           nullptr, // nullptr editline_name means no history loaded/saved
155514d8cd8SZachary Turner           llvm::StringRef(), // No prompt
156514d8cd8SZachary Turner           llvm::StringRef(), // No continuation prompt
15744d93782SGreg Clayton           false,             // Multi-line
158e30f11d9SKate Stone           false, // Don't colorize the prompt (i.e. the confirm message.)
159d77c2e09SJonas Devlieghere           0, *this, nullptr),
160b9c1b51eSKate Stone       m_default_response(default_response), m_user_response(default_response) {
16144d93782SGreg Clayton   StreamString prompt_stream;
16244d93782SGreg Clayton   prompt_stream.PutCString(prompt);
16344d93782SGreg Clayton   if (m_default_response)
16444d93782SGreg Clayton     prompt_stream.Printf(": [Y/n] ");
16544d93782SGreg Clayton   else
16644d93782SGreg Clayton     prompt_stream.Printf(": [y/N] ");
16744d93782SGreg Clayton 
168514d8cd8SZachary Turner   SetPrompt(prompt_stream.GetString());
16944d93782SGreg Clayton }
17044d93782SGreg Clayton 
171315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
17244d93782SGreg Clayton 
1737f88829cSRaphael Isemann int IOHandlerConfirm::IOHandlerComplete(
1747f88829cSRaphael Isemann     IOHandler &io_handler, const char *current_line, const char *cursor,
175*b22860daSJonas Devlieghere     const char *last_char, StringList &matches, StringList &descriptions) {
176b9c1b51eSKate Stone   if (current_line == cursor) {
177b9c1b51eSKate Stone     if (m_default_response) {
17844d93782SGreg Clayton       matches.AppendString("y");
179b9c1b51eSKate Stone     } else {
18044d93782SGreg Clayton       matches.AppendString("n");
18144d93782SGreg Clayton     }
18244d93782SGreg Clayton   }
18344d93782SGreg Clayton   return matches.GetSize();
18444d93782SGreg Clayton }
18544d93782SGreg Clayton 
186b9c1b51eSKate Stone void IOHandlerConfirm::IOHandlerInputComplete(IOHandler &io_handler,
187b9c1b51eSKate Stone                                               std::string &line) {
188b9c1b51eSKate Stone   if (line.empty()) {
18944d93782SGreg Clayton     // User just hit enter, set the response to the default
19044d93782SGreg Clayton     m_user_response = m_default_response;
19144d93782SGreg Clayton     io_handler.SetIsDone(true);
19244d93782SGreg Clayton     return;
19344d93782SGreg Clayton   }
19444d93782SGreg Clayton 
195b9c1b51eSKate Stone   if (line.size() == 1) {
196b9c1b51eSKate Stone     switch (line[0]) {
19744d93782SGreg Clayton     case 'y':
19844d93782SGreg Clayton     case 'Y':
19944d93782SGreg Clayton       m_user_response = true;
20044d93782SGreg Clayton       io_handler.SetIsDone(true);
20144d93782SGreg Clayton       return;
20244d93782SGreg Clayton     case 'n':
20344d93782SGreg Clayton     case 'N':
20444d93782SGreg Clayton       m_user_response = false;
20544d93782SGreg Clayton       io_handler.SetIsDone(true);
20644d93782SGreg Clayton       return;
20744d93782SGreg Clayton     default:
20844d93782SGreg Clayton       break;
20944d93782SGreg Clayton     }
21044d93782SGreg Clayton   }
21144d93782SGreg Clayton 
212b9c1b51eSKate Stone   if (line == "yes" || line == "YES" || line == "Yes") {
21344d93782SGreg Clayton     m_user_response = true;
21444d93782SGreg Clayton     io_handler.SetIsDone(true);
215b9c1b51eSKate Stone   } else if (line == "no" || line == "NO" || line == "No") {
21644d93782SGreg Clayton     m_user_response = false;
21744d93782SGreg Clayton     io_handler.SetIsDone(true);
21844d93782SGreg Clayton   }
21944d93782SGreg Clayton }
22044d93782SGreg Clayton 
2217f88829cSRaphael Isemann int IOHandlerDelegate::IOHandlerComplete(
2227f88829cSRaphael Isemann     IOHandler &io_handler, const char *current_line, const char *cursor,
223*b22860daSJonas Devlieghere     const char *last_char, StringList &matches, StringList &descriptions) {
224b9c1b51eSKate Stone   switch (m_completion) {
22544d93782SGreg Clayton   case Completion::None:
22644d93782SGreg Clayton     break;
22744d93782SGreg Clayton 
22844d93782SGreg Clayton   case Completion::LLDBCommand:
229b9c1b51eSKate Stone     return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion(
230*b22860daSJonas Devlieghere         current_line, cursor, last_char, matches, descriptions);
231b9c1b51eSKate Stone   case Completion::Expression: {
2327f88829cSRaphael Isemann     CompletionResult result;
233*b22860daSJonas Devlieghere     CompletionRequest request(current_line, cursor - current_line, result);
234b9c1b51eSKate Stone     CommandCompletions::InvokeCommonCompletionCallbacks(
235b9c1b51eSKate Stone         io_handler.GetDebugger().GetCommandInterpreter(),
236a2e76c0bSRaphael Isemann         CommandCompletions::eVariablePathCompletion, request, nullptr);
2377f88829cSRaphael Isemann     result.GetMatches(matches);
2387f88829cSRaphael Isemann     result.GetDescriptions(descriptions);
23944d93782SGreg Clayton 
2401a6d7ab5SRaphael Isemann     size_t num_matches = request.GetNumberOfMatches();
241b9c1b51eSKate Stone     if (num_matches > 0) {
242175f0930SJonas Devlieghere       std::string common_prefix = matches.LongestCommonPrefix();
243a2e76c0bSRaphael Isemann       const size_t partial_name_len = request.GetCursorArgumentPrefix().size();
24444d93782SGreg Clayton 
24505097246SAdrian Prantl       // If we matched a unique single command, add a space... Only do this if
24605097246SAdrian Prantl       // the completer told us this was a complete word, however...
247a2e76c0bSRaphael Isemann       if (num_matches == 1 && request.GetWordComplete()) {
24844d93782SGreg Clayton         common_prefix.push_back(' ');
24944d93782SGreg Clayton       }
25044d93782SGreg Clayton       common_prefix.erase(0, partial_name_len);
25144d93782SGreg Clayton       matches.InsertStringAtIndex(0, std::move(common_prefix));
25244d93782SGreg Clayton     }
25344d93782SGreg Clayton     return num_matches;
254b9c1b51eSKate Stone   } break;
25544d93782SGreg Clayton   }
25644d93782SGreg Clayton 
25744d93782SGreg Clayton   return 0;
25844d93782SGreg Clayton }
25944d93782SGreg Clayton 
260b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
261b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
26244d93782SGreg Clayton     const char *editline_name, // Used for saving history files
263514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
264514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
265d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
266b9c1b51eSKate Stone     : IOHandlerEditline(debugger, type,
26744d93782SGreg Clayton                         StreamFileSP(), // Inherit input from top input reader
26844d93782SGreg Clayton                         StreamFileSP(), // Inherit output from top input reader
26944d93782SGreg Clayton                         StreamFileSP(), // Inherit error from top input reader
270340b0309SGreg Clayton                         0,              // Flags
27144d93782SGreg Clayton                         editline_name,  // Used for saving history files
272b9c1b51eSKate Stone                         prompt, continuation_prompt, multi_line, color_prompts,
273d77c2e09SJonas Devlieghere                         line_number_start, delegate, data_recorder) {}
27444d93782SGreg Clayton 
275b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
276b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
277b9c1b51eSKate Stone     const lldb::StreamFileSP &input_sp, const lldb::StreamFileSP &output_sp,
278b9c1b51eSKate Stone     const lldb::StreamFileSP &error_sp, uint32_t flags,
27944d93782SGreg Clayton     const char *editline_name, // Used for saving history files
280514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
281514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
282d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
283d77c2e09SJonas Devlieghere     : IOHandler(debugger, type, input_sp, output_sp, error_sp, flags,
284d77c2e09SJonas Devlieghere                 data_recorder),
285cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
286d5b44036SJonas Devlieghere       m_editline_up(),
287cacde7dfSTodd Fiala #endif
288b9c1b51eSKate Stone       m_delegate(delegate), m_prompt(), m_continuation_prompt(),
289b9c1b51eSKate Stone       m_current_lines_ptr(nullptr), m_base_line_number(line_number_start),
290b9c1b51eSKate Stone       m_curr_line_idx(UINT32_MAX), m_multi_line(multi_line),
291b9c1b51eSKate Stone       m_color_prompts(color_prompts), m_interrupt_exits(true),
292b9c1b51eSKate Stone       m_editing(false) {
29344d93782SGreg Clayton   SetPrompt(prompt);
29444d93782SGreg Clayton 
295cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
296914b8d98SDeepak Panickal   bool use_editline = false;
297340b0309SGreg Clayton 
298340b0309SGreg Clayton   use_editline = m_input_sp->GetFile().GetIsRealTerminal();
29944d93782SGreg Clayton 
300b9c1b51eSKate Stone   if (use_editline) {
301d5b44036SJonas Devlieghere     m_editline_up.reset(new Editline(editline_name, GetInputFILE(),
302b9c1b51eSKate Stone                                      GetOutputFILE(), GetErrorFILE(),
303e30f11d9SKate Stone                                      m_color_prompts));
304d5b44036SJonas Devlieghere     m_editline_up->SetIsInputCompleteCallback(IsInputCompleteCallback, this);
305d5b44036SJonas Devlieghere     m_editline_up->SetAutoCompleteCallback(AutoCompleteCallback, this);
306e30f11d9SKate Stone     // See if the delegate supports fixing indentation
307e30f11d9SKate Stone     const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
308b9c1b51eSKate Stone     if (indent_chars) {
309b9c1b51eSKate Stone       // The delegate does support indentation, hook it up so when any
31005097246SAdrian Prantl       // indentation character is typed, the delegate gets a chance to fix it
311d5b44036SJonas Devlieghere       m_editline_up->SetFixIndentationCallback(FixIndentationCallback, this,
312b9c1b51eSKate Stone                                                indent_chars);
313e30f11d9SKate Stone     }
31444d93782SGreg Clayton   }
315cacde7dfSTodd Fiala #endif
316e30f11d9SKate Stone   SetBaseLineNumber(m_base_line_number);
317514d8cd8SZachary Turner   SetPrompt(prompt);
318e30f11d9SKate Stone   SetContinuationPrompt(continuation_prompt);
31944d93782SGreg Clayton }
32044d93782SGreg Clayton 
321b9c1b51eSKate Stone IOHandlerEditline::~IOHandlerEditline() {
322cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
323d5b44036SJonas Devlieghere   m_editline_up.reset();
324cacde7dfSTodd Fiala #endif
32544d93782SGreg Clayton }
32644d93782SGreg Clayton 
327b9c1b51eSKate Stone void IOHandlerEditline::Activate() {
328e30f11d9SKate Stone   IOHandler::Activate();
3290affb582SDave Lee   m_delegate.IOHandlerActivated(*this, GetIsInteractive());
330e30f11d9SKate Stone }
331e30f11d9SKate Stone 
332b9c1b51eSKate Stone void IOHandlerEditline::Deactivate() {
333e30f11d9SKate Stone   IOHandler::Deactivate();
334e30f11d9SKate Stone   m_delegate.IOHandlerDeactivated(*this);
335e30f11d9SKate Stone }
336e30f11d9SKate Stone 
337b9c1b51eSKate Stone bool IOHandlerEditline::GetLine(std::string &line, bool &interrupted) {
338cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
339d5b44036SJonas Devlieghere   if (m_editline_up) {
340d77c2e09SJonas Devlieghere     bool b = m_editline_up->GetLine(line, interrupted);
341d77c2e09SJonas Devlieghere     if (m_data_recorder)
342d77c2e09SJonas Devlieghere       m_data_recorder->Record(line, true);
343d77c2e09SJonas Devlieghere     return b;
344b9c1b51eSKate Stone   } else {
345cacde7dfSTodd Fiala #endif
34644d93782SGreg Clayton     line.clear();
34744d93782SGreg Clayton 
34844d93782SGreg Clayton     FILE *in = GetInputFILE();
349b9c1b51eSKate Stone     if (in) {
350b9c1b51eSKate Stone       if (GetIsInteractive()) {
351c5dac77aSEugene Zelenko         const char *prompt = nullptr;
352e30f11d9SKate Stone 
353e30f11d9SKate Stone         if (m_multi_line && m_curr_line_idx > 0)
354e30f11d9SKate Stone           prompt = GetContinuationPrompt();
355e30f11d9SKate Stone 
356c5dac77aSEugene Zelenko         if (prompt == nullptr)
357e30f11d9SKate Stone           prompt = GetPrompt();
358e30f11d9SKate Stone 
359b9c1b51eSKate Stone         if (prompt && prompt[0]) {
36044d93782SGreg Clayton           FILE *out = GetOutputFILE();
361b9c1b51eSKate Stone           if (out) {
36244d93782SGreg Clayton             ::fprintf(out, "%s", prompt);
36344d93782SGreg Clayton             ::fflush(out);
36444d93782SGreg Clayton           }
36544d93782SGreg Clayton         }
36644d93782SGreg Clayton       }
36744d93782SGreg Clayton       char buffer[256];
36844d93782SGreg Clayton       bool done = false;
3690f86e6e7SGreg Clayton       bool got_line = false;
370e034a04eSGreg Clayton       m_editing = true;
371b9c1b51eSKate Stone       while (!done) {
372e7167908SNathan Lanza #ifdef _WIN32
373e7167908SNathan Lanza         // ReadFile on Windows is supposed to set ERROR_OPERATION_ABORTED
374e7167908SNathan Lanza         // according to the docs on MSDN. However, this has evidently been a
375e7167908SNathan Lanza         // known bug since Windows 8. Therefore, we can't detect if a signal
376e7167908SNathan Lanza         // interrupted in the fgets. So pressing ctrl-c causes the repl to end
377e7167908SNathan Lanza         // and the process to exit. A temporary workaround is just to attempt to
378e7167908SNathan Lanza         // fgets twice until this bug is fixed.
379e7167908SNathan Lanza         if (fgets(buffer, sizeof(buffer), in) == nullptr &&
380e7167908SNathan Lanza             fgets(buffer, sizeof(buffer), in) == nullptr) {
381cb305205SNathan Lanza           // this is the equivalent of EINTR for Windows
382cb305205SNathan Lanza           if (GetLastError() == ERROR_OPERATION_ABORTED)
383cb305205SNathan Lanza             continue;
384e7167908SNathan Lanza #else
385b9c1b51eSKate Stone         if (fgets(buffer, sizeof(buffer), in) == nullptr) {
386e7167908SNathan Lanza #endif
387c7797accSGreg Clayton           const int saved_errno = errno;
388c9cf5798SGreg Clayton           if (feof(in))
38944d93782SGreg Clayton             done = true;
390b9c1b51eSKate Stone           else if (ferror(in)) {
391c7797accSGreg Clayton             if (saved_errno != EINTR)
392c7797accSGreg Clayton               done = true;
393c7797accSGreg Clayton           }
394b9c1b51eSKate Stone         } else {
3950f86e6e7SGreg Clayton           got_line = true;
39644d93782SGreg Clayton           size_t buffer_len = strlen(buffer);
39744d93782SGreg Clayton           assert(buffer[buffer_len] == '\0');
39844d93782SGreg Clayton           char last_char = buffer[buffer_len - 1];
399b9c1b51eSKate Stone           if (last_char == '\r' || last_char == '\n') {
40044d93782SGreg Clayton             done = true;
40144d93782SGreg Clayton             // Strip trailing newlines
402b9c1b51eSKate Stone             while (last_char == '\r' || last_char == '\n') {
40344d93782SGreg Clayton               --buffer_len;
40444d93782SGreg Clayton               if (buffer_len == 0)
40544d93782SGreg Clayton                 break;
40644d93782SGreg Clayton               last_char = buffer[buffer_len - 1];
40744d93782SGreg Clayton             }
40844d93782SGreg Clayton           }
40944d93782SGreg Clayton           line.append(buffer, buffer_len);
41044d93782SGreg Clayton         }
41144d93782SGreg Clayton       }
412e034a04eSGreg Clayton       m_editing = false;
413d77c2e09SJonas Devlieghere       if (m_data_recorder && got_line)
414d77c2e09SJonas Devlieghere         m_data_recorder->Record(line, true);
41505097246SAdrian Prantl       // We might have gotten a newline on a line by itself make sure to return
41605097246SAdrian Prantl       // true in this case.
4170f86e6e7SGreg Clayton       return got_line;
418b9c1b51eSKate Stone     } else {
41944d93782SGreg Clayton       // No more input file, we are done...
42044d93782SGreg Clayton       SetIsDone(true);
42144d93782SGreg Clayton     }
422340b0309SGreg Clayton     return false;
423cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
42444d93782SGreg Clayton   }
425cacde7dfSTodd Fiala #endif
42644d93782SGreg Clayton }
42744d93782SGreg Clayton 
428cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
429b9c1b51eSKate Stone bool IOHandlerEditline::IsInputCompleteCallback(Editline *editline,
43044d93782SGreg Clayton                                                 StringList &lines,
431b9c1b51eSKate Stone                                                 void *baton) {
43244d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
433b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader,
434b9c1b51eSKate Stone                                                               lines);
435e30f11d9SKate Stone }
436e30f11d9SKate Stone 
437b9c1b51eSKate Stone int IOHandlerEditline::FixIndentationCallback(Editline *editline,
438e30f11d9SKate Stone                                               const StringList &lines,
439e30f11d9SKate Stone                                               int cursor_position,
440b9c1b51eSKate Stone                                               void *baton) {
441e30f11d9SKate Stone   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
442b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerFixIndentation(
443b9c1b51eSKate Stone       *editline_reader, lines, cursor_position);
44444d93782SGreg Clayton }
44544d93782SGreg Clayton 
4467f88829cSRaphael Isemann int IOHandlerEditline::AutoCompleteCallback(
4477f88829cSRaphael Isemann     const char *current_line, const char *cursor, const char *last_char,
448*b22860daSJonas Devlieghere     StringList &matches, StringList &descriptions, void *baton) {
44944d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
45044d93782SGreg Clayton   if (editline_reader)
451b9c1b51eSKate Stone     return editline_reader->m_delegate.IOHandlerComplete(
452*b22860daSJonas Devlieghere         *editline_reader, current_line, cursor, last_char, matches,
453*b22860daSJonas Devlieghere         descriptions);
45444d93782SGreg Clayton   return 0;
45544d93782SGreg Clayton }
456cacde7dfSTodd Fiala #endif
45744d93782SGreg Clayton 
458b9c1b51eSKate Stone const char *IOHandlerEditline::GetPrompt() {
459cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
460d5b44036SJonas Devlieghere   if (m_editline_up) {
461d5b44036SJonas Devlieghere     return m_editline_up->GetPrompt();
462b9c1b51eSKate Stone   } else {
463cacde7dfSTodd Fiala #endif
464cacde7dfSTodd Fiala     if (m_prompt.empty())
465c5dac77aSEugene Zelenko       return nullptr;
466cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
467cacde7dfSTodd Fiala   }
468cacde7dfSTodd Fiala #endif
46944d93782SGreg Clayton   return m_prompt.c_str();
47044d93782SGreg Clayton }
47144d93782SGreg Clayton 
472514d8cd8SZachary Turner bool IOHandlerEditline::SetPrompt(llvm::StringRef prompt) {
473514d8cd8SZachary Turner   m_prompt = prompt;
474514d8cd8SZachary Turner 
475cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
476d5b44036SJonas Devlieghere   if (m_editline_up)
477d5b44036SJonas Devlieghere     m_editline_up->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
478cacde7dfSTodd Fiala #endif
47944d93782SGreg Clayton   return true;
48044d93782SGreg Clayton }
48144d93782SGreg Clayton 
482b9c1b51eSKate Stone const char *IOHandlerEditline::GetContinuationPrompt() {
483b9c1b51eSKate Stone   return (m_continuation_prompt.empty() ? nullptr
484b9c1b51eSKate Stone                                         : m_continuation_prompt.c_str());
485e30f11d9SKate Stone }
486e30f11d9SKate Stone 
487514d8cd8SZachary Turner void IOHandlerEditline::SetContinuationPrompt(llvm::StringRef prompt) {
488514d8cd8SZachary Turner   m_continuation_prompt = prompt;
489e30f11d9SKate Stone 
490d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
491d5b44036SJonas Devlieghere   if (m_editline_up)
492d5b44036SJonas Devlieghere     m_editline_up->SetContinuationPrompt(m_continuation_prompt.empty()
493b9c1b51eSKate Stone                                              ? nullptr
494b9c1b51eSKate Stone                                              : m_continuation_prompt.c_str());
495d553d00cSZachary Turner #endif
496e30f11d9SKate Stone }
497e30f11d9SKate Stone 
498b9c1b51eSKate Stone void IOHandlerEditline::SetBaseLineNumber(uint32_t line) {
499f6913cd7SGreg Clayton   m_base_line_number = line;
500f6913cd7SGreg Clayton }
501e30f11d9SKate Stone 
502b9c1b51eSKate Stone uint32_t IOHandlerEditline::GetCurrentLineIndex() const {
503d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
504d5b44036SJonas Devlieghere   if (m_editline_up)
505d5b44036SJonas Devlieghere     return m_editline_up->GetCurrentLine();
506e30f11d9SKate Stone #endif
507e30f11d9SKate Stone   return m_curr_line_idx;
508e30f11d9SKate Stone }
509e30f11d9SKate Stone 
510b9c1b51eSKate Stone bool IOHandlerEditline::GetLines(StringList &lines, bool &interrupted) {
511e30f11d9SKate Stone   m_current_lines_ptr = &lines;
512e30f11d9SKate Stone 
51344d93782SGreg Clayton   bool success = false;
514cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
515d5b44036SJonas Devlieghere   if (m_editline_up) {
516d5b44036SJonas Devlieghere     return m_editline_up->GetLines(m_base_line_number, lines, interrupted);
517b9c1b51eSKate Stone   } else {
518cacde7dfSTodd Fiala #endif
519e30f11d9SKate Stone     bool done = false;
52097206d57SZachary Turner     Status error;
52144d93782SGreg Clayton 
522b9c1b51eSKate Stone     while (!done) {
523f6913cd7SGreg Clayton       // Show line numbers if we are asked to
52444d93782SGreg Clayton       std::string line;
525b9c1b51eSKate Stone       if (m_base_line_number > 0 && GetIsInteractive()) {
526f6913cd7SGreg Clayton         FILE *out = GetOutputFILE();
527f6913cd7SGreg Clayton         if (out)
528b9c1b51eSKate Stone           ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(),
529b9c1b51eSKate Stone                     GetPrompt() == nullptr ? " " : "");
530f6913cd7SGreg Clayton       }
531f6913cd7SGreg Clayton 
532e30f11d9SKate Stone       m_curr_line_idx = lines.GetSize();
533e30f11d9SKate Stone 
534f0066ad0SGreg Clayton       bool interrupted = false;
535b9c1b51eSKate Stone       if (GetLine(line, interrupted) && !interrupted) {
53644d93782SGreg Clayton         lines.AppendString(line);
537e30f11d9SKate Stone         done = m_delegate.IOHandlerIsInputComplete(*this, lines);
538b9c1b51eSKate Stone       } else {
539e30f11d9SKate Stone         done = true;
54044d93782SGreg Clayton       }
54144d93782SGreg Clayton     }
54244d93782SGreg Clayton     success = lines.GetSize() > 0;
543cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
54444d93782SGreg Clayton   }
545cacde7dfSTodd Fiala #endif
54644d93782SGreg Clayton   return success;
54744d93782SGreg Clayton }
54844d93782SGreg Clayton 
54905097246SAdrian Prantl // Each IOHandler gets to run until it is done. It should read data from the
55005097246SAdrian Prantl // "in" and place output into "out" and "err and return when done.
551b9c1b51eSKate Stone void IOHandlerEditline::Run() {
55244d93782SGreg Clayton   std::string line;
553b9c1b51eSKate Stone   while (IsActive()) {
554f0066ad0SGreg Clayton     bool interrupted = false;
555b9c1b51eSKate Stone     if (m_multi_line) {
55644d93782SGreg Clayton       StringList lines;
557b9c1b51eSKate Stone       if (GetLines(lines, interrupted)) {
558b9c1b51eSKate Stone         if (interrupted) {
559e30f11d9SKate Stone           m_done = m_interrupt_exits;
560e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
561e30f11d9SKate Stone 
562b9c1b51eSKate Stone         } else {
56344d93782SGreg Clayton           line = lines.CopyList();
56444d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
56544d93782SGreg Clayton         }
566b9c1b51eSKate Stone       } else {
56744d93782SGreg Clayton         m_done = true;
56844d93782SGreg Clayton       }
569b9c1b51eSKate Stone     } else {
570b9c1b51eSKate Stone       if (GetLine(line, interrupted)) {
571e30f11d9SKate Stone         if (interrupted)
572e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
573e30f11d9SKate Stone         else
57444d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
575b9c1b51eSKate Stone       } else {
57644d93782SGreg Clayton         m_done = true;
57744d93782SGreg Clayton       }
57844d93782SGreg Clayton     }
57944d93782SGreg Clayton   }
58044d93782SGreg Clayton }
58144d93782SGreg Clayton 
582b9c1b51eSKate Stone void IOHandlerEditline::Cancel() {
583cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
584d5b44036SJonas Devlieghere   if (m_editline_up)
585d5b44036SJonas Devlieghere     m_editline_up->Cancel();
586cacde7dfSTodd Fiala #endif
587e68f5d6bSGreg Clayton }
588e68f5d6bSGreg Clayton 
589b9c1b51eSKate Stone bool IOHandlerEditline::Interrupt() {
590f0066ad0SGreg Clayton   // Let the delgate handle it first
591f0066ad0SGreg Clayton   if (m_delegate.IOHandlerInterrupt(*this))
592f0066ad0SGreg Clayton     return true;
593f0066ad0SGreg Clayton 
594cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
595d5b44036SJonas Devlieghere   if (m_editline_up)
596d5b44036SJonas Devlieghere     return m_editline_up->Interrupt();
597cacde7dfSTodd Fiala #endif
598f0066ad0SGreg Clayton   return false;
59944d93782SGreg Clayton }
60044d93782SGreg Clayton 
601b9c1b51eSKate Stone void IOHandlerEditline::GotEOF() {
602cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
603d5b44036SJonas Devlieghere   if (m_editline_up)
604d5b44036SJonas Devlieghere     m_editline_up->Interrupt();
605cacde7dfSTodd Fiala #endif
60644d93782SGreg Clayton }
60744d93782SGreg Clayton 
608b9c1b51eSKate Stone void IOHandlerEditline::PrintAsync(Stream *stream, const char *s, size_t len) {
6094446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
610d5b44036SJonas Devlieghere   if (m_editline_up)
611d5b44036SJonas Devlieghere     m_editline_up->PrintAsync(stream, s, len);
6124446487dSPavel Labath   else
6134446487dSPavel Labath #endif
614fab31220STed Woodward   {
615fab31220STed Woodward #ifdef _MSC_VER
616341e4789SDawn Perchik     const char *prompt = GetPrompt();
617b9c1b51eSKate Stone     if (prompt) {
618fab31220STed Woodward       // Back up over previous prompt using Windows API
619fab31220STed Woodward       CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
620fab31220STed Woodward       HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
621fab31220STed Woodward       GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
622fab31220STed Woodward       COORD coord = screen_buffer_info.dwCursorPosition;
623fab31220STed Woodward       coord.X -= strlen(prompt);
624fab31220STed Woodward       if (coord.X < 0)
625fab31220STed Woodward         coord.X = 0;
626fab31220STed Woodward       SetConsoleCursorPosition(console_handle, coord);
627fab31220STed Woodward     }
628fab31220STed Woodward #endif
6294446487dSPavel Labath     IOHandler::PrintAsync(stream, s, len);
630341e4789SDawn Perchik #ifdef _MSC_VER
631fab31220STed Woodward     if (prompt)
632b9c1b51eSKate Stone       IOHandler::PrintAsync(GetOutputStreamFile().get(), prompt,
633b9c1b51eSKate Stone                             strlen(prompt));
634341e4789SDawn Perchik #endif
635fab31220STed Woodward   }
6364446487dSPavel Labath }
6374446487dSPavel Labath 
63805097246SAdrian Prantl // we may want curses to be disabled for some builds for instance, windows
639914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
640914b8d98SDeepak Panickal 
64144d93782SGreg Clayton #define KEY_RETURN 10
64244d93782SGreg Clayton #define KEY_ESCAPE 27
64344d93782SGreg Clayton 
644b9c1b51eSKate Stone namespace curses {
64544d93782SGreg Clayton class Menu;
64644d93782SGreg Clayton class MenuDelegate;
64744d93782SGreg Clayton class Window;
64844d93782SGreg Clayton class WindowDelegate;
64944d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP;
65044d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
65144d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP;
65244d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
65344d93782SGreg Clayton typedef std::vector<MenuSP> Menus;
65444d93782SGreg Clayton typedef std::vector<WindowSP> Windows;
65544d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates;
65644d93782SGreg Clayton 
65744d93782SGreg Clayton #if 0
65844d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
65944d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
66044d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
66144d93782SGreg Clayton #endif
662315b6884SEugene Zelenko 
663b9c1b51eSKate Stone struct Point {
66444d93782SGreg Clayton   int x;
66544d93782SGreg Clayton   int y;
66644d93782SGreg Clayton 
667b9c1b51eSKate Stone   Point(int _x = 0, int _y = 0) : x(_x), y(_y) {}
66844d93782SGreg Clayton 
669b9c1b51eSKate Stone   void Clear() {
67044d93782SGreg Clayton     x = 0;
67144d93782SGreg Clayton     y = 0;
67244d93782SGreg Clayton   }
67344d93782SGreg Clayton 
674b9c1b51eSKate Stone   Point &operator+=(const Point &rhs) {
67544d93782SGreg Clayton     x += rhs.x;
67644d93782SGreg Clayton     y += rhs.y;
67744d93782SGreg Clayton     return *this;
67844d93782SGreg Clayton   }
67944d93782SGreg Clayton 
680b9c1b51eSKate Stone   void Dump() { printf("(x=%i, y=%i)\n", x, y); }
68144d93782SGreg Clayton };
68244d93782SGreg Clayton 
683b9c1b51eSKate Stone bool operator==(const Point &lhs, const Point &rhs) {
68444d93782SGreg Clayton   return lhs.x == rhs.x && lhs.y == rhs.y;
68544d93782SGreg Clayton }
686315b6884SEugene Zelenko 
687b9c1b51eSKate Stone bool operator!=(const Point &lhs, const Point &rhs) {
68844d93782SGreg Clayton   return lhs.x != rhs.x || lhs.y != rhs.y;
68944d93782SGreg Clayton }
69044d93782SGreg Clayton 
691b9c1b51eSKate Stone struct Size {
69244d93782SGreg Clayton   int width;
69344d93782SGreg Clayton   int height;
694b9c1b51eSKate Stone   Size(int w = 0, int h = 0) : width(w), height(h) {}
69544d93782SGreg Clayton 
696b9c1b51eSKate Stone   void Clear() {
69744d93782SGreg Clayton     width = 0;
69844d93782SGreg Clayton     height = 0;
69944d93782SGreg Clayton   }
70044d93782SGreg Clayton 
701b9c1b51eSKate Stone   void Dump() { printf("(w=%i, h=%i)\n", width, height); }
70244d93782SGreg Clayton };
70344d93782SGreg Clayton 
704b9c1b51eSKate Stone bool operator==(const Size &lhs, const Size &rhs) {
70544d93782SGreg Clayton   return lhs.width == rhs.width && lhs.height == rhs.height;
70644d93782SGreg Clayton }
707315b6884SEugene Zelenko 
708b9c1b51eSKate Stone bool operator!=(const Size &lhs, const Size &rhs) {
70944d93782SGreg Clayton   return lhs.width != rhs.width || lhs.height != rhs.height;
71044d93782SGreg Clayton }
71144d93782SGreg Clayton 
712b9c1b51eSKate Stone struct Rect {
71344d93782SGreg Clayton   Point origin;
71444d93782SGreg Clayton   Size size;
71544d93782SGreg Clayton 
716b9c1b51eSKate Stone   Rect() : origin(), size() {}
71744d93782SGreg Clayton 
718b9c1b51eSKate Stone   Rect(const Point &p, const Size &s) : origin(p), size(s) {}
71944d93782SGreg Clayton 
720b9c1b51eSKate Stone   void Clear() {
72144d93782SGreg Clayton     origin.Clear();
72244d93782SGreg Clayton     size.Clear();
72344d93782SGreg Clayton   }
72444d93782SGreg Clayton 
725b9c1b51eSKate Stone   void Dump() {
726b9c1b51eSKate Stone     printf("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width,
727b9c1b51eSKate Stone            size.height);
72844d93782SGreg Clayton   }
72944d93782SGreg Clayton 
730b9c1b51eSKate Stone   void Inset(int w, int h) {
73144d93782SGreg Clayton     if (size.width > w * 2)
73244d93782SGreg Clayton       size.width -= w * 2;
73344d93782SGreg Clayton     origin.x += w;
73444d93782SGreg Clayton 
73544d93782SGreg Clayton     if (size.height > h * 2)
73644d93782SGreg Clayton       size.height -= h * 2;
73744d93782SGreg Clayton     origin.y += h;
73844d93782SGreg Clayton   }
739315b6884SEugene Zelenko 
74005097246SAdrian Prantl   // Return a status bar rectangle which is the last line of this rectangle.
74105097246SAdrian Prantl   // This rectangle will be modified to not include the status bar area.
742b9c1b51eSKate Stone   Rect MakeStatusBar() {
74344d93782SGreg Clayton     Rect status_bar;
744b9c1b51eSKate Stone     if (size.height > 1) {
74544d93782SGreg Clayton       status_bar.origin.x = origin.x;
74644d93782SGreg Clayton       status_bar.origin.y = size.height;
74744d93782SGreg Clayton       status_bar.size.width = size.width;
74844d93782SGreg Clayton       status_bar.size.height = 1;
74944d93782SGreg Clayton       --size.height;
75044d93782SGreg Clayton     }
75144d93782SGreg Clayton     return status_bar;
75244d93782SGreg Clayton   }
75344d93782SGreg Clayton 
75405097246SAdrian Prantl   // Return a menubar rectangle which is the first line of this rectangle. This
75505097246SAdrian Prantl   // rectangle will be modified to not include the menubar area.
756b9c1b51eSKate Stone   Rect MakeMenuBar() {
75744d93782SGreg Clayton     Rect menubar;
758b9c1b51eSKate Stone     if (size.height > 1) {
75944d93782SGreg Clayton       menubar.origin.x = origin.x;
76044d93782SGreg Clayton       menubar.origin.y = origin.y;
76144d93782SGreg Clayton       menubar.size.width = size.width;
76244d93782SGreg Clayton       menubar.size.height = 1;
76344d93782SGreg Clayton       ++origin.y;
76444d93782SGreg Clayton       --size.height;
76544d93782SGreg Clayton     }
76644d93782SGreg Clayton     return menubar;
76744d93782SGreg Clayton   }
76844d93782SGreg Clayton 
769b9c1b51eSKate Stone   void HorizontalSplitPercentage(float top_percentage, Rect &top,
770b9c1b51eSKate Stone                                  Rect &bottom) const {
77144d93782SGreg Clayton     float top_height = top_percentage * size.height;
77244d93782SGreg Clayton     HorizontalSplit(top_height, top, bottom);
77344d93782SGreg Clayton   }
77444d93782SGreg Clayton 
775b9c1b51eSKate Stone   void HorizontalSplit(int top_height, Rect &top, Rect &bottom) const {
77644d93782SGreg Clayton     top = *this;
777b9c1b51eSKate Stone     if (top_height < size.height) {
77844d93782SGreg Clayton       top.size.height = top_height;
77944d93782SGreg Clayton       bottom.origin.x = origin.x;
78044d93782SGreg Clayton       bottom.origin.y = origin.y + top.size.height;
78144d93782SGreg Clayton       bottom.size.width = size.width;
78244d93782SGreg Clayton       bottom.size.height = size.height - top.size.height;
783b9c1b51eSKate Stone     } else {
78444d93782SGreg Clayton       bottom.Clear();
78544d93782SGreg Clayton     }
78644d93782SGreg Clayton   }
78744d93782SGreg Clayton 
788b9c1b51eSKate Stone   void VerticalSplitPercentage(float left_percentage, Rect &left,
789b9c1b51eSKate Stone                                Rect &right) const {
79044d93782SGreg Clayton     float left_width = left_percentage * size.width;
79144d93782SGreg Clayton     VerticalSplit(left_width, left, right);
79244d93782SGreg Clayton   }
79344d93782SGreg Clayton 
794b9c1b51eSKate Stone   void VerticalSplit(int left_width, Rect &left, Rect &right) const {
79544d93782SGreg Clayton     left = *this;
796b9c1b51eSKate Stone     if (left_width < size.width) {
79744d93782SGreg Clayton       left.size.width = left_width;
79844d93782SGreg Clayton       right.origin.x = origin.x + left.size.width;
79944d93782SGreg Clayton       right.origin.y = origin.y;
80044d93782SGreg Clayton       right.size.width = size.width - left.size.width;
80144d93782SGreg Clayton       right.size.height = size.height;
802b9c1b51eSKate Stone     } else {
80344d93782SGreg Clayton       right.Clear();
80444d93782SGreg Clayton     }
80544d93782SGreg Clayton   }
80644d93782SGreg Clayton };
80744d93782SGreg Clayton 
808b9c1b51eSKate Stone bool operator==(const Rect &lhs, const Rect &rhs) {
80944d93782SGreg Clayton   return lhs.origin == rhs.origin && lhs.size == rhs.size;
81044d93782SGreg Clayton }
811315b6884SEugene Zelenko 
812b9c1b51eSKate Stone bool operator!=(const Rect &lhs, const Rect &rhs) {
81344d93782SGreg Clayton   return lhs.origin != rhs.origin || lhs.size != rhs.size;
81444d93782SGreg Clayton }
81544d93782SGreg Clayton 
816b9c1b51eSKate Stone enum HandleCharResult {
81744d93782SGreg Clayton   eKeyNotHandled = 0,
81844d93782SGreg Clayton   eKeyHandled = 1,
81944d93782SGreg Clayton   eQuitApplication = 2
82044d93782SGreg Clayton };
82144d93782SGreg Clayton 
822b9c1b51eSKate Stone enum class MenuActionResult {
82344d93782SGreg Clayton   Handled,
82444d93782SGreg Clayton   NotHandled,
82544d93782SGreg Clayton   Quit // Exit all menus and quit
82644d93782SGreg Clayton };
82744d93782SGreg Clayton 
828b9c1b51eSKate Stone struct KeyHelp {
82944d93782SGreg Clayton   int ch;
83044d93782SGreg Clayton   const char *description;
83144d93782SGreg Clayton };
83244d93782SGreg Clayton 
833b9c1b51eSKate Stone class WindowDelegate {
83444d93782SGreg Clayton public:
835b9c1b51eSKate Stone   virtual ~WindowDelegate() = default;
83644d93782SGreg Clayton 
837b9c1b51eSKate Stone   virtual bool WindowDelegateDraw(Window &window, bool force) {
83844d93782SGreg Clayton     return false; // Drawing not handled
83944d93782SGreg Clayton   }
84044d93782SGreg Clayton 
841b9c1b51eSKate Stone   virtual HandleCharResult WindowDelegateHandleChar(Window &window, int key) {
84244d93782SGreg Clayton     return eKeyNotHandled;
84344d93782SGreg Clayton   }
84444d93782SGreg Clayton 
845b9c1b51eSKate Stone   virtual const char *WindowDelegateGetHelpText() { return nullptr; }
84644d93782SGreg Clayton 
847b9c1b51eSKate Stone   virtual KeyHelp *WindowDelegateGetKeyHelp() { return nullptr; }
84844d93782SGreg Clayton };
84944d93782SGreg Clayton 
850b9c1b51eSKate Stone class HelpDialogDelegate : public WindowDelegate {
85144d93782SGreg Clayton public:
85244d93782SGreg Clayton   HelpDialogDelegate(const char *text, KeyHelp *key_help_array);
85344d93782SGreg Clayton 
854bd5ae6b4SGreg Clayton   ~HelpDialogDelegate() override;
85544d93782SGreg Clayton 
856b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
85744d93782SGreg Clayton 
858b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
85944d93782SGreg Clayton 
860b9c1b51eSKate Stone   size_t GetNumLines() const { return m_text.GetSize(); }
86144d93782SGreg Clayton 
862b9c1b51eSKate Stone   size_t GetMaxLineLength() const { return m_text.GetMaxStringLength(); }
86344d93782SGreg Clayton 
86444d93782SGreg Clayton protected:
86544d93782SGreg Clayton   StringList m_text;
86644d93782SGreg Clayton   int m_first_visible_line;
86744d93782SGreg Clayton };
86844d93782SGreg Clayton 
869b9c1b51eSKate Stone class Window {
87044d93782SGreg Clayton public:
871b9c1b51eSKate Stone   Window(const char *name)
872b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
873b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
874b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(false),
875b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {}
87644d93782SGreg Clayton 
877b9c1b51eSKate Stone   Window(const char *name, WINDOW *w, bool del = true)
878b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
879b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
880b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(del),
881b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
88244d93782SGreg Clayton     if (w)
88344d93782SGreg Clayton       Reset(w);
88444d93782SGreg Clayton   }
88544d93782SGreg Clayton 
886b9c1b51eSKate Stone   Window(const char *name, const Rect &bounds)
887b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_parent(nullptr), m_subwindows(),
888b9c1b51eSKate Stone         m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
889b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(true),
890b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
891b9c1b51eSKate Stone     Reset(::newwin(bounds.size.height, bounds.size.width, bounds.origin.y,
892b9c1b51eSKate Stone                    bounds.origin.y));
89344d93782SGreg Clayton   }
89444d93782SGreg Clayton 
895b9c1b51eSKate Stone   virtual ~Window() {
89644d93782SGreg Clayton     RemoveSubWindows();
89744d93782SGreg Clayton     Reset();
89844d93782SGreg Clayton   }
89944d93782SGreg Clayton 
900b9c1b51eSKate Stone   void Reset(WINDOW *w = nullptr, bool del = true) {
90144d93782SGreg Clayton     if (m_window == w)
90244d93782SGreg Clayton       return;
90344d93782SGreg Clayton 
904b9c1b51eSKate Stone     if (m_panel) {
90544d93782SGreg Clayton       ::del_panel(m_panel);
906c5dac77aSEugene Zelenko       m_panel = nullptr;
90744d93782SGreg Clayton     }
908b9c1b51eSKate Stone     if (m_window && m_delete) {
90944d93782SGreg Clayton       ::delwin(m_window);
910c5dac77aSEugene Zelenko       m_window = nullptr;
91144d93782SGreg Clayton       m_delete = false;
91244d93782SGreg Clayton     }
913b9c1b51eSKate Stone     if (w) {
91444d93782SGreg Clayton       m_window = w;
91544d93782SGreg Clayton       m_panel = ::new_panel(m_window);
91644d93782SGreg Clayton       m_delete = del;
91744d93782SGreg Clayton     }
91844d93782SGreg Clayton   }
91944d93782SGreg Clayton 
92044d93782SGreg Clayton   void AttributeOn(attr_t attr) { ::wattron(m_window, attr); }
92144d93782SGreg Clayton   void AttributeOff(attr_t attr) { ::wattroff(m_window, attr); }
922b9c1b51eSKate Stone   void Box(chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) {
923b9c1b51eSKate Stone     ::box(m_window, v_char, h_char);
924b9c1b51eSKate Stone   }
92544d93782SGreg Clayton   void Clear() { ::wclear(m_window); }
92644d93782SGreg Clayton   void Erase() { ::werase(m_window); }
927b9c1b51eSKate Stone   Rect GetBounds() {
928b9c1b51eSKate Stone     return Rect(GetParentOrigin(), GetSize());
929b9c1b51eSKate Stone   } // Get the rectangle in our parent window
93044d93782SGreg Clayton   int GetChar() { return ::wgetch(m_window); }
93144d93782SGreg Clayton   int GetCursorX() { return getcurx(m_window); }
93244d93782SGreg Clayton   int GetCursorY() { return getcury(m_window); }
933b9c1b51eSKate Stone   Rect GetFrame() {
934b9c1b51eSKate Stone     return Rect(Point(), GetSize());
935b9c1b51eSKate Stone   } // Get our rectangle in our own coordinate system
93644d93782SGreg Clayton   Point GetParentOrigin() { return Point(GetParentX(), GetParentY()); }
93744d93782SGreg Clayton   Size GetSize() { return Size(GetWidth(), GetHeight()); }
93844d93782SGreg Clayton   int GetParentX() { return getparx(m_window); }
93944d93782SGreg Clayton   int GetParentY() { return getpary(m_window); }
94044d93782SGreg Clayton   int GetMaxX() { return getmaxx(m_window); }
94144d93782SGreg Clayton   int GetMaxY() { return getmaxy(m_window); }
94244d93782SGreg Clayton   int GetWidth() { return GetMaxX(); }
94344d93782SGreg Clayton   int GetHeight() { return GetMaxY(); }
94444d93782SGreg Clayton   void MoveCursor(int x, int y) { ::wmove(m_window, y, x); }
94544d93782SGreg Clayton   void MoveWindow(int x, int y) { MoveWindow(Point(x, y)); }
94644d93782SGreg Clayton   void Resize(int w, int h) { ::wresize(m_window, h, w); }
947b9c1b51eSKate Stone   void Resize(const Size &size) {
948b9c1b51eSKate Stone     ::wresize(m_window, size.height, size.width);
949b9c1b51eSKate Stone   }
95044d93782SGreg Clayton   void PutChar(int ch) { ::waddch(m_window, ch); }
95144d93782SGreg Clayton   void PutCString(const char *s, int len = -1) { ::waddnstr(m_window, s, len); }
95244d93782SGreg Clayton   void Refresh() { ::wrefresh(m_window); }
953b9c1b51eSKate Stone   void DeferredRefresh() {
95444d93782SGreg Clayton     // We are using panels, so we don't need to call this...
95544d93782SGreg Clayton     //::wnoutrefresh(m_window);
95644d93782SGreg Clayton   }
957b9c1b51eSKate Stone   void SetBackground(int color_pair_idx) {
958b9c1b51eSKate Stone     ::wbkgd(m_window, COLOR_PAIR(color_pair_idx));
959b9c1b51eSKate Stone   }
96044d93782SGreg Clayton   void UnderlineOn() { AttributeOn(A_UNDERLINE); }
96144d93782SGreg Clayton   void UnderlineOff() { AttributeOff(A_UNDERLINE); }
96244d93782SGreg Clayton 
963b9c1b51eSKate Stone   void PutCStringTruncated(const char *s, int right_pad) {
96444d93782SGreg Clayton     int bytes_left = GetWidth() - GetCursorX();
965b9c1b51eSKate Stone     if (bytes_left > right_pad) {
96644d93782SGreg Clayton       bytes_left -= right_pad;
96744d93782SGreg Clayton       ::waddnstr(m_window, s, bytes_left);
96844d93782SGreg Clayton     }
96944d93782SGreg Clayton   }
97044d93782SGreg Clayton 
971b9c1b51eSKate Stone   void MoveWindow(const Point &origin) {
97244d93782SGreg Clayton     const bool moving_window = origin != GetParentOrigin();
973b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
97444d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
97544d93782SGreg Clayton       Size size = GetSize();
976b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, size.height, size.width, origin.y,
977b9c1b51eSKate Stone                      origin.x),
978b9c1b51eSKate Stone             true);
979b9c1b51eSKate Stone     } else {
98044d93782SGreg Clayton       ::mvwin(m_window, origin.y, origin.x);
98144d93782SGreg Clayton     }
98244d93782SGreg Clayton   }
98344d93782SGreg Clayton 
984b9c1b51eSKate Stone   void SetBounds(const Rect &bounds) {
98544d93782SGreg Clayton     const bool moving_window = bounds.origin != GetParentOrigin();
986b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
98744d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
988b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, bounds.size.height, bounds.size.width,
989b9c1b51eSKate Stone                      bounds.origin.y, bounds.origin.x),
990b9c1b51eSKate Stone             true);
991b9c1b51eSKate Stone     } else {
99244d93782SGreg Clayton       if (moving_window)
99344d93782SGreg Clayton         MoveWindow(bounds.origin);
99444d93782SGreg Clayton       Resize(bounds.size);
99544d93782SGreg Clayton     }
99644d93782SGreg Clayton   }
99744d93782SGreg Clayton 
998b9c1b51eSKate Stone   void Printf(const char *format, ...) __attribute__((format(printf, 2, 3))) {
99944d93782SGreg Clayton     va_list args;
100044d93782SGreg Clayton     va_start(args, format);
100144d93782SGreg Clayton     vwprintw(m_window, format, args);
100244d93782SGreg Clayton     va_end(args);
100344d93782SGreg Clayton   }
100444d93782SGreg Clayton 
1005b9c1b51eSKate Stone   void Touch() {
100644d93782SGreg Clayton     ::touchwin(m_window);
100744d93782SGreg Clayton     if (m_parent)
100844d93782SGreg Clayton       m_parent->Touch();
100944d93782SGreg Clayton   }
101044d93782SGreg Clayton 
1011b9c1b51eSKate Stone   WindowSP CreateSubWindow(const char *name, const Rect &bounds,
1012b9c1b51eSKate Stone                            bool make_active) {
1013c6091d2bSJonas Devlieghere     auto get_window = [this, &bounds]() {
1014c6091d2bSJonas Devlieghere       return m_window
1015c6091d2bSJonas Devlieghere                  ? ::subwin(m_window, bounds.size.height, bounds.size.width,
1016c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x)
1017c6091d2bSJonas Devlieghere                  : ::newwin(bounds.size.height, bounds.size.width,
1018c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x);
1019c6091d2bSJonas Devlieghere     };
1020c6091d2bSJonas Devlieghere     WindowSP subwindow_sp = std::make_shared<Window>(name, get_window(), true);
1021c6091d2bSJonas Devlieghere     subwindow_sp->m_is_subwin = subwindow_sp.operator bool();
102244d93782SGreg Clayton     subwindow_sp->m_parent = this;
1023b9c1b51eSKate Stone     if (make_active) {
102444d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
102544d93782SGreg Clayton       m_curr_active_window_idx = m_subwindows.size();
102644d93782SGreg Clayton     }
102744d93782SGreg Clayton     m_subwindows.push_back(subwindow_sp);
102844d93782SGreg Clayton     ::top_panel(subwindow_sp->m_panel);
102944d93782SGreg Clayton     m_needs_update = true;
103044d93782SGreg Clayton     return subwindow_sp;
103144d93782SGreg Clayton   }
103244d93782SGreg Clayton 
1033b9c1b51eSKate Stone   bool RemoveSubWindow(Window *window) {
103444d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
103544d93782SGreg Clayton     size_t i = 0;
1036b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
1037b9c1b51eSKate Stone       if ((*pos).get() == window) {
103844d93782SGreg Clayton         if (m_prev_active_window_idx == i)
103944d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1040b9c1b51eSKate Stone         else if (m_prev_active_window_idx != UINT32_MAX &&
1041b9c1b51eSKate Stone                  m_prev_active_window_idx > i)
104244d93782SGreg Clayton           --m_prev_active_window_idx;
104344d93782SGreg Clayton 
104444d93782SGreg Clayton         if (m_curr_active_window_idx == i)
104544d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
1046b9c1b51eSKate Stone         else if (m_curr_active_window_idx != UINT32_MAX &&
1047b9c1b51eSKate Stone                  m_curr_active_window_idx > i)
104844d93782SGreg Clayton           --m_curr_active_window_idx;
104944d93782SGreg Clayton         window->Erase();
105044d93782SGreg Clayton         m_subwindows.erase(pos);
105144d93782SGreg Clayton         m_needs_update = true;
105244d93782SGreg Clayton         if (m_parent)
105344d93782SGreg Clayton           m_parent->Touch();
105444d93782SGreg Clayton         else
105544d93782SGreg Clayton           ::touchwin(stdscr);
105644d93782SGreg Clayton         return true;
105744d93782SGreg Clayton       }
105844d93782SGreg Clayton     }
105944d93782SGreg Clayton     return false;
106044d93782SGreg Clayton   }
106144d93782SGreg Clayton 
1062b9c1b51eSKate Stone   WindowSP FindSubWindow(const char *name) {
106344d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
106444d93782SGreg Clayton     size_t i = 0;
1065b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
10668d20cfdfSJonas Devlieghere       if ((*pos)->m_name == name)
106744d93782SGreg Clayton         return *pos;
106844d93782SGreg Clayton     }
106944d93782SGreg Clayton     return WindowSP();
107044d93782SGreg Clayton   }
107144d93782SGreg Clayton 
1072b9c1b51eSKate Stone   void RemoveSubWindows() {
107344d93782SGreg Clayton     m_curr_active_window_idx = UINT32_MAX;
107444d93782SGreg Clayton     m_prev_active_window_idx = UINT32_MAX;
107544d93782SGreg Clayton     for (Windows::iterator pos = m_subwindows.begin();
1076b9c1b51eSKate Stone          pos != m_subwindows.end(); pos = m_subwindows.erase(pos)) {
107744d93782SGreg Clayton       (*pos)->Erase();
107844d93782SGreg Clayton     }
107944d93782SGreg Clayton     if (m_parent)
108044d93782SGreg Clayton       m_parent->Touch();
108144d93782SGreg Clayton     else
108244d93782SGreg Clayton       ::touchwin(stdscr);
108344d93782SGreg Clayton   }
108444d93782SGreg Clayton 
1085b9c1b51eSKate Stone   WINDOW *get() { return m_window; }
108644d93782SGreg Clayton 
1087b9c1b51eSKate Stone   operator WINDOW *() { return m_window; }
108844d93782SGreg Clayton 
108944d93782SGreg Clayton   // Window drawing utilities
1090b9c1b51eSKate Stone   void DrawTitleBox(const char *title, const char *bottom_message = nullptr) {
109144d93782SGreg Clayton     attr_t attr = 0;
109244d93782SGreg Clayton     if (IsActive())
109344d93782SGreg Clayton       attr = A_BOLD | COLOR_PAIR(2);
109444d93782SGreg Clayton     else
109544d93782SGreg Clayton       attr = 0;
109644d93782SGreg Clayton     if (attr)
109744d93782SGreg Clayton       AttributeOn(attr);
109844d93782SGreg Clayton 
109944d93782SGreg Clayton     Box();
110044d93782SGreg Clayton     MoveCursor(3, 0);
110144d93782SGreg Clayton 
1102b9c1b51eSKate Stone     if (title && title[0]) {
110344d93782SGreg Clayton       PutChar('<');
110444d93782SGreg Clayton       PutCString(title);
110544d93782SGreg Clayton       PutChar('>');
110644d93782SGreg Clayton     }
110744d93782SGreg Clayton 
1108b9c1b51eSKate Stone     if (bottom_message && bottom_message[0]) {
110944d93782SGreg Clayton       int bottom_message_length = strlen(bottom_message);
111044d93782SGreg Clayton       int x = GetWidth() - 3 - (bottom_message_length + 2);
111144d93782SGreg Clayton 
1112b9c1b51eSKate Stone       if (x > 0) {
111344d93782SGreg Clayton         MoveCursor(x, GetHeight() - 1);
111444d93782SGreg Clayton         PutChar('[');
111544d93782SGreg Clayton         PutCString(bottom_message);
111644d93782SGreg Clayton         PutChar(']');
1117b9c1b51eSKate Stone       } else {
111844d93782SGreg Clayton         MoveCursor(1, GetHeight() - 1);
111944d93782SGreg Clayton         PutChar('[');
112044d93782SGreg Clayton         PutCStringTruncated(bottom_message, 1);
112144d93782SGreg Clayton       }
112244d93782SGreg Clayton     }
112344d93782SGreg Clayton     if (attr)
112444d93782SGreg Clayton       AttributeOff(attr);
112544d93782SGreg Clayton   }
112644d93782SGreg Clayton 
1127b9c1b51eSKate Stone   virtual void Draw(bool force) {
112844d93782SGreg Clayton     if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw(*this, force))
112944d93782SGreg Clayton       return;
113044d93782SGreg Clayton 
113144d93782SGreg Clayton     for (auto &subwindow_sp : m_subwindows)
113244d93782SGreg Clayton       subwindow_sp->Draw(force);
113344d93782SGreg Clayton   }
113444d93782SGreg Clayton 
1135b9c1b51eSKate Stone   bool CreateHelpSubwindow() {
1136b9c1b51eSKate Stone     if (m_delegate_sp) {
113744d93782SGreg Clayton       const char *text = m_delegate_sp->WindowDelegateGetHelpText();
113844d93782SGreg Clayton       KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp();
1139b9c1b51eSKate Stone       if ((text && text[0]) || key_help) {
1140d5b44036SJonas Devlieghere         std::unique_ptr<HelpDialogDelegate> help_delegate_up(
1141b9c1b51eSKate Stone             new HelpDialogDelegate(text, key_help));
1142d5b44036SJonas Devlieghere         const size_t num_lines = help_delegate_up->GetNumLines();
1143d5b44036SJonas Devlieghere         const size_t max_length = help_delegate_up->GetMaxLineLength();
114444d93782SGreg Clayton         Rect bounds = GetBounds();
114544d93782SGreg Clayton         bounds.Inset(1, 1);
1146b9c1b51eSKate Stone         if (max_length + 4 < static_cast<size_t>(bounds.size.width)) {
114744d93782SGreg Clayton           bounds.origin.x += (bounds.size.width - max_length + 4) / 2;
114844d93782SGreg Clayton           bounds.size.width = max_length + 4;
1149b9c1b51eSKate Stone         } else {
1150b9c1b51eSKate Stone           if (bounds.size.width > 100) {
115144d93782SGreg Clayton             const int inset_w = bounds.size.width / 4;
115244d93782SGreg Clayton             bounds.origin.x += inset_w;
115344d93782SGreg Clayton             bounds.size.width -= 2 * inset_w;
115444d93782SGreg Clayton           }
115544d93782SGreg Clayton         }
115644d93782SGreg Clayton 
1157b9c1b51eSKate Stone         if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) {
115844d93782SGreg Clayton           bounds.origin.y += (bounds.size.height - num_lines + 2) / 2;
115944d93782SGreg Clayton           bounds.size.height = num_lines + 2;
1160b9c1b51eSKate Stone         } else {
1161b9c1b51eSKate Stone           if (bounds.size.height > 100) {
116244d93782SGreg Clayton             const int inset_h = bounds.size.height / 4;
116344d93782SGreg Clayton             bounds.origin.y += inset_h;
116444d93782SGreg Clayton             bounds.size.height -= 2 * inset_h;
116544d93782SGreg Clayton           }
116644d93782SGreg Clayton         }
11675fdb09bbSGreg Clayton         WindowSP help_window_sp;
11685fdb09bbSGreg Clayton         Window *parent_window = GetParent();
11695fdb09bbSGreg Clayton         if (parent_window)
11705fdb09bbSGreg Clayton           help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
11715fdb09bbSGreg Clayton         else
11725fdb09bbSGreg Clayton           help_window_sp = CreateSubWindow("Help", bounds, true);
1173b9c1b51eSKate Stone         help_window_sp->SetDelegate(
1174d5b44036SJonas Devlieghere             WindowDelegateSP(help_delegate_up.release()));
117544d93782SGreg Clayton         return true;
117644d93782SGreg Clayton       }
117744d93782SGreg Clayton     }
117844d93782SGreg Clayton     return false;
117944d93782SGreg Clayton   }
118044d93782SGreg Clayton 
1181b9c1b51eSKate Stone   virtual HandleCharResult HandleChar(int key) {
118244d93782SGreg Clayton     // Always check the active window first
118344d93782SGreg Clayton     HandleCharResult result = eKeyNotHandled;
118444d93782SGreg Clayton     WindowSP active_window_sp = GetActiveWindow();
1185b9c1b51eSKate Stone     if (active_window_sp) {
118644d93782SGreg Clayton       result = active_window_sp->HandleChar(key);
118744d93782SGreg Clayton       if (result != eKeyNotHandled)
118844d93782SGreg Clayton         return result;
118944d93782SGreg Clayton     }
119044d93782SGreg Clayton 
1191b9c1b51eSKate Stone     if (m_delegate_sp) {
119244d93782SGreg Clayton       result = m_delegate_sp->WindowDelegateHandleChar(*this, key);
119344d93782SGreg Clayton       if (result != eKeyNotHandled)
119444d93782SGreg Clayton         return result;
119544d93782SGreg Clayton     }
119644d93782SGreg Clayton 
119705097246SAdrian Prantl     // Then check for any windows that want any keys that weren't handled. This
119805097246SAdrian Prantl     // is typically only for a menubar. Make a copy of the subwindows in case
119905097246SAdrian Prantl     // any HandleChar() functions muck with the subwindows. If we don't do
120005097246SAdrian Prantl     // this, we can crash when iterating over the subwindows.
120144d93782SGreg Clayton     Windows subwindows(m_subwindows);
1202b9c1b51eSKate Stone     for (auto subwindow_sp : subwindows) {
1203b9c1b51eSKate Stone       if (!subwindow_sp->m_can_activate) {
120444d93782SGreg Clayton         HandleCharResult result = subwindow_sp->HandleChar(key);
120544d93782SGreg Clayton         if (result != eKeyNotHandled)
120644d93782SGreg Clayton           return result;
120744d93782SGreg Clayton       }
120844d93782SGreg Clayton     }
120944d93782SGreg Clayton 
121044d93782SGreg Clayton     return eKeyNotHandled;
121144d93782SGreg Clayton   }
121244d93782SGreg Clayton 
1213b9c1b51eSKate Stone   bool SetActiveWindow(Window *window) {
121444d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1215b9c1b51eSKate Stone     for (size_t i = 0; i < num_subwindows; ++i) {
1216b9c1b51eSKate Stone       if (m_subwindows[i].get() == window) {
121744d93782SGreg Clayton         m_prev_active_window_idx = m_curr_active_window_idx;
121844d93782SGreg Clayton         ::top_panel(window->m_panel);
121944d93782SGreg Clayton         m_curr_active_window_idx = i;
122044d93782SGreg Clayton         return true;
122144d93782SGreg Clayton       }
122244d93782SGreg Clayton     }
122344d93782SGreg Clayton     return false;
122444d93782SGreg Clayton   }
122544d93782SGreg Clayton 
1226b9c1b51eSKate Stone   WindowSP GetActiveWindow() {
1227b9c1b51eSKate Stone     if (!m_subwindows.empty()) {
1228b9c1b51eSKate Stone       if (m_curr_active_window_idx >= m_subwindows.size()) {
1229b9c1b51eSKate Stone         if (m_prev_active_window_idx < m_subwindows.size()) {
123044d93782SGreg Clayton           m_curr_active_window_idx = m_prev_active_window_idx;
123144d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1232b9c1b51eSKate Stone         } else if (IsActive()) {
123344d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
123444d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
123544d93782SGreg Clayton 
123644d93782SGreg Clayton           // Find first window that wants to be active if this window is active
123744d93782SGreg Clayton           const size_t num_subwindows = m_subwindows.size();
1238b9c1b51eSKate Stone           for (size_t i = 0; i < num_subwindows; ++i) {
1239b9c1b51eSKate Stone             if (m_subwindows[i]->GetCanBeActive()) {
124044d93782SGreg Clayton               m_curr_active_window_idx = i;
124144d93782SGreg Clayton               break;
124244d93782SGreg Clayton             }
124344d93782SGreg Clayton           }
124444d93782SGreg Clayton         }
124544d93782SGreg Clayton       }
124644d93782SGreg Clayton 
124744d93782SGreg Clayton       if (m_curr_active_window_idx < m_subwindows.size())
124844d93782SGreg Clayton         return m_subwindows[m_curr_active_window_idx];
124944d93782SGreg Clayton     }
125044d93782SGreg Clayton     return WindowSP();
125144d93782SGreg Clayton   }
125244d93782SGreg Clayton 
1253b9c1b51eSKate Stone   bool GetCanBeActive() const { return m_can_activate; }
125444d93782SGreg Clayton 
1255b9c1b51eSKate Stone   void SetCanBeActive(bool b) { m_can_activate = b; }
125644d93782SGreg Clayton 
1257b9c1b51eSKate Stone   const WindowDelegateSP &GetDelegate() const { return m_delegate_sp; }
125844d93782SGreg Clayton 
1259b9c1b51eSKate Stone   void SetDelegate(const WindowDelegateSP &delegate_sp) {
126044d93782SGreg Clayton     m_delegate_sp = delegate_sp;
126144d93782SGreg Clayton   }
126244d93782SGreg Clayton 
1263b9c1b51eSKate Stone   Window *GetParent() const { return m_parent; }
126444d93782SGreg Clayton 
1265b9c1b51eSKate Stone   bool IsActive() const {
126644d93782SGreg Clayton     if (m_parent)
126744d93782SGreg Clayton       return m_parent->GetActiveWindow().get() == this;
126844d93782SGreg Clayton     else
126944d93782SGreg Clayton       return true; // Top level window is always active
127044d93782SGreg Clayton   }
127144d93782SGreg Clayton 
1272b9c1b51eSKate Stone   void SelectNextWindowAsActive() {
127344d93782SGreg Clayton     // Move active focus to next window
127444d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1275b9c1b51eSKate Stone     if (m_curr_active_window_idx == UINT32_MAX) {
127644d93782SGreg Clayton       uint32_t idx = 0;
1277b9c1b51eSKate Stone       for (auto subwindow_sp : m_subwindows) {
1278b9c1b51eSKate Stone         if (subwindow_sp->GetCanBeActive()) {
127944d93782SGreg Clayton           m_curr_active_window_idx = idx;
128044d93782SGreg Clayton           break;
128144d93782SGreg Clayton         }
128244d93782SGreg Clayton         ++idx;
128344d93782SGreg Clayton       }
1284b9c1b51eSKate Stone     } else if (m_curr_active_window_idx + 1 < num_subwindows) {
128544d93782SGreg Clayton       bool handled = false;
128644d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1287b9c1b51eSKate Stone       for (size_t idx = m_curr_active_window_idx + 1; idx < num_subwindows;
1288b9c1b51eSKate Stone            ++idx) {
1289b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
129044d93782SGreg Clayton           m_curr_active_window_idx = idx;
129144d93782SGreg Clayton           handled = true;
129244d93782SGreg Clayton           break;
129344d93782SGreg Clayton         }
129444d93782SGreg Clayton       }
1295b9c1b51eSKate Stone       if (!handled) {
1296b9c1b51eSKate Stone         for (size_t idx = 0; idx <= m_prev_active_window_idx; ++idx) {
1297b9c1b51eSKate Stone           if (m_subwindows[idx]->GetCanBeActive()) {
129844d93782SGreg Clayton             m_curr_active_window_idx = idx;
129944d93782SGreg Clayton             break;
130044d93782SGreg Clayton           }
130144d93782SGreg Clayton         }
130244d93782SGreg Clayton       }
1303b9c1b51eSKate Stone     } else {
130444d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1305b9c1b51eSKate Stone       for (size_t idx = 0; idx < num_subwindows; ++idx) {
1306b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
130744d93782SGreg Clayton           m_curr_active_window_idx = idx;
130844d93782SGreg Clayton           break;
130944d93782SGreg Clayton         }
131044d93782SGreg Clayton       }
131144d93782SGreg Clayton     }
131244d93782SGreg Clayton   }
131344d93782SGreg Clayton 
1314b9c1b51eSKate Stone   const char *GetName() const { return m_name.c_str(); }
1315315b6884SEugene Zelenko 
131644d93782SGreg Clayton protected:
131744d93782SGreg Clayton   std::string m_name;
131844d93782SGreg Clayton   WINDOW *m_window;
131944d93782SGreg Clayton   PANEL *m_panel;
132044d93782SGreg Clayton   Window *m_parent;
132144d93782SGreg Clayton   Windows m_subwindows;
132244d93782SGreg Clayton   WindowDelegateSP m_delegate_sp;
132344d93782SGreg Clayton   uint32_t m_curr_active_window_idx;
132444d93782SGreg Clayton   uint32_t m_prev_active_window_idx;
132544d93782SGreg Clayton   bool m_delete;
132644d93782SGreg Clayton   bool m_needs_update;
132744d93782SGreg Clayton   bool m_can_activate;
132844d93782SGreg Clayton   bool m_is_subwin;
132944d93782SGreg Clayton 
133044d93782SGreg Clayton private:
133144d93782SGreg Clayton   DISALLOW_COPY_AND_ASSIGN(Window);
133244d93782SGreg Clayton };
133344d93782SGreg Clayton 
1334b9c1b51eSKate Stone class MenuDelegate {
133544d93782SGreg Clayton public:
1336315b6884SEugene Zelenko   virtual ~MenuDelegate() = default;
133744d93782SGreg Clayton 
1338b9c1b51eSKate Stone   virtual MenuActionResult MenuDelegateAction(Menu &menu) = 0;
133944d93782SGreg Clayton };
134044d93782SGreg Clayton 
1341b9c1b51eSKate Stone class Menu : public WindowDelegate {
134244d93782SGreg Clayton public:
1343b9c1b51eSKate Stone   enum class Type { Invalid, Bar, Item, Separator };
134444d93782SGreg Clayton 
134544d93782SGreg Clayton   // Menubar or separator constructor
134644d93782SGreg Clayton   Menu(Type type);
134744d93782SGreg Clayton 
134844d93782SGreg Clayton   // Menuitem constructor
1349b9c1b51eSKate Stone   Menu(const char *name, const char *key_name, int key_value,
135044d93782SGreg Clayton        uint64_t identifier);
135144d93782SGreg Clayton 
1352315b6884SEugene Zelenko   ~Menu() override = default;
135344d93782SGreg Clayton 
1354b9c1b51eSKate Stone   const MenuDelegateSP &GetDelegate() const { return m_delegate_sp; }
135544d93782SGreg Clayton 
1356b9c1b51eSKate Stone   void SetDelegate(const MenuDelegateSP &delegate_sp) {
135744d93782SGreg Clayton     m_delegate_sp = delegate_sp;
135844d93782SGreg Clayton   }
135944d93782SGreg Clayton 
1360b9c1b51eSKate Stone   void RecalculateNameLengths();
136144d93782SGreg Clayton 
1362b9c1b51eSKate Stone   void AddSubmenu(const MenuSP &menu_sp);
136344d93782SGreg Clayton 
1364b9c1b51eSKate Stone   int DrawAndRunMenu(Window &window);
136544d93782SGreg Clayton 
1366b9c1b51eSKate Stone   void DrawMenuTitle(Window &window, bool highlight);
136744d93782SGreg Clayton 
1368b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
136944d93782SGreg Clayton 
1370b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
137144d93782SGreg Clayton 
1372b9c1b51eSKate Stone   MenuActionResult ActionPrivate(Menu &menu) {
137344d93782SGreg Clayton     MenuActionResult result = MenuActionResult::NotHandled;
1374b9c1b51eSKate Stone     if (m_delegate_sp) {
137544d93782SGreg Clayton       result = m_delegate_sp->MenuDelegateAction(menu);
137644d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
137744d93782SGreg Clayton         return result;
1378b9c1b51eSKate Stone     } else if (m_parent) {
137944d93782SGreg Clayton       result = m_parent->ActionPrivate(menu);
138044d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
138144d93782SGreg Clayton         return result;
138244d93782SGreg Clayton     }
138344d93782SGreg Clayton     return m_canned_result;
138444d93782SGreg Clayton   }
138544d93782SGreg Clayton 
1386b9c1b51eSKate Stone   MenuActionResult Action() {
138705097246SAdrian Prantl     // Call the recursive action so it can try to handle it with the menu
138805097246SAdrian Prantl     // delegate, and if not, try our parent menu
138944d93782SGreg Clayton     return ActionPrivate(*this);
139044d93782SGreg Clayton   }
139144d93782SGreg Clayton 
1392b9c1b51eSKate Stone   void SetCannedResult(MenuActionResult result) { m_canned_result = result; }
139344d93782SGreg Clayton 
1394b9c1b51eSKate Stone   Menus &GetSubmenus() { return m_submenus; }
139544d93782SGreg Clayton 
1396b9c1b51eSKate Stone   const Menus &GetSubmenus() const { return m_submenus; }
139744d93782SGreg Clayton 
1398b9c1b51eSKate Stone   int GetSelectedSubmenuIndex() const { return m_selected; }
139944d93782SGreg Clayton 
1400b9c1b51eSKate Stone   void SetSelectedSubmenuIndex(int idx) { m_selected = idx; }
140144d93782SGreg Clayton 
1402b9c1b51eSKate Stone   Type GetType() const { return m_type; }
140344d93782SGreg Clayton 
1404b9c1b51eSKate Stone   int GetStartingColumn() const { return m_start_col; }
140544d93782SGreg Clayton 
1406b9c1b51eSKate Stone   void SetStartingColumn(int col) { m_start_col = col; }
140744d93782SGreg Clayton 
1408b9c1b51eSKate Stone   int GetKeyValue() const { return m_key_value; }
140944d93782SGreg Clayton 
1410b9c1b51eSKate Stone   void SetKeyValue(int key_value) { m_key_value = key_value; }
141144d93782SGreg Clayton 
1412b9c1b51eSKate Stone   std::string &GetName() { return m_name; }
141344d93782SGreg Clayton 
1414b9c1b51eSKate Stone   std::string &GetKeyName() { return m_key_name; }
141544d93782SGreg Clayton 
1416b9c1b51eSKate Stone   int GetDrawWidth() const {
141744d93782SGreg Clayton     return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
141844d93782SGreg Clayton   }
141944d93782SGreg Clayton 
1420b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
142144d93782SGreg Clayton 
1422b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
142344d93782SGreg Clayton 
142444d93782SGreg Clayton protected:
142544d93782SGreg Clayton   std::string m_name;
142644d93782SGreg Clayton   std::string m_key_name;
142744d93782SGreg Clayton   uint64_t m_identifier;
142844d93782SGreg Clayton   Type m_type;
142944d93782SGreg Clayton   int m_key_value;
143044d93782SGreg Clayton   int m_start_col;
143144d93782SGreg Clayton   int m_max_submenu_name_length;
143244d93782SGreg Clayton   int m_max_submenu_key_name_length;
143344d93782SGreg Clayton   int m_selected;
143444d93782SGreg Clayton   Menu *m_parent;
143544d93782SGreg Clayton   Menus m_submenus;
143644d93782SGreg Clayton   WindowSP m_menu_window_sp;
143744d93782SGreg Clayton   MenuActionResult m_canned_result;
143844d93782SGreg Clayton   MenuDelegateSP m_delegate_sp;
143944d93782SGreg Clayton };
144044d93782SGreg Clayton 
144144d93782SGreg Clayton // Menubar or separator constructor
1442b9c1b51eSKate Stone Menu::Menu(Type type)
1443b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(0), m_type(type), m_key_value(0),
1444b9c1b51eSKate Stone       m_start_col(0), m_max_submenu_name_length(0),
1445b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1446b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1447b9c1b51eSKate Stone       m_delegate_sp() {}
144844d93782SGreg Clayton 
144944d93782SGreg Clayton // Menuitem constructor
1450b9c1b51eSKate Stone Menu::Menu(const char *name, const char *key_name, int key_value,
1451b9c1b51eSKate Stone            uint64_t identifier)
1452b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(identifier), m_type(Type::Invalid),
1453b9c1b51eSKate Stone       m_key_value(key_value), m_start_col(0), m_max_submenu_name_length(0),
1454b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1455b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1456b9c1b51eSKate Stone       m_delegate_sp() {
1457b9c1b51eSKate Stone   if (name && name[0]) {
145844d93782SGreg Clayton     m_name = name;
145944d93782SGreg Clayton     m_type = Type::Item;
146044d93782SGreg Clayton     if (key_name && key_name[0])
146144d93782SGreg Clayton       m_key_name = key_name;
1462b9c1b51eSKate Stone   } else {
146344d93782SGreg Clayton     m_type = Type::Separator;
146444d93782SGreg Clayton   }
146544d93782SGreg Clayton }
146644d93782SGreg Clayton 
1467b9c1b51eSKate Stone void Menu::RecalculateNameLengths() {
146844d93782SGreg Clayton   m_max_submenu_name_length = 0;
146944d93782SGreg Clayton   m_max_submenu_key_name_length = 0;
147044d93782SGreg Clayton   Menus &submenus = GetSubmenus();
147144d93782SGreg Clayton   const size_t num_submenus = submenus.size();
1472b9c1b51eSKate Stone   for (size_t i = 0; i < num_submenus; ++i) {
147344d93782SGreg Clayton     Menu *submenu = submenus[i].get();
14743985c8c6SSaleem Abdulrasool     if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
147544d93782SGreg Clayton       m_max_submenu_name_length = submenu->m_name.size();
1476b9c1b51eSKate Stone     if (static_cast<size_t>(m_max_submenu_key_name_length) <
1477b9c1b51eSKate Stone         submenu->m_key_name.size())
147844d93782SGreg Clayton       m_max_submenu_key_name_length = submenu->m_key_name.size();
147944d93782SGreg Clayton   }
148044d93782SGreg Clayton }
148144d93782SGreg Clayton 
1482b9c1b51eSKate Stone void Menu::AddSubmenu(const MenuSP &menu_sp) {
148344d93782SGreg Clayton   menu_sp->m_parent = this;
14843985c8c6SSaleem Abdulrasool   if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
148544d93782SGreg Clayton     m_max_submenu_name_length = menu_sp->m_name.size();
1486b9c1b51eSKate Stone   if (static_cast<size_t>(m_max_submenu_key_name_length) <
1487b9c1b51eSKate Stone       menu_sp->m_key_name.size())
148844d93782SGreg Clayton     m_max_submenu_key_name_length = menu_sp->m_key_name.size();
148944d93782SGreg Clayton   m_submenus.push_back(menu_sp);
149044d93782SGreg Clayton }
149144d93782SGreg Clayton 
1492b9c1b51eSKate Stone void Menu::DrawMenuTitle(Window &window, bool highlight) {
1493b9c1b51eSKate Stone   if (m_type == Type::Separator) {
149444d93782SGreg Clayton     window.MoveCursor(0, window.GetCursorY());
149544d93782SGreg Clayton     window.PutChar(ACS_LTEE);
149644d93782SGreg Clayton     int width = window.GetWidth();
1497b9c1b51eSKate Stone     if (width > 2) {
149844d93782SGreg Clayton       width -= 2;
14993985c8c6SSaleem Abdulrasool       for (int i = 0; i < width; ++i)
150044d93782SGreg Clayton         window.PutChar(ACS_HLINE);
150144d93782SGreg Clayton     }
150244d93782SGreg Clayton     window.PutChar(ACS_RTEE);
1503b9c1b51eSKate Stone   } else {
150444d93782SGreg Clayton     const int shortcut_key = m_key_value;
150544d93782SGreg Clayton     bool underlined_shortcut = false;
150644d93782SGreg Clayton     const attr_t hilgight_attr = A_REVERSE;
150744d93782SGreg Clayton     if (highlight)
150844d93782SGreg Clayton       window.AttributeOn(hilgight_attr);
1509b9c1b51eSKate Stone     if (isprint(shortcut_key)) {
151044d93782SGreg Clayton       size_t lower_pos = m_name.find(tolower(shortcut_key));
151144d93782SGreg Clayton       size_t upper_pos = m_name.find(toupper(shortcut_key));
151244d93782SGreg Clayton       const char *name = m_name.c_str();
151344d93782SGreg Clayton       size_t pos = std::min<size_t>(lower_pos, upper_pos);
1514b9c1b51eSKate Stone       if (pos != std::string::npos) {
151544d93782SGreg Clayton         underlined_shortcut = true;
1516b9c1b51eSKate Stone         if (pos > 0) {
151744d93782SGreg Clayton           window.PutCString(name, pos);
151844d93782SGreg Clayton           name += pos;
151944d93782SGreg Clayton         }
152044d93782SGreg Clayton         const attr_t shortcut_attr = A_UNDERLINE | A_BOLD;
152144d93782SGreg Clayton         window.AttributeOn(shortcut_attr);
152244d93782SGreg Clayton         window.PutChar(name[0]);
152344d93782SGreg Clayton         window.AttributeOff(shortcut_attr);
152444d93782SGreg Clayton         name++;
152544d93782SGreg Clayton         if (name[0])
152644d93782SGreg Clayton           window.PutCString(name);
152744d93782SGreg Clayton       }
152844d93782SGreg Clayton     }
152944d93782SGreg Clayton 
1530b9c1b51eSKate Stone     if (!underlined_shortcut) {
153144d93782SGreg Clayton       window.PutCString(m_name.c_str());
153244d93782SGreg Clayton     }
153344d93782SGreg Clayton 
153444d93782SGreg Clayton     if (highlight)
153544d93782SGreg Clayton       window.AttributeOff(hilgight_attr);
153644d93782SGreg Clayton 
1537b9c1b51eSKate Stone     if (m_key_name.empty()) {
1538b9c1b51eSKate Stone       if (!underlined_shortcut && isprint(m_key_value)) {
153944d93782SGreg Clayton         window.AttributeOn(COLOR_PAIR(3));
154044d93782SGreg Clayton         window.Printf(" (%c)", m_key_value);
154144d93782SGreg Clayton         window.AttributeOff(COLOR_PAIR(3));
154244d93782SGreg Clayton       }
1543b9c1b51eSKate Stone     } else {
154444d93782SGreg Clayton       window.AttributeOn(COLOR_PAIR(3));
154544d93782SGreg Clayton       window.Printf(" (%s)", m_key_name.c_str());
154644d93782SGreg Clayton       window.AttributeOff(COLOR_PAIR(3));
154744d93782SGreg Clayton     }
154844d93782SGreg Clayton   }
154944d93782SGreg Clayton }
155044d93782SGreg Clayton 
1551b9c1b51eSKate Stone bool Menu::WindowDelegateDraw(Window &window, bool force) {
155244d93782SGreg Clayton   Menus &submenus = GetSubmenus();
155344d93782SGreg Clayton   const size_t num_submenus = submenus.size();
155444d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
155544d93782SGreg Clayton   Menu::Type menu_type = GetType();
1556b9c1b51eSKate Stone   switch (menu_type) {
1557b9c1b51eSKate Stone   case Menu::Type::Bar: {
155844d93782SGreg Clayton     window.SetBackground(2);
155944d93782SGreg Clayton     window.MoveCursor(0, 0);
1560b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
156144d93782SGreg Clayton       Menu *menu = submenus[i].get();
156244d93782SGreg Clayton       if (i > 0)
156344d93782SGreg Clayton         window.PutChar(' ');
156444d93782SGreg Clayton       menu->SetStartingColumn(window.GetCursorX());
156544d93782SGreg Clayton       window.PutCString("| ");
156644d93782SGreg Clayton       menu->DrawMenuTitle(window, false);
156744d93782SGreg Clayton     }
156844d93782SGreg Clayton     window.PutCString(" |");
156944d93782SGreg Clayton     window.DeferredRefresh();
1570b9c1b51eSKate Stone   } break;
157144d93782SGreg Clayton 
1572b9c1b51eSKate Stone   case Menu::Type::Item: {
157344d93782SGreg Clayton     int y = 1;
157444d93782SGreg Clayton     int x = 3;
157544d93782SGreg Clayton     // Draw the menu
157644d93782SGreg Clayton     int cursor_x = 0;
157744d93782SGreg Clayton     int cursor_y = 0;
157844d93782SGreg Clayton     window.Erase();
157944d93782SGreg Clayton     window.SetBackground(2);
158044d93782SGreg Clayton     window.Box();
1581b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
1582b9c1b51eSKate Stone       const bool is_selected = (i == static_cast<size_t>(selected_idx));
158344d93782SGreg Clayton       window.MoveCursor(x, y + i);
1584b9c1b51eSKate Stone       if (is_selected) {
158544d93782SGreg Clayton         // Remember where we want the cursor to be
158644d93782SGreg Clayton         cursor_x = x - 1;
158744d93782SGreg Clayton         cursor_y = y + i;
158844d93782SGreg Clayton       }
158944d93782SGreg Clayton       submenus[i]->DrawMenuTitle(window, is_selected);
159044d93782SGreg Clayton     }
159144d93782SGreg Clayton     window.MoveCursor(cursor_x, cursor_y);
159244d93782SGreg Clayton     window.DeferredRefresh();
1593b9c1b51eSKate Stone   } break;
159444d93782SGreg Clayton 
159544d93782SGreg Clayton   default:
159644d93782SGreg Clayton   case Menu::Type::Separator:
159744d93782SGreg Clayton     break;
159844d93782SGreg Clayton   }
159944d93782SGreg Clayton   return true; // Drawing handled...
160044d93782SGreg Clayton }
160144d93782SGreg Clayton 
1602b9c1b51eSKate Stone HandleCharResult Menu::WindowDelegateHandleChar(Window &window, int key) {
160344d93782SGreg Clayton   HandleCharResult result = eKeyNotHandled;
160444d93782SGreg Clayton 
160544d93782SGreg Clayton   Menus &submenus = GetSubmenus();
160644d93782SGreg Clayton   const size_t num_submenus = submenus.size();
160744d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
160844d93782SGreg Clayton   Menu::Type menu_type = GetType();
1609b9c1b51eSKate Stone   if (menu_type == Menu::Type::Bar) {
161044d93782SGreg Clayton     MenuSP run_menu_sp;
1611b9c1b51eSKate Stone     switch (key) {
161244d93782SGreg Clayton     case KEY_DOWN:
161344d93782SGreg Clayton     case KEY_UP:
161444d93782SGreg Clayton       // Show last menu or first menu
16153985c8c6SSaleem Abdulrasool       if (selected_idx < static_cast<int>(num_submenus))
161644d93782SGreg Clayton         run_menu_sp = submenus[selected_idx];
161744d93782SGreg Clayton       else if (!submenus.empty())
161844d93782SGreg Clayton         run_menu_sp = submenus.front();
161944d93782SGreg Clayton       result = eKeyHandled;
162044d93782SGreg Clayton       break;
162144d93782SGreg Clayton 
162244d93782SGreg Clayton     case KEY_RIGHT:
162344d93782SGreg Clayton       ++m_selected;
16243985c8c6SSaleem Abdulrasool       if (m_selected >= static_cast<int>(num_submenus))
162544d93782SGreg Clayton         m_selected = 0;
16263985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
162744d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
162844d93782SGreg Clayton       else if (!submenus.empty())
162944d93782SGreg Clayton         run_menu_sp = submenus.front();
163044d93782SGreg Clayton       result = eKeyHandled;
163144d93782SGreg Clayton       break;
163244d93782SGreg Clayton 
163344d93782SGreg Clayton     case KEY_LEFT:
163444d93782SGreg Clayton       --m_selected;
163544d93782SGreg Clayton       if (m_selected < 0)
163644d93782SGreg Clayton         m_selected = num_submenus - 1;
16373985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
163844d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
163944d93782SGreg Clayton       else if (!submenus.empty())
164044d93782SGreg Clayton         run_menu_sp = submenus.front();
164144d93782SGreg Clayton       result = eKeyHandled;
164244d93782SGreg Clayton       break;
164344d93782SGreg Clayton 
164444d93782SGreg Clayton     default:
1645b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
1646b9c1b51eSKate Stone         if (submenus[i]->GetKeyValue() == key) {
164744d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
164844d93782SGreg Clayton           run_menu_sp = submenus[i];
164944d93782SGreg Clayton           result = eKeyHandled;
165044d93782SGreg Clayton           break;
165144d93782SGreg Clayton         }
165244d93782SGreg Clayton       }
165344d93782SGreg Clayton       break;
165444d93782SGreg Clayton     }
165544d93782SGreg Clayton 
1656b9c1b51eSKate Stone     if (run_menu_sp) {
165705097246SAdrian Prantl       // Run the action on this menu in case we need to populate the menu with
165805097246SAdrian Prantl       // dynamic content and also in case check marks, and any other menu
165905097246SAdrian Prantl       // decorations need to be calculated
166044d93782SGreg Clayton       if (run_menu_sp->Action() == MenuActionResult::Quit)
166144d93782SGreg Clayton         return eQuitApplication;
166244d93782SGreg Clayton 
166344d93782SGreg Clayton       Rect menu_bounds;
166444d93782SGreg Clayton       menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
166544d93782SGreg Clayton       menu_bounds.origin.y = 1;
166644d93782SGreg Clayton       menu_bounds.size.width = run_menu_sp->GetDrawWidth();
166744d93782SGreg Clayton       menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
166844d93782SGreg Clayton       if (m_menu_window_sp)
166944d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
167044d93782SGreg Clayton 
1671b9c1b51eSKate Stone       m_menu_window_sp = window.GetParent()->CreateSubWindow(
1672b9c1b51eSKate Stone           run_menu_sp->GetName().c_str(), menu_bounds, true);
167344d93782SGreg Clayton       m_menu_window_sp->SetDelegate(run_menu_sp);
167444d93782SGreg Clayton     }
1675b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Item) {
1676b9c1b51eSKate Stone     switch (key) {
167744d93782SGreg Clayton     case KEY_DOWN:
1678b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
167944d93782SGreg Clayton         const int start_select = m_selected;
1680b9c1b51eSKate Stone         while (++m_selected != start_select) {
16813985c8c6SSaleem Abdulrasool           if (static_cast<size_t>(m_selected) >= num_submenus)
168244d93782SGreg Clayton             m_selected = 0;
168344d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
168444d93782SGreg Clayton             continue;
168544d93782SGreg Clayton           else
168644d93782SGreg Clayton             break;
168744d93782SGreg Clayton         }
168844d93782SGreg Clayton         return eKeyHandled;
168944d93782SGreg Clayton       }
169044d93782SGreg Clayton       break;
169144d93782SGreg Clayton 
169244d93782SGreg Clayton     case KEY_UP:
1693b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
169444d93782SGreg Clayton         const int start_select = m_selected;
1695b9c1b51eSKate Stone         while (--m_selected != start_select) {
16963985c8c6SSaleem Abdulrasool           if (m_selected < static_cast<int>(0))
169744d93782SGreg Clayton             m_selected = num_submenus - 1;
169844d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
169944d93782SGreg Clayton             continue;
170044d93782SGreg Clayton           else
170144d93782SGreg Clayton             break;
170244d93782SGreg Clayton         }
170344d93782SGreg Clayton         return eKeyHandled;
170444d93782SGreg Clayton       }
170544d93782SGreg Clayton       break;
170644d93782SGreg Clayton 
170744d93782SGreg Clayton     case KEY_RETURN:
1708b9c1b51eSKate Stone       if (static_cast<size_t>(selected_idx) < num_submenus) {
170944d93782SGreg Clayton         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
171044d93782SGreg Clayton           return eQuitApplication;
171144d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
171244d93782SGreg Clayton         return eKeyHandled;
171344d93782SGreg Clayton       }
171444d93782SGreg Clayton       break;
171544d93782SGreg Clayton 
1716b9c1b51eSKate Stone     case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in
1717b9c1b51eSKate Stone                      // case other chars are entered for escaped sequences
171844d93782SGreg Clayton       window.GetParent()->RemoveSubWindow(&window);
171944d93782SGreg Clayton       return eKeyHandled;
172044d93782SGreg Clayton 
172144d93782SGreg Clayton     default:
1722b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
172344d93782SGreg Clayton         Menu *menu = submenus[i].get();
1724b9c1b51eSKate Stone         if (menu->GetKeyValue() == key) {
172544d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
172644d93782SGreg Clayton           window.GetParent()->RemoveSubWindow(&window);
172744d93782SGreg Clayton           if (menu->Action() == MenuActionResult::Quit)
172844d93782SGreg Clayton             return eQuitApplication;
172944d93782SGreg Clayton           return eKeyHandled;
173044d93782SGreg Clayton         }
173144d93782SGreg Clayton       }
173244d93782SGreg Clayton       break;
173344d93782SGreg Clayton     }
1734b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Separator) {
173544d93782SGreg Clayton   }
173644d93782SGreg Clayton   return result;
173744d93782SGreg Clayton }
173844d93782SGreg Clayton 
1739b9c1b51eSKate Stone class Application {
174044d93782SGreg Clayton public:
1741b9c1b51eSKate Stone   Application(FILE *in, FILE *out)
1742b9c1b51eSKate Stone       : m_window_sp(), m_screen(nullptr), m_in(in), m_out(out) {}
174344d93782SGreg Clayton 
1744b9c1b51eSKate Stone   ~Application() {
174544d93782SGreg Clayton     m_window_delegates.clear();
174644d93782SGreg Clayton     m_window_sp.reset();
1747b9c1b51eSKate Stone     if (m_screen) {
174844d93782SGreg Clayton       ::delscreen(m_screen);
1749c5dac77aSEugene Zelenko       m_screen = nullptr;
175044d93782SGreg Clayton     }
175144d93782SGreg Clayton   }
175244d93782SGreg Clayton 
1753b9c1b51eSKate Stone   void Initialize() {
175444d93782SGreg Clayton     ::setlocale(LC_ALL, "");
175544d93782SGreg Clayton     ::setlocale(LC_CTYPE, "");
1756c5dac77aSEugene Zelenko     m_screen = ::newterm(nullptr, m_out, m_in);
175744d93782SGreg Clayton     ::start_color();
175844d93782SGreg Clayton     ::curs_set(0);
175944d93782SGreg Clayton     ::noecho();
176044d93782SGreg Clayton     ::keypad(stdscr, TRUE);
176144d93782SGreg Clayton   }
176244d93782SGreg Clayton 
1763b9c1b51eSKate Stone   void Terminate() { ::endwin(); }
176444d93782SGreg Clayton 
1765b9c1b51eSKate Stone   void Run(Debugger &debugger) {
176644d93782SGreg Clayton     bool done = false;
176744d93782SGreg Clayton     int delay_in_tenths_of_a_second = 1;
176844d93782SGreg Clayton 
176905097246SAdrian Prantl     // Alas the threading model in curses is a bit lame so we need to resort to
177005097246SAdrian Prantl     // polling every 0.5 seconds. We could poll for stdin ourselves and then
177105097246SAdrian Prantl     // pass the keys down but then we need to translate all of the escape
177205097246SAdrian Prantl     // sequences ourselves. So we resort to polling for input because we need
177305097246SAdrian Prantl     // to receive async process events while in this loop.
177444d93782SGreg Clayton 
1775b9c1b51eSKate Stone     halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths
1776b9c1b51eSKate Stone                                             // of seconds seconds when calling
1777b9c1b51eSKate Stone                                             // Window::GetChar()
177844d93782SGreg Clayton 
1779b9c1b51eSKate Stone     ListenerSP listener_sp(
1780b9c1b51eSKate Stone         Listener::MakeListener("lldb.IOHandler.curses.Application"));
178144d93782SGreg Clayton     ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
178244d93782SGreg Clayton     ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
178344d93782SGreg Clayton     ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
178444d93782SGreg Clayton     debugger.EnableForwardEvents(listener_sp);
178544d93782SGreg Clayton 
178644d93782SGreg Clayton     bool update = true;
178744d93782SGreg Clayton #if defined(__APPLE__)
178844d93782SGreg Clayton     std::deque<int> escape_chars;
178944d93782SGreg Clayton #endif
179044d93782SGreg Clayton 
1791b9c1b51eSKate Stone     while (!done) {
1792b9c1b51eSKate Stone       if (update) {
179344d93782SGreg Clayton         m_window_sp->Draw(false);
179405097246SAdrian Prantl         // All windows should be calling Window::DeferredRefresh() instead of
179505097246SAdrian Prantl         // Window::Refresh() so we can do a single update and avoid any screen
179605097246SAdrian Prantl         // blinking
179744d93782SGreg Clayton         update_panels();
179844d93782SGreg Clayton 
1799b9c1b51eSKate Stone         // Cursor hiding isn't working on MacOSX, so hide it in the top left
1800b9c1b51eSKate Stone         // corner
180144d93782SGreg Clayton         m_window_sp->MoveCursor(0, 0);
180244d93782SGreg Clayton 
180344d93782SGreg Clayton         doupdate();
180444d93782SGreg Clayton         update = false;
180544d93782SGreg Clayton       }
180644d93782SGreg Clayton 
180744d93782SGreg Clayton #if defined(__APPLE__)
180805097246SAdrian Prantl       // Terminal.app doesn't map its function keys correctly, F1-F4 default
180905097246SAdrian Prantl       // to: \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if
1810b9c1b51eSKate Stone       // possible
181144d93782SGreg Clayton       int ch;
181244d93782SGreg Clayton       if (escape_chars.empty())
181344d93782SGreg Clayton         ch = m_window_sp->GetChar();
1814b9c1b51eSKate Stone       else {
181544d93782SGreg Clayton         ch = escape_chars.front();
181644d93782SGreg Clayton         escape_chars.pop_front();
181744d93782SGreg Clayton       }
1818b9c1b51eSKate Stone       if (ch == KEY_ESCAPE) {
181944d93782SGreg Clayton         int ch2 = m_window_sp->GetChar();
1820b9c1b51eSKate Stone         if (ch2 == 'O') {
182144d93782SGreg Clayton           int ch3 = m_window_sp->GetChar();
1822b9c1b51eSKate Stone           switch (ch3) {
1823b9c1b51eSKate Stone           case 'P':
1824b9c1b51eSKate Stone             ch = KEY_F(1);
1825b9c1b51eSKate Stone             break;
1826b9c1b51eSKate Stone           case 'Q':
1827b9c1b51eSKate Stone             ch = KEY_F(2);
1828b9c1b51eSKate Stone             break;
1829b9c1b51eSKate Stone           case 'R':
1830b9c1b51eSKate Stone             ch = KEY_F(3);
1831b9c1b51eSKate Stone             break;
1832b9c1b51eSKate Stone           case 'S':
1833b9c1b51eSKate Stone             ch = KEY_F(4);
1834b9c1b51eSKate Stone             break;
183544d93782SGreg Clayton           default:
183644d93782SGreg Clayton             escape_chars.push_back(ch2);
183744d93782SGreg Clayton             if (ch3 != -1)
183844d93782SGreg Clayton               escape_chars.push_back(ch3);
183944d93782SGreg Clayton             break;
184044d93782SGreg Clayton           }
1841b9c1b51eSKate Stone         } else if (ch2 != -1)
184244d93782SGreg Clayton           escape_chars.push_back(ch2);
184344d93782SGreg Clayton       }
184444d93782SGreg Clayton #else
184544d93782SGreg Clayton       int ch = m_window_sp->GetChar();
184644d93782SGreg Clayton 
184744d93782SGreg Clayton #endif
1848b9c1b51eSKate Stone       if (ch == -1) {
1849b9c1b51eSKate Stone         if (feof(m_in) || ferror(m_in)) {
185044d93782SGreg Clayton           done = true;
1851b9c1b51eSKate Stone         } else {
185244d93782SGreg Clayton           // Just a timeout from using halfdelay(), check for events
185344d93782SGreg Clayton           EventSP event_sp;
1854b9c1b51eSKate Stone           while (listener_sp->PeekAtNextEvent()) {
1855d35031e1SPavel Labath             listener_sp->GetEvent(event_sp, std::chrono::seconds(0));
185644d93782SGreg Clayton 
1857b9c1b51eSKate Stone             if (event_sp) {
185844d93782SGreg Clayton               Broadcaster *broadcaster = event_sp->GetBroadcaster();
1859b9c1b51eSKate Stone               if (broadcaster) {
186044d93782SGreg Clayton                 // uint32_t event_type = event_sp->GetType();
1861b9c1b51eSKate Stone                 ConstString broadcaster_class(
1862b9c1b51eSKate Stone                     broadcaster->GetBroadcasterClass());
1863b9c1b51eSKate Stone                 if (broadcaster_class == broadcaster_class_process) {
1864b9c1b51eSKate Stone                   debugger.GetCommandInterpreter().UpdateExecutionContext(
1865b9c1b51eSKate Stone                       nullptr);
186644d93782SGreg Clayton                   update = true;
186744d93782SGreg Clayton                   continue; // Don't get any key, just update our view
186844d93782SGreg Clayton                 }
186944d93782SGreg Clayton               }
187044d93782SGreg Clayton             }
187144d93782SGreg Clayton           }
187244d93782SGreg Clayton         }
1873b9c1b51eSKate Stone       } else {
187444d93782SGreg Clayton         HandleCharResult key_result = m_window_sp->HandleChar(ch);
1875b9c1b51eSKate Stone         switch (key_result) {
187644d93782SGreg Clayton         case eKeyHandled:
1877c5dac77aSEugene Zelenko           debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
187844d93782SGreg Clayton           update = true;
187944d93782SGreg Clayton           break;
188044d93782SGreg Clayton         case eKeyNotHandled:
188144d93782SGreg Clayton           break;
188244d93782SGreg Clayton         case eQuitApplication:
188344d93782SGreg Clayton           done = true;
188444d93782SGreg Clayton           break;
188544d93782SGreg Clayton         }
188644d93782SGreg Clayton       }
188744d93782SGreg Clayton     }
188844d93782SGreg Clayton 
188944d93782SGreg Clayton     debugger.CancelForwardEvents(listener_sp);
189044d93782SGreg Clayton   }
189144d93782SGreg Clayton 
1892b9c1b51eSKate Stone   WindowSP &GetMainWindow() {
189344d93782SGreg Clayton     if (!m_window_sp)
1894796ac80bSJonas Devlieghere       m_window_sp = std::make_shared<Window>("main", stdscr, false);
189544d93782SGreg Clayton     return m_window_sp;
189644d93782SGreg Clayton   }
189744d93782SGreg Clayton 
1898b9c1b51eSKate Stone   WindowDelegates &GetWindowDelegates() { return m_window_delegates; }
189944d93782SGreg Clayton 
190044d93782SGreg Clayton protected:
190144d93782SGreg Clayton   WindowSP m_window_sp;
190244d93782SGreg Clayton   WindowDelegates m_window_delegates;
190344d93782SGreg Clayton   SCREEN *m_screen;
190444d93782SGreg Clayton   FILE *m_in;
190544d93782SGreg Clayton   FILE *m_out;
190644d93782SGreg Clayton };
190744d93782SGreg Clayton 
190844d93782SGreg Clayton } // namespace curses
190944d93782SGreg Clayton 
191044d93782SGreg Clayton using namespace curses;
191144d93782SGreg Clayton 
1912b9c1b51eSKate Stone struct Row {
19138369b28dSGreg Clayton   ValueObjectManager value;
191444d93782SGreg Clayton   Row *parent;
19158369b28dSGreg Clayton   // The process stop ID when the children were calculated.
19168369b28dSGreg Clayton   uint32_t children_stop_id;
191744d93782SGreg Clayton   int row_idx;
191844d93782SGreg Clayton   int x;
191944d93782SGreg Clayton   int y;
192044d93782SGreg Clayton   bool might_have_children;
192144d93782SGreg Clayton   bool expanded;
192244d93782SGreg Clayton   bool calculated_children;
192344d93782SGreg Clayton   std::vector<Row> children;
192444d93782SGreg Clayton 
1925b9c1b51eSKate Stone   Row(const ValueObjectSP &v, Row *p)
19268369b28dSGreg Clayton       : value(v, lldb::eDynamicDontRunTarget, true), parent(p), row_idx(0),
19278369b28dSGreg Clayton         x(1), y(1), might_have_children(v ? v->MightHaveChildren() : false),
1928b9c1b51eSKate Stone         expanded(false), calculated_children(false), children() {}
192944d93782SGreg Clayton 
1930b9c1b51eSKate Stone   size_t GetDepth() const {
193144d93782SGreg Clayton     if (parent)
193244d93782SGreg Clayton       return 1 + parent->GetDepth();
193344d93782SGreg Clayton     return 0;
193444d93782SGreg Clayton   }
193544d93782SGreg Clayton 
1936b9c1b51eSKate Stone   void Expand() {
193744d93782SGreg Clayton     expanded = true;
19388369b28dSGreg Clayton   }
19398369b28dSGreg Clayton 
19408369b28dSGreg Clayton   std::vector<Row> &GetChildren() {
19418369b28dSGreg Clayton     ProcessSP process_sp = value.GetProcessSP();
19428369b28dSGreg Clayton     auto stop_id = process_sp->GetStopID();
19438369b28dSGreg Clayton     if (process_sp && stop_id != children_stop_id) {
19448369b28dSGreg Clayton       children_stop_id = stop_id;
19458369b28dSGreg Clayton       calculated_children = false;
19468369b28dSGreg Clayton     }
1947b9c1b51eSKate Stone     if (!calculated_children) {
19488369b28dSGreg Clayton       children.clear();
194944d93782SGreg Clayton       calculated_children = true;
19508369b28dSGreg Clayton       ValueObjectSP valobj = value.GetSP();
1951b9c1b51eSKate Stone       if (valobj) {
195244d93782SGreg Clayton         const size_t num_children = valobj->GetNumChildren();
1953b9c1b51eSKate Stone         for (size_t i = 0; i < num_children; ++i) {
195444d93782SGreg Clayton           children.push_back(Row(valobj->GetChildAtIndex(i, true), this));
195544d93782SGreg Clayton         }
195644d93782SGreg Clayton       }
195744d93782SGreg Clayton     }
19588369b28dSGreg Clayton     return children;
195944d93782SGreg Clayton   }
196044d93782SGreg Clayton 
19618369b28dSGreg Clayton   void Unexpand() {
19628369b28dSGreg Clayton     expanded = false;
19638369b28dSGreg Clayton     calculated_children = false;
19648369b28dSGreg Clayton     children.clear();
19658369b28dSGreg Clayton   }
196644d93782SGreg Clayton 
1967b9c1b51eSKate Stone   void DrawTree(Window &window) {
196844d93782SGreg Clayton     if (parent)
196944d93782SGreg Clayton       parent->DrawTreeForChild(window, this, 0);
197044d93782SGreg Clayton 
1971b9c1b51eSKate Stone     if (might_have_children) {
197205097246SAdrian Prantl       // It we can get UTF8 characters to work we should try to use the
197305097246SAdrian Prantl       // "symbol" UTF8 string below
197444d93782SGreg Clayton       //            const char *symbol = "";
197544d93782SGreg Clayton       //            if (row.expanded)
197644d93782SGreg Clayton       //                symbol = "\xe2\x96\xbd ";
197744d93782SGreg Clayton       //            else
197844d93782SGreg Clayton       //                symbol = "\xe2\x96\xb7 ";
197944d93782SGreg Clayton       //            window.PutCString (symbol);
198044d93782SGreg Clayton 
198105097246SAdrian Prantl       // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 'v'
198205097246SAdrian Prantl       // or '>' character...
198344d93782SGreg Clayton       //            if (expanded)
198444d93782SGreg Clayton       //                window.PutChar (ACS_DARROW);
198544d93782SGreg Clayton       //            else
198644d93782SGreg Clayton       //                window.PutChar (ACS_RARROW);
198705097246SAdrian Prantl       // Since we can't find any good looking right arrow/down arrow symbols,
198805097246SAdrian Prantl       // just use a diamond...
198944d93782SGreg Clayton       window.PutChar(ACS_DIAMOND);
199044d93782SGreg Clayton       window.PutChar(ACS_HLINE);
199144d93782SGreg Clayton     }
199244d93782SGreg Clayton   }
199344d93782SGreg Clayton 
1994b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, Row *child, uint32_t reverse_depth) {
199544d93782SGreg Clayton     if (parent)
199644d93782SGreg Clayton       parent->DrawTreeForChild(window, this, reverse_depth + 1);
199744d93782SGreg Clayton 
19988369b28dSGreg Clayton     if (&GetChildren().back() == child) {
199944d93782SGreg Clayton       // Last child
2000b9c1b51eSKate Stone       if (reverse_depth == 0) {
200144d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
200244d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2003b9c1b51eSKate Stone       } else {
200444d93782SGreg Clayton         window.PutChar(' ');
200544d93782SGreg Clayton         window.PutChar(' ');
200644d93782SGreg Clayton       }
2007b9c1b51eSKate Stone     } else {
2008b9c1b51eSKate Stone       if (reverse_depth == 0) {
200944d93782SGreg Clayton         window.PutChar(ACS_LTEE);
201044d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2011b9c1b51eSKate Stone       } else {
201244d93782SGreg Clayton         window.PutChar(ACS_VLINE);
201344d93782SGreg Clayton         window.PutChar(' ');
201444d93782SGreg Clayton       }
201544d93782SGreg Clayton     }
201644d93782SGreg Clayton   }
201744d93782SGreg Clayton };
201844d93782SGreg Clayton 
2019b9c1b51eSKate Stone struct DisplayOptions {
202044d93782SGreg Clayton   bool show_types;
202144d93782SGreg Clayton };
202244d93782SGreg Clayton 
202344d93782SGreg Clayton class TreeItem;
202444d93782SGreg Clayton 
2025b9c1b51eSKate Stone class TreeDelegate {
202644d93782SGreg Clayton public:
2027c5dac77aSEugene Zelenko   TreeDelegate() = default;
2028315b6884SEugene Zelenko   virtual ~TreeDelegate() = default;
2029315b6884SEugene Zelenko 
203044d93782SGreg Clayton   virtual void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) = 0;
203144d93782SGreg Clayton   virtual void TreeDelegateGenerateChildren(TreeItem &item) = 0;
2032b9c1b51eSKate Stone   virtual bool TreeDelegateItemSelected(
2033b9c1b51eSKate Stone       TreeItem &item) = 0; // Return true if we need to update views
203444d93782SGreg Clayton };
2035315b6884SEugene Zelenko 
203644d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
203744d93782SGreg Clayton 
2038b9c1b51eSKate Stone class TreeItem {
203944d93782SGreg Clayton public:
2040b9c1b51eSKate Stone   TreeItem(TreeItem *parent, TreeDelegate &delegate, bool might_have_children)
2041b9c1b51eSKate Stone       : m_parent(parent), m_delegate(delegate), m_user_data(nullptr),
2042b9c1b51eSKate Stone         m_identifier(0), m_row_idx(-1), m_children(),
2043b9c1b51eSKate Stone         m_might_have_children(might_have_children), m_is_expanded(false) {}
204444d93782SGreg Clayton 
2045b9c1b51eSKate Stone   TreeItem &operator=(const TreeItem &rhs) {
2046b9c1b51eSKate Stone     if (this != &rhs) {
204744d93782SGreg Clayton       m_parent = rhs.m_parent;
204844d93782SGreg Clayton       m_delegate = rhs.m_delegate;
2049ec990867SGreg Clayton       m_user_data = rhs.m_user_data;
205044d93782SGreg Clayton       m_identifier = rhs.m_identifier;
205144d93782SGreg Clayton       m_row_idx = rhs.m_row_idx;
205244d93782SGreg Clayton       m_children = rhs.m_children;
205344d93782SGreg Clayton       m_might_have_children = rhs.m_might_have_children;
205444d93782SGreg Clayton       m_is_expanded = rhs.m_is_expanded;
205544d93782SGreg Clayton     }
205644d93782SGreg Clayton     return *this;
205744d93782SGreg Clayton   }
205844d93782SGreg Clayton 
2059b9c1b51eSKate Stone   size_t GetDepth() const {
206044d93782SGreg Clayton     if (m_parent)
206144d93782SGreg Clayton       return 1 + m_parent->GetDepth();
206244d93782SGreg Clayton     return 0;
206344d93782SGreg Clayton   }
206444d93782SGreg Clayton 
2065b9c1b51eSKate Stone   int GetRowIndex() const { return m_row_idx; }
206644d93782SGreg Clayton 
2067b9c1b51eSKate Stone   void ClearChildren() { m_children.clear(); }
206844d93782SGreg Clayton 
2069b9c1b51eSKate Stone   void Resize(size_t n, const TreeItem &t) { m_children.resize(n, t); }
207044d93782SGreg Clayton 
2071b9c1b51eSKate Stone   TreeItem &operator[](size_t i) { return m_children[i]; }
207244d93782SGreg Clayton 
2073b9c1b51eSKate Stone   void SetRowIndex(int row_idx) { m_row_idx = row_idx; }
207444d93782SGreg Clayton 
2075b9c1b51eSKate Stone   size_t GetNumChildren() {
207644d93782SGreg Clayton     m_delegate.TreeDelegateGenerateChildren(*this);
207744d93782SGreg Clayton     return m_children.size();
207844d93782SGreg Clayton   }
207944d93782SGreg Clayton 
2080b9c1b51eSKate Stone   void ItemWasSelected() { m_delegate.TreeDelegateItemSelected(*this); }
2081315b6884SEugene Zelenko 
2082b9c1b51eSKate Stone   void CalculateRowIndexes(int &row_idx) {
208344d93782SGreg Clayton     SetRowIndex(row_idx);
208444d93782SGreg Clayton     ++row_idx;
208544d93782SGreg Clayton 
2086ec990867SGreg Clayton     const bool expanded = IsExpanded();
2087ec990867SGreg Clayton 
208805097246SAdrian Prantl     // The root item must calculate its children, or we must calculate the
208905097246SAdrian Prantl     // number of children if the item is expanded
2090c5dac77aSEugene Zelenko     if (m_parent == nullptr || expanded)
209144d93782SGreg Clayton       GetNumChildren();
209244d93782SGreg Clayton 
2093b9c1b51eSKate Stone     for (auto &item : m_children) {
209444d93782SGreg Clayton       if (expanded)
209544d93782SGreg Clayton         item.CalculateRowIndexes(row_idx);
209644d93782SGreg Clayton       else
209744d93782SGreg Clayton         item.SetRowIndex(-1);
209844d93782SGreg Clayton     }
209944d93782SGreg Clayton   }
210044d93782SGreg Clayton 
2101b9c1b51eSKate Stone   TreeItem *GetParent() { return m_parent; }
210244d93782SGreg Clayton 
2103b9c1b51eSKate Stone   bool IsExpanded() const { return m_is_expanded; }
210444d93782SGreg Clayton 
2105b9c1b51eSKate Stone   void Expand() { m_is_expanded = true; }
210644d93782SGreg Clayton 
2107b9c1b51eSKate Stone   void Unexpand() { m_is_expanded = false; }
210844d93782SGreg Clayton 
2109b9c1b51eSKate Stone   bool Draw(Window &window, const int first_visible_row,
2110b9c1b51eSKate Stone             const uint32_t selected_row_idx, int &row_idx, int &num_rows_left) {
211144d93782SGreg Clayton     if (num_rows_left <= 0)
211244d93782SGreg Clayton       return false;
211344d93782SGreg Clayton 
2114b9c1b51eSKate Stone     if (m_row_idx >= first_visible_row) {
211544d93782SGreg Clayton       window.MoveCursor(2, row_idx + 1);
211644d93782SGreg Clayton 
211744d93782SGreg Clayton       if (m_parent)
211844d93782SGreg Clayton         m_parent->DrawTreeForChild(window, this, 0);
211944d93782SGreg Clayton 
2120b9c1b51eSKate Stone       if (m_might_have_children) {
2121b9c1b51eSKate Stone         // It we can get UTF8 characters to work we should try to use the
212205097246SAdrian Prantl         // "symbol" UTF8 string below
212344d93782SGreg Clayton         //            const char *symbol = "";
212444d93782SGreg Clayton         //            if (row.expanded)
212544d93782SGreg Clayton         //                symbol = "\xe2\x96\xbd ";
212644d93782SGreg Clayton         //            else
212744d93782SGreg Clayton         //                symbol = "\xe2\x96\xb7 ";
212844d93782SGreg Clayton         //            window.PutCString (symbol);
212944d93782SGreg Clayton 
213044d93782SGreg Clayton         // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
213144d93782SGreg Clayton         // 'v' or '>' character...
213244d93782SGreg Clayton         //            if (expanded)
213344d93782SGreg Clayton         //                window.PutChar (ACS_DARROW);
213444d93782SGreg Clayton         //            else
213544d93782SGreg Clayton         //                window.PutChar (ACS_RARROW);
213605097246SAdrian Prantl         // Since we can't find any good looking right arrow/down arrow symbols,
213705097246SAdrian Prantl         // just use a diamond...
213844d93782SGreg Clayton         window.PutChar(ACS_DIAMOND);
213944d93782SGreg Clayton         window.PutChar(ACS_HLINE);
214044d93782SGreg Clayton       }
2141b9c1b51eSKate Stone       bool highlight = (selected_row_idx == static_cast<size_t>(m_row_idx)) &&
2142b9c1b51eSKate Stone                        window.IsActive();
214344d93782SGreg Clayton 
214444d93782SGreg Clayton       if (highlight)
214544d93782SGreg Clayton         window.AttributeOn(A_REVERSE);
214644d93782SGreg Clayton 
214744d93782SGreg Clayton       m_delegate.TreeDelegateDrawTreeItem(*this, window);
214844d93782SGreg Clayton 
214944d93782SGreg Clayton       if (highlight)
215044d93782SGreg Clayton         window.AttributeOff(A_REVERSE);
215144d93782SGreg Clayton       ++row_idx;
215244d93782SGreg Clayton       --num_rows_left;
215344d93782SGreg Clayton     }
215444d93782SGreg Clayton 
215544d93782SGreg Clayton     if (num_rows_left <= 0)
215644d93782SGreg Clayton       return false; // We are done drawing...
215744d93782SGreg Clayton 
2158b9c1b51eSKate Stone     if (IsExpanded()) {
2159b9c1b51eSKate Stone       for (auto &item : m_children) {
216005097246SAdrian Prantl         // If we displayed all the rows and item.Draw() returns false we are
216105097246SAdrian Prantl         // done drawing and can exit this for loop
2162b9c1b51eSKate Stone         if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx,
2163b9c1b51eSKate Stone                        num_rows_left))
216444d93782SGreg Clayton           break;
216544d93782SGreg Clayton       }
216644d93782SGreg Clayton     }
216744d93782SGreg Clayton     return num_rows_left >= 0; // Return true if not done drawing yet
216844d93782SGreg Clayton   }
216944d93782SGreg Clayton 
2170b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, TreeItem *child,
2171b9c1b51eSKate Stone                         uint32_t reverse_depth) {
217244d93782SGreg Clayton     if (m_parent)
217344d93782SGreg Clayton       m_parent->DrawTreeForChild(window, this, reverse_depth + 1);
217444d93782SGreg Clayton 
2175b9c1b51eSKate Stone     if (&m_children.back() == child) {
217644d93782SGreg Clayton       // Last child
2177b9c1b51eSKate Stone       if (reverse_depth == 0) {
217844d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
217944d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2180b9c1b51eSKate Stone       } else {
218144d93782SGreg Clayton         window.PutChar(' ');
218244d93782SGreg Clayton         window.PutChar(' ');
218344d93782SGreg Clayton       }
2184b9c1b51eSKate Stone     } else {
2185b9c1b51eSKate Stone       if (reverse_depth == 0) {
218644d93782SGreg Clayton         window.PutChar(ACS_LTEE);
218744d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2188b9c1b51eSKate Stone       } else {
218944d93782SGreg Clayton         window.PutChar(ACS_VLINE);
219044d93782SGreg Clayton         window.PutChar(' ');
219144d93782SGreg Clayton       }
219244d93782SGreg Clayton     }
219344d93782SGreg Clayton   }
219444d93782SGreg Clayton 
2195b9c1b51eSKate Stone   TreeItem *GetItemForRowIndex(uint32_t row_idx) {
21963985c8c6SSaleem Abdulrasool     if (static_cast<uint32_t>(m_row_idx) == row_idx)
219744d93782SGreg Clayton       return this;
219844d93782SGreg Clayton     if (m_children.empty())
2199c5dac77aSEugene Zelenko       return nullptr;
2200b9c1b51eSKate Stone     if (IsExpanded()) {
2201b9c1b51eSKate Stone       for (auto &item : m_children) {
220244d93782SGreg Clayton         TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
220344d93782SGreg Clayton         if (selected_item_ptr)
220444d93782SGreg Clayton           return selected_item_ptr;
220544d93782SGreg Clayton       }
220644d93782SGreg Clayton     }
2207c5dac77aSEugene Zelenko     return nullptr;
220844d93782SGreg Clayton   }
220944d93782SGreg Clayton 
2210b9c1b51eSKate Stone   void *GetUserData() const { return m_user_data; }
2211ec990867SGreg Clayton 
2212b9c1b51eSKate Stone   void SetUserData(void *user_data) { m_user_data = user_data; }
2213ec990867SGreg Clayton 
2214b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
221544d93782SGreg Clayton 
2216b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
221744d93782SGreg Clayton 
2218b9c1b51eSKate Stone   void SetMightHaveChildren(bool b) { m_might_have_children = b; }
2219ec990867SGreg Clayton 
222044d93782SGreg Clayton protected:
222144d93782SGreg Clayton   TreeItem *m_parent;
222244d93782SGreg Clayton   TreeDelegate &m_delegate;
2223ec990867SGreg Clayton   void *m_user_data;
222444d93782SGreg Clayton   uint64_t m_identifier;
2225b9c1b51eSKate Stone   int m_row_idx; // Zero based visible row index, -1 if not visible or for the
2226b9c1b51eSKate Stone                  // root item
222744d93782SGreg Clayton   std::vector<TreeItem> m_children;
222844d93782SGreg Clayton   bool m_might_have_children;
222944d93782SGreg Clayton   bool m_is_expanded;
223044d93782SGreg Clayton };
223144d93782SGreg Clayton 
2232b9c1b51eSKate Stone class TreeWindowDelegate : public WindowDelegate {
223344d93782SGreg Clayton public:
2234b9c1b51eSKate Stone   TreeWindowDelegate(Debugger &debugger, const TreeDelegateSP &delegate_sp)
2235b9c1b51eSKate Stone       : m_debugger(debugger), m_delegate_sp(delegate_sp),
2236b9c1b51eSKate Stone         m_root(nullptr, *delegate_sp, true), m_selected_item(nullptr),
2237b9c1b51eSKate Stone         m_num_rows(0), m_selected_row_idx(0), m_first_visible_row(0),
2238b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
223944d93782SGreg Clayton 
2240b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
224144d93782SGreg Clayton 
2242b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2243b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2244b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
224544d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
224644d93782SGreg Clayton 
224744d93782SGreg Clayton     bool display_content = false;
2248b9c1b51eSKate Stone     if (process) {
224944d93782SGreg Clayton       StateType state = process->GetState();
2250b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
225144d93782SGreg Clayton         // We are stopped, so it is ok to
225244d93782SGreg Clayton         display_content = true;
2253b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
225444d93782SGreg Clayton         return true; // Don't do any updating when we are running
225544d93782SGreg Clayton       }
225644d93782SGreg Clayton     }
225744d93782SGreg Clayton 
225844d93782SGreg Clayton     m_min_x = 2;
225944d93782SGreg Clayton     m_min_y = 1;
226044d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
226144d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
226244d93782SGreg Clayton 
226344d93782SGreg Clayton     window.Erase();
226444d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
226544d93782SGreg Clayton 
2266b9c1b51eSKate Stone     if (display_content) {
226744d93782SGreg Clayton       const int num_visible_rows = NumVisibleRows();
226844d93782SGreg Clayton       m_num_rows = 0;
226944d93782SGreg Clayton       m_root.CalculateRowIndexes(m_num_rows);
227044d93782SGreg Clayton 
227105097246SAdrian Prantl       // If we unexpanded while having something selected our total number of
227205097246SAdrian Prantl       // rows is less than the num visible rows, then make sure we show all the
227305097246SAdrian Prantl       // rows by setting the first visible row accordingly.
227444d93782SGreg Clayton       if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
227544d93782SGreg Clayton         m_first_visible_row = 0;
227644d93782SGreg Clayton 
227744d93782SGreg Clayton       // Make sure the selected row is always visible
227844d93782SGreg Clayton       if (m_selected_row_idx < m_first_visible_row)
227944d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx;
228044d93782SGreg Clayton       else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
228144d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
228244d93782SGreg Clayton 
228344d93782SGreg Clayton       int row_idx = 0;
228444d93782SGreg Clayton       int num_rows_left = num_visible_rows;
2285b9c1b51eSKate Stone       m_root.Draw(window, m_first_visible_row, m_selected_row_idx, row_idx,
2286b9c1b51eSKate Stone                   num_rows_left);
228744d93782SGreg Clayton       // Get the selected row
228844d93782SGreg Clayton       m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
2289b9c1b51eSKate Stone     } else {
2290c5dac77aSEugene Zelenko       m_selected_item = nullptr;
229144d93782SGreg Clayton     }
229244d93782SGreg Clayton 
229344d93782SGreg Clayton     window.DeferredRefresh();
229444d93782SGreg Clayton 
229544d93782SGreg Clayton     return true; // Drawing handled
229644d93782SGreg Clayton   }
229744d93782SGreg Clayton 
2298b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
229944d93782SGreg Clayton     return "Thread window keyboard shortcuts:";
230044d93782SGreg Clayton   }
230144d93782SGreg Clayton 
2302b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
230344d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
230444d93782SGreg Clayton         {KEY_UP, "Select previous item"},
230544d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
230644d93782SGreg Clayton         {KEY_RIGHT, "Expand the selected item"},
2307b9c1b51eSKate Stone         {KEY_LEFT,
2308b9c1b51eSKate Stone          "Unexpand the selected item or select parent if not expanded"},
230944d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
231044d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
231144d93782SGreg Clayton         {'h', "Show help dialog"},
231244d93782SGreg Clayton         {' ', "Toggle item expansion"},
231344d93782SGreg Clayton         {',', "Page up"},
231444d93782SGreg Clayton         {'.', "Page down"},
2315b9c1b51eSKate Stone         {'\0', nullptr}};
231644d93782SGreg Clayton     return g_source_view_key_help;
231744d93782SGreg Clayton   }
231844d93782SGreg Clayton 
2319b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2320b9c1b51eSKate Stone     switch (c) {
232144d93782SGreg Clayton     case ',':
232244d93782SGreg Clayton     case KEY_PPAGE:
232344d93782SGreg Clayton       // Page up key
2324b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
232544d93782SGreg Clayton         if (m_first_visible_row > m_max_y)
232644d93782SGreg Clayton           m_first_visible_row -= m_max_y;
232744d93782SGreg Clayton         else
232844d93782SGreg Clayton           m_first_visible_row = 0;
232944d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
233044d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
233144d93782SGreg Clayton         if (m_selected_item)
233244d93782SGreg Clayton           m_selected_item->ItemWasSelected();
233344d93782SGreg Clayton       }
233444d93782SGreg Clayton       return eKeyHandled;
233544d93782SGreg Clayton 
233644d93782SGreg Clayton     case '.':
233744d93782SGreg Clayton     case KEY_NPAGE:
233844d93782SGreg Clayton       // Page down key
2339b9c1b51eSKate Stone       if (m_num_rows > m_max_y) {
2340b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
234144d93782SGreg Clayton           m_first_visible_row += m_max_y;
234244d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
234344d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
234444d93782SGreg Clayton           if (m_selected_item)
234544d93782SGreg Clayton             m_selected_item->ItemWasSelected();
234644d93782SGreg Clayton         }
234744d93782SGreg Clayton       }
234844d93782SGreg Clayton       return eKeyHandled;
234944d93782SGreg Clayton 
235044d93782SGreg Clayton     case KEY_UP:
2351b9c1b51eSKate Stone       if (m_selected_row_idx > 0) {
235244d93782SGreg Clayton         --m_selected_row_idx;
235344d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
235444d93782SGreg Clayton         if (m_selected_item)
235544d93782SGreg Clayton           m_selected_item->ItemWasSelected();
235644d93782SGreg Clayton       }
235744d93782SGreg Clayton       return eKeyHandled;
2358315b6884SEugene Zelenko 
235944d93782SGreg Clayton     case KEY_DOWN:
2360b9c1b51eSKate Stone       if (m_selected_row_idx + 1 < m_num_rows) {
236144d93782SGreg Clayton         ++m_selected_row_idx;
236244d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
236344d93782SGreg Clayton         if (m_selected_item)
236444d93782SGreg Clayton           m_selected_item->ItemWasSelected();
236544d93782SGreg Clayton       }
236644d93782SGreg Clayton       return eKeyHandled;
236744d93782SGreg Clayton 
236844d93782SGreg Clayton     case KEY_RIGHT:
2369b9c1b51eSKate Stone       if (m_selected_item) {
237044d93782SGreg Clayton         if (!m_selected_item->IsExpanded())
237144d93782SGreg Clayton           m_selected_item->Expand();
237244d93782SGreg Clayton       }
237344d93782SGreg Clayton       return eKeyHandled;
237444d93782SGreg Clayton 
237544d93782SGreg Clayton     case KEY_LEFT:
2376b9c1b51eSKate Stone       if (m_selected_item) {
237744d93782SGreg Clayton         if (m_selected_item->IsExpanded())
237844d93782SGreg Clayton           m_selected_item->Unexpand();
2379b9c1b51eSKate Stone         else if (m_selected_item->GetParent()) {
238044d93782SGreg Clayton           m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
238144d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
238244d93782SGreg Clayton           if (m_selected_item)
238344d93782SGreg Clayton             m_selected_item->ItemWasSelected();
238444d93782SGreg Clayton         }
238544d93782SGreg Clayton       }
238644d93782SGreg Clayton       return eKeyHandled;
238744d93782SGreg Clayton 
238844d93782SGreg Clayton     case ' ':
238944d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2390b9c1b51eSKate Stone       if (m_selected_item) {
239144d93782SGreg Clayton         if (m_selected_item->IsExpanded())
239244d93782SGreg Clayton           m_selected_item->Unexpand();
239344d93782SGreg Clayton         else
239444d93782SGreg Clayton           m_selected_item->Expand();
239544d93782SGreg Clayton       }
239644d93782SGreg Clayton       return eKeyHandled;
239744d93782SGreg Clayton 
239844d93782SGreg Clayton     case 'h':
239944d93782SGreg Clayton       window.CreateHelpSubwindow();
240044d93782SGreg Clayton       return eKeyHandled;
240144d93782SGreg Clayton 
240244d93782SGreg Clayton     default:
240344d93782SGreg Clayton       break;
240444d93782SGreg Clayton     }
240544d93782SGreg Clayton     return eKeyNotHandled;
240644d93782SGreg Clayton   }
240744d93782SGreg Clayton 
240844d93782SGreg Clayton protected:
240944d93782SGreg Clayton   Debugger &m_debugger;
241044d93782SGreg Clayton   TreeDelegateSP m_delegate_sp;
241144d93782SGreg Clayton   TreeItem m_root;
241244d93782SGreg Clayton   TreeItem *m_selected_item;
241344d93782SGreg Clayton   int m_num_rows;
241444d93782SGreg Clayton   int m_selected_row_idx;
241544d93782SGreg Clayton   int m_first_visible_row;
241644d93782SGreg Clayton   int m_min_x;
241744d93782SGreg Clayton   int m_min_y;
241844d93782SGreg Clayton   int m_max_x;
241944d93782SGreg Clayton   int m_max_y;
242044d93782SGreg Clayton };
242144d93782SGreg Clayton 
2422b9c1b51eSKate Stone class FrameTreeDelegate : public TreeDelegate {
242344d93782SGreg Clayton public:
2424b9c1b51eSKate Stone   FrameTreeDelegate() : TreeDelegate() {
2425b9c1b51eSKate Stone     FormatEntity::Parse(
2426b9c1b51eSKate Stone         "frame #${frame.index}: {${function.name}${function.pc-offset}}}",
2427554f68d3SGreg Clayton         m_format);
242844d93782SGreg Clayton   }
242944d93782SGreg Clayton 
2430315b6884SEugene Zelenko   ~FrameTreeDelegate() override = default;
243144d93782SGreg Clayton 
2432b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2433ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2434b9c1b51eSKate Stone     if (thread) {
243544d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2436ec990867SGreg Clayton       StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
2437b9c1b51eSKate Stone       if (frame_sp) {
243844d93782SGreg Clayton         StreamString strm;
2439b9c1b51eSKate Stone         const SymbolContext &sc =
2440b9c1b51eSKate Stone             frame_sp->GetSymbolContext(eSymbolContextEverything);
244144d93782SGreg Clayton         ExecutionContext exe_ctx(frame_sp);
2442b9c1b51eSKate Stone         if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr,
2443b9c1b51eSKate Stone                                  nullptr, false, false)) {
244444d93782SGreg Clayton           int right_pad = 1;
2445c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
244644d93782SGreg Clayton         }
244744d93782SGreg Clayton       }
244844d93782SGreg Clayton     }
244944d93782SGreg Clayton   }
2450315b6884SEugene Zelenko 
2451b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
245244d93782SGreg Clayton     // No children for frames yet...
245344d93782SGreg Clayton   }
245444d93782SGreg Clayton 
2455b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2456ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2457b9c1b51eSKate Stone     if (thread) {
2458b9c1b51eSKate Stone       thread->GetProcess()->GetThreadList().SetSelectedThreadByID(
2459b9c1b51eSKate Stone           thread->GetID());
246044d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2461ec990867SGreg Clayton       thread->SetSelectedFrameByIndex(frame_idx);
246244d93782SGreg Clayton       return true;
246344d93782SGreg Clayton     }
246444d93782SGreg Clayton     return false;
246544d93782SGreg Clayton   }
2466315b6884SEugene Zelenko 
2467554f68d3SGreg Clayton protected:
2468554f68d3SGreg Clayton   FormatEntity::Entry m_format;
246944d93782SGreg Clayton };
247044d93782SGreg Clayton 
2471b9c1b51eSKate Stone class ThreadTreeDelegate : public TreeDelegate {
247244d93782SGreg Clayton public:
2473b9c1b51eSKate Stone   ThreadTreeDelegate(Debugger &debugger)
2474b9c1b51eSKate Stone       : TreeDelegate(), m_debugger(debugger), m_tid(LLDB_INVALID_THREAD_ID),
2475b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2476b9c1b51eSKate Stone     FormatEntity::Parse("thread #${thread.index}: tid = ${thread.id}{, stop "
2477b9c1b51eSKate Stone                         "reason = ${thread.stop-reason}}",
2478554f68d3SGreg Clayton                         m_format);
247944d93782SGreg Clayton   }
248044d93782SGreg Clayton 
2481315b6884SEugene Zelenko   ~ThreadTreeDelegate() override = default;
248244d93782SGreg Clayton 
2483b9c1b51eSKate Stone   ProcessSP GetProcess() {
2484b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2485b9c1b51eSKate Stone         .GetExecutionContext()
2486b9c1b51eSKate Stone         .GetProcessSP();
2487ec990867SGreg Clayton   }
2488ec990867SGreg Clayton 
2489b9c1b51eSKate Stone   ThreadSP GetThread(const TreeItem &item) {
2490ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2491ec990867SGreg Clayton     if (process_sp)
2492ec990867SGreg Clayton       return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
2493ec990867SGreg Clayton     return ThreadSP();
2494ec990867SGreg Clayton   }
2495ec990867SGreg Clayton 
2496b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2497ec990867SGreg Clayton     ThreadSP thread_sp = GetThread(item);
2498b9c1b51eSKate Stone     if (thread_sp) {
249944d93782SGreg Clayton       StreamString strm;
250044d93782SGreg Clayton       ExecutionContext exe_ctx(thread_sp);
2501b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2502b9c1b51eSKate Stone                                nullptr, false, false)) {
250344d93782SGreg Clayton         int right_pad = 1;
2504c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
250544d93782SGreg Clayton       }
250644d93782SGreg Clayton     }
250744d93782SGreg Clayton   }
2508315b6884SEugene Zelenko 
2509b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2510ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2511b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
251244d93782SGreg Clayton       StateType state = process_sp->GetState();
2513b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2514ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2515b9c1b51eSKate Stone         if (thread_sp) {
2516b9c1b51eSKate Stone           if (m_stop_id == process_sp->GetStopID() &&
2517b9c1b51eSKate Stone               thread_sp->GetID() == m_tid)
251844d93782SGreg Clayton             return; // Children are already up to date
2519b9c1b51eSKate Stone           if (!m_frame_delegate_sp) {
252044d93782SGreg Clayton             // Always expand the thread item the first time we show it
2521796ac80bSJonas Devlieghere             m_frame_delegate_sp = std::make_shared<FrameTreeDelegate>();
252244d93782SGreg Clayton           }
252344d93782SGreg Clayton 
252444d93782SGreg Clayton           m_stop_id = process_sp->GetStopID();
252544d93782SGreg Clayton           m_tid = thread_sp->GetID();
252644d93782SGreg Clayton 
252744d93782SGreg Clayton           TreeItem t(&item, *m_frame_delegate_sp, false);
252844d93782SGreg Clayton           size_t num_frames = thread_sp->GetStackFrameCount();
252944d93782SGreg Clayton           item.Resize(num_frames, t);
2530b9c1b51eSKate Stone           for (size_t i = 0; i < num_frames; ++i) {
2531ec990867SGreg Clayton             item[i].SetUserData(thread_sp.get());
253244d93782SGreg Clayton             item[i].SetIdentifier(i);
253344d93782SGreg Clayton           }
253444d93782SGreg Clayton         }
253544d93782SGreg Clayton         return;
253644d93782SGreg Clayton       }
253744d93782SGreg Clayton     }
253844d93782SGreg Clayton     item.ClearChildren();
253944d93782SGreg Clayton   }
254044d93782SGreg Clayton 
2541b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2542ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2543b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2544ec990867SGreg Clayton       StateType state = process_sp->GetState();
2545b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2546ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2547b9c1b51eSKate Stone         if (thread_sp) {
254844d93782SGreg Clayton           ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
2549bb19a13cSSaleem Abdulrasool           std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
255044d93782SGreg Clayton           ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
2551b9c1b51eSKate Stone           if (selected_thread_sp->GetID() != thread_sp->GetID()) {
255244d93782SGreg Clayton             thread_list.SetSelectedThreadByID(thread_sp->GetID());
255344d93782SGreg Clayton             return true;
255444d93782SGreg Clayton           }
255544d93782SGreg Clayton         }
2556ec990867SGreg Clayton       }
2557ec990867SGreg Clayton     }
255844d93782SGreg Clayton     return false;
255944d93782SGreg Clayton   }
256044d93782SGreg Clayton 
256144d93782SGreg Clayton protected:
256244d93782SGreg Clayton   Debugger &m_debugger;
256344d93782SGreg Clayton   std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
256444d93782SGreg Clayton   lldb::user_id_t m_tid;
256544d93782SGreg Clayton   uint32_t m_stop_id;
2566554f68d3SGreg Clayton   FormatEntity::Entry m_format;
256744d93782SGreg Clayton };
256844d93782SGreg Clayton 
2569b9c1b51eSKate Stone class ThreadsTreeDelegate : public TreeDelegate {
2570ec990867SGreg Clayton public:
2571b9c1b51eSKate Stone   ThreadsTreeDelegate(Debugger &debugger)
2572b9c1b51eSKate Stone       : TreeDelegate(), m_thread_delegate_sp(), m_debugger(debugger),
2573b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2574554f68d3SGreg Clayton     FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
2575554f68d3SGreg Clayton                         m_format);
2576ec990867SGreg Clayton   }
2577ec990867SGreg Clayton 
2578315b6884SEugene Zelenko   ~ThreadsTreeDelegate() override = default;
2579ec990867SGreg Clayton 
2580b9c1b51eSKate Stone   ProcessSP GetProcess() {
2581b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2582b9c1b51eSKate Stone         .GetExecutionContext()
2583b9c1b51eSKate Stone         .GetProcessSP();
2584ec990867SGreg Clayton   }
2585ec990867SGreg Clayton 
2586b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2587ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2588b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2589ec990867SGreg Clayton       StreamString strm;
2590ec990867SGreg Clayton       ExecutionContext exe_ctx(process_sp);
2591b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2592b9c1b51eSKate Stone                                nullptr, false, false)) {
2593ec990867SGreg Clayton         int right_pad = 1;
2594c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
2595ec990867SGreg Clayton       }
2596ec990867SGreg Clayton     }
2597ec990867SGreg Clayton   }
2598ec990867SGreg Clayton 
2599b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2600ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2601b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2602ec990867SGreg Clayton       StateType state = process_sp->GetState();
2603b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2604ec990867SGreg Clayton         const uint32_t stop_id = process_sp->GetStopID();
2605ec990867SGreg Clayton         if (m_stop_id == stop_id)
2606ec990867SGreg Clayton           return; // Children are already up to date
2607ec990867SGreg Clayton 
2608ec990867SGreg Clayton         m_stop_id = stop_id;
2609ec990867SGreg Clayton 
2610b9c1b51eSKate Stone         if (!m_thread_delegate_sp) {
2611ec990867SGreg Clayton           // Always expand the thread item the first time we show it
2612ec990867SGreg Clayton           // item.Expand();
2613796ac80bSJonas Devlieghere           m_thread_delegate_sp =
2614796ac80bSJonas Devlieghere               std::make_shared<ThreadTreeDelegate>(m_debugger);
2615ec990867SGreg Clayton         }
2616ec990867SGreg Clayton 
2617ec990867SGreg Clayton         TreeItem t(&item, *m_thread_delegate_sp, false);
2618ec990867SGreg Clayton         ThreadList &threads = process_sp->GetThreadList();
2619bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
2620ec990867SGreg Clayton         size_t num_threads = threads.GetSize();
2621ec990867SGreg Clayton         item.Resize(num_threads, t);
2622b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
2623ec990867SGreg Clayton           item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
2624ec990867SGreg Clayton           item[i].SetMightHaveChildren(true);
2625ec990867SGreg Clayton         }
2626ec990867SGreg Clayton         return;
2627ec990867SGreg Clayton       }
2628ec990867SGreg Clayton     }
2629ec990867SGreg Clayton     item.ClearChildren();
2630ec990867SGreg Clayton   }
2631ec990867SGreg Clayton 
2632b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override { return false; }
2633ec990867SGreg Clayton 
2634ec990867SGreg Clayton protected:
2635ec990867SGreg Clayton   std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
2636ec990867SGreg Clayton   Debugger &m_debugger;
2637ec990867SGreg Clayton   uint32_t m_stop_id;
2638554f68d3SGreg Clayton   FormatEntity::Entry m_format;
2639ec990867SGreg Clayton };
2640ec990867SGreg Clayton 
2641b9c1b51eSKate Stone class ValueObjectListDelegate : public WindowDelegate {
264244d93782SGreg Clayton public:
2643b9c1b51eSKate Stone   ValueObjectListDelegate()
26448369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2645b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2646b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {}
264744d93782SGreg Clayton 
2648b9c1b51eSKate Stone   ValueObjectListDelegate(ValueObjectList &valobj_list)
26498369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2650b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2651b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {
265244d93782SGreg Clayton     SetValues(valobj_list);
265344d93782SGreg Clayton   }
265444d93782SGreg Clayton 
2655315b6884SEugene Zelenko   ~ValueObjectListDelegate() override = default;
265644d93782SGreg Clayton 
2657b9c1b51eSKate Stone   void SetValues(ValueObjectList &valobj_list) {
2658c5dac77aSEugene Zelenko     m_selected_row = nullptr;
265944d93782SGreg Clayton     m_selected_row_idx = 0;
266044d93782SGreg Clayton     m_first_visible_row = 0;
266144d93782SGreg Clayton     m_num_rows = 0;
266244d93782SGreg Clayton     m_rows.clear();
26638369b28dSGreg Clayton     for (auto &valobj_sp : valobj_list.GetObjects())
26648369b28dSGreg Clayton       m_rows.push_back(Row(valobj_sp, nullptr));
266544d93782SGreg Clayton   }
266644d93782SGreg Clayton 
2667b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
266844d93782SGreg Clayton     m_num_rows = 0;
266944d93782SGreg Clayton     m_min_x = 2;
267044d93782SGreg Clayton     m_min_y = 1;
267144d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
267244d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
267344d93782SGreg Clayton 
267444d93782SGreg Clayton     window.Erase();
267544d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
267644d93782SGreg Clayton 
267744d93782SGreg Clayton     const int num_visible_rows = NumVisibleRows();
267844d93782SGreg Clayton     const int num_rows = CalculateTotalNumberRows(m_rows);
267944d93782SGreg Clayton 
268005097246SAdrian Prantl     // If we unexpanded while having something selected our total number of
268105097246SAdrian Prantl     // rows is less than the num visible rows, then make sure we show all the
268205097246SAdrian Prantl     // rows by setting the first visible row accordingly.
268344d93782SGreg Clayton     if (m_first_visible_row > 0 && num_rows < num_visible_rows)
268444d93782SGreg Clayton       m_first_visible_row = 0;
268544d93782SGreg Clayton 
268644d93782SGreg Clayton     // Make sure the selected row is always visible
268744d93782SGreg Clayton     if (m_selected_row_idx < m_first_visible_row)
268844d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx;
268944d93782SGreg Clayton     else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
269044d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
269144d93782SGreg Clayton 
269244d93782SGreg Clayton     DisplayRows(window, m_rows, g_options);
269344d93782SGreg Clayton 
269444d93782SGreg Clayton     window.DeferredRefresh();
269544d93782SGreg Clayton 
269644d93782SGreg Clayton     // Get the selected row
269744d93782SGreg Clayton     m_selected_row = GetRowForRowIndex(m_selected_row_idx);
269805097246SAdrian Prantl     // Keep the cursor on the selected row so the highlight and the cursor are
269905097246SAdrian Prantl     // always on the same line
270044d93782SGreg Clayton     if (m_selected_row)
2701b9c1b51eSKate Stone       window.MoveCursor(m_selected_row->x, m_selected_row->y);
270244d93782SGreg Clayton 
270344d93782SGreg Clayton     return true; // Drawing handled
270444d93782SGreg Clayton   }
270544d93782SGreg Clayton 
2706b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
270744d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
270844d93782SGreg Clayton         {KEY_UP, "Select previous item"},
270944d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
271044d93782SGreg Clayton         {KEY_RIGHT, "Expand selected item"},
271144d93782SGreg Clayton         {KEY_LEFT, "Unexpand selected item or select parent if not expanded"},
271244d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
271344d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
271444d93782SGreg Clayton         {'A', "Format as annotated address"},
271544d93782SGreg Clayton         {'b', "Format as binary"},
271644d93782SGreg Clayton         {'B', "Format as hex bytes with ASCII"},
271744d93782SGreg Clayton         {'c', "Format as character"},
271844d93782SGreg Clayton         {'d', "Format as a signed integer"},
271944d93782SGreg Clayton         {'D', "Format selected value using the default format for the type"},
272044d93782SGreg Clayton         {'f', "Format as float"},
272144d93782SGreg Clayton         {'h', "Show help dialog"},
272244d93782SGreg Clayton         {'i', "Format as instructions"},
272344d93782SGreg Clayton         {'o', "Format as octal"},
272444d93782SGreg Clayton         {'p', "Format as pointer"},
272544d93782SGreg Clayton         {'s', "Format as C string"},
272644d93782SGreg Clayton         {'t', "Toggle showing/hiding type names"},
272744d93782SGreg Clayton         {'u', "Format as an unsigned integer"},
272844d93782SGreg Clayton         {'x', "Format as hex"},
272944d93782SGreg Clayton         {'X', "Format as uppercase hex"},
273044d93782SGreg Clayton         {' ', "Toggle item expansion"},
273144d93782SGreg Clayton         {',', "Page up"},
273244d93782SGreg Clayton         {'.', "Page down"},
2733b9c1b51eSKate Stone         {'\0', nullptr}};
273444d93782SGreg Clayton     return g_source_view_key_help;
273544d93782SGreg Clayton   }
273644d93782SGreg Clayton 
2737b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2738b9c1b51eSKate Stone     switch (c) {
273944d93782SGreg Clayton     case 'x':
274044d93782SGreg Clayton     case 'X':
274144d93782SGreg Clayton     case 'o':
274244d93782SGreg Clayton     case 's':
274344d93782SGreg Clayton     case 'u':
274444d93782SGreg Clayton     case 'd':
274544d93782SGreg Clayton     case 'D':
274644d93782SGreg Clayton     case 'i':
274744d93782SGreg Clayton     case 'A':
274844d93782SGreg Clayton     case 'p':
274944d93782SGreg Clayton     case 'c':
275044d93782SGreg Clayton     case 'b':
275144d93782SGreg Clayton     case 'B':
275244d93782SGreg Clayton     case 'f':
275344d93782SGreg Clayton       // Change the format for the currently selected item
27548369b28dSGreg Clayton       if (m_selected_row) {
27558369b28dSGreg Clayton         auto valobj_sp = m_selected_row->value.GetSP();
27568369b28dSGreg Clayton         if (valobj_sp)
27578369b28dSGreg Clayton           valobj_sp->SetFormat(FormatForChar(c));
27588369b28dSGreg Clayton       }
275944d93782SGreg Clayton       return eKeyHandled;
276044d93782SGreg Clayton 
276144d93782SGreg Clayton     case 't':
276244d93782SGreg Clayton       // Toggle showing type names
276344d93782SGreg Clayton       g_options.show_types = !g_options.show_types;
276444d93782SGreg Clayton       return eKeyHandled;
276544d93782SGreg Clayton 
276644d93782SGreg Clayton     case ',':
276744d93782SGreg Clayton     case KEY_PPAGE:
276844d93782SGreg Clayton       // Page up key
2769b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
27703985c8c6SSaleem Abdulrasool         if (static_cast<int>(m_first_visible_row) > m_max_y)
277144d93782SGreg Clayton           m_first_visible_row -= m_max_y;
277244d93782SGreg Clayton         else
277344d93782SGreg Clayton           m_first_visible_row = 0;
277444d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
277544d93782SGreg Clayton       }
277644d93782SGreg Clayton       return eKeyHandled;
277744d93782SGreg Clayton 
277844d93782SGreg Clayton     case '.':
277944d93782SGreg Clayton     case KEY_NPAGE:
278044d93782SGreg Clayton       // Page down key
2781b9c1b51eSKate Stone       if (m_num_rows > static_cast<size_t>(m_max_y)) {
2782b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
278344d93782SGreg Clayton           m_first_visible_row += m_max_y;
278444d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
278544d93782SGreg Clayton         }
278644d93782SGreg Clayton       }
278744d93782SGreg Clayton       return eKeyHandled;
278844d93782SGreg Clayton 
278944d93782SGreg Clayton     case KEY_UP:
279044d93782SGreg Clayton       if (m_selected_row_idx > 0)
279144d93782SGreg Clayton         --m_selected_row_idx;
279244d93782SGreg Clayton       return eKeyHandled;
2793315b6884SEugene Zelenko 
279444d93782SGreg Clayton     case KEY_DOWN:
279544d93782SGreg Clayton       if (m_selected_row_idx + 1 < m_num_rows)
279644d93782SGreg Clayton         ++m_selected_row_idx;
279744d93782SGreg Clayton       return eKeyHandled;
279844d93782SGreg Clayton 
279944d93782SGreg Clayton     case KEY_RIGHT:
2800b9c1b51eSKate Stone       if (m_selected_row) {
280144d93782SGreg Clayton         if (!m_selected_row->expanded)
280244d93782SGreg Clayton           m_selected_row->Expand();
280344d93782SGreg Clayton       }
280444d93782SGreg Clayton       return eKeyHandled;
280544d93782SGreg Clayton 
280644d93782SGreg Clayton     case KEY_LEFT:
2807b9c1b51eSKate Stone       if (m_selected_row) {
280844d93782SGreg Clayton         if (m_selected_row->expanded)
280944d93782SGreg Clayton           m_selected_row->Unexpand();
281044d93782SGreg Clayton         else if (m_selected_row->parent)
281144d93782SGreg Clayton           m_selected_row_idx = m_selected_row->parent->row_idx;
281244d93782SGreg Clayton       }
281344d93782SGreg Clayton       return eKeyHandled;
281444d93782SGreg Clayton 
281544d93782SGreg Clayton     case ' ':
281644d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2817b9c1b51eSKate Stone       if (m_selected_row) {
281844d93782SGreg Clayton         if (m_selected_row->expanded)
281944d93782SGreg Clayton           m_selected_row->Unexpand();
282044d93782SGreg Clayton         else
282144d93782SGreg Clayton           m_selected_row->Expand();
282244d93782SGreg Clayton       }
282344d93782SGreg Clayton       return eKeyHandled;
282444d93782SGreg Clayton 
282544d93782SGreg Clayton     case 'h':
282644d93782SGreg Clayton       window.CreateHelpSubwindow();
282744d93782SGreg Clayton       return eKeyHandled;
282844d93782SGreg Clayton 
282944d93782SGreg Clayton     default:
283044d93782SGreg Clayton       break;
283144d93782SGreg Clayton     }
283244d93782SGreg Clayton     return eKeyNotHandled;
283344d93782SGreg Clayton   }
283444d93782SGreg Clayton 
283544d93782SGreg Clayton protected:
283644d93782SGreg Clayton   std::vector<Row> m_rows;
283744d93782SGreg Clayton   Row *m_selected_row;
283844d93782SGreg Clayton   uint32_t m_selected_row_idx;
283944d93782SGreg Clayton   uint32_t m_first_visible_row;
284044d93782SGreg Clayton   uint32_t m_num_rows;
284144d93782SGreg Clayton   int m_min_x;
284244d93782SGreg Clayton   int m_min_y;
284344d93782SGreg Clayton   int m_max_x;
284444d93782SGreg Clayton   int m_max_y;
284544d93782SGreg Clayton 
2846b9c1b51eSKate Stone   static Format FormatForChar(int c) {
2847b9c1b51eSKate Stone     switch (c) {
2848b9c1b51eSKate Stone     case 'x':
2849b9c1b51eSKate Stone       return eFormatHex;
2850b9c1b51eSKate Stone     case 'X':
2851b9c1b51eSKate Stone       return eFormatHexUppercase;
2852b9c1b51eSKate Stone     case 'o':
2853b9c1b51eSKate Stone       return eFormatOctal;
2854b9c1b51eSKate Stone     case 's':
2855b9c1b51eSKate Stone       return eFormatCString;
2856b9c1b51eSKate Stone     case 'u':
2857b9c1b51eSKate Stone       return eFormatUnsigned;
2858b9c1b51eSKate Stone     case 'd':
2859b9c1b51eSKate Stone       return eFormatDecimal;
2860b9c1b51eSKate Stone     case 'D':
2861b9c1b51eSKate Stone       return eFormatDefault;
2862b9c1b51eSKate Stone     case 'i':
2863b9c1b51eSKate Stone       return eFormatInstruction;
2864b9c1b51eSKate Stone     case 'A':
2865b9c1b51eSKate Stone       return eFormatAddressInfo;
2866b9c1b51eSKate Stone     case 'p':
2867b9c1b51eSKate Stone       return eFormatPointer;
2868b9c1b51eSKate Stone     case 'c':
2869b9c1b51eSKate Stone       return eFormatChar;
2870b9c1b51eSKate Stone     case 'b':
2871b9c1b51eSKate Stone       return eFormatBinary;
2872b9c1b51eSKate Stone     case 'B':
2873b9c1b51eSKate Stone       return eFormatBytesWithASCII;
2874b9c1b51eSKate Stone     case 'f':
2875b9c1b51eSKate Stone       return eFormatFloat;
287644d93782SGreg Clayton     }
287744d93782SGreg Clayton     return eFormatDefault;
287844d93782SGreg Clayton   }
287944d93782SGreg Clayton 
2880b9c1b51eSKate Stone   bool DisplayRowObject(Window &window, Row &row, DisplayOptions &options,
2881b9c1b51eSKate Stone                         bool highlight, bool last_child) {
28828369b28dSGreg Clayton     ValueObject *valobj = row.value.GetSP().get();
288344d93782SGreg Clayton 
2884c5dac77aSEugene Zelenko     if (valobj == nullptr)
288544d93782SGreg Clayton       return false;
288644d93782SGreg Clayton 
2887b9c1b51eSKate Stone     const char *type_name =
2888b9c1b51eSKate Stone         options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
288944d93782SGreg Clayton     const char *name = valobj->GetName().GetCString();
289044d93782SGreg Clayton     const char *value = valobj->GetValueAsCString();
289144d93782SGreg Clayton     const char *summary = valobj->GetSummaryAsCString();
289244d93782SGreg Clayton 
289344d93782SGreg Clayton     window.MoveCursor(row.x, row.y);
289444d93782SGreg Clayton 
289544d93782SGreg Clayton     row.DrawTree(window);
289644d93782SGreg Clayton 
289744d93782SGreg Clayton     if (highlight)
289844d93782SGreg Clayton       window.AttributeOn(A_REVERSE);
289944d93782SGreg Clayton 
290044d93782SGreg Clayton     if (type_name && type_name[0])
290144d93782SGreg Clayton       window.Printf("(%s) ", type_name);
290244d93782SGreg Clayton 
290344d93782SGreg Clayton     if (name && name[0])
290444d93782SGreg Clayton       window.PutCString(name);
290544d93782SGreg Clayton 
290644d93782SGreg Clayton     attr_t changd_attr = 0;
290744d93782SGreg Clayton     if (valobj->GetValueDidChange())
290844d93782SGreg Clayton       changd_attr = COLOR_PAIR(5) | A_BOLD;
290944d93782SGreg Clayton 
2910b9c1b51eSKate Stone     if (value && value[0]) {
291144d93782SGreg Clayton       window.PutCString(" = ");
291244d93782SGreg Clayton       if (changd_attr)
291344d93782SGreg Clayton         window.AttributeOn(changd_attr);
291444d93782SGreg Clayton       window.PutCString(value);
291544d93782SGreg Clayton       if (changd_attr)
291644d93782SGreg Clayton         window.AttributeOff(changd_attr);
291744d93782SGreg Clayton     }
291844d93782SGreg Clayton 
2919b9c1b51eSKate Stone     if (summary && summary[0]) {
292044d93782SGreg Clayton       window.PutChar(' ');
292144d93782SGreg Clayton       if (changd_attr)
292244d93782SGreg Clayton         window.AttributeOn(changd_attr);
292344d93782SGreg Clayton       window.PutCString(summary);
292444d93782SGreg Clayton       if (changd_attr)
292544d93782SGreg Clayton         window.AttributeOff(changd_attr);
292644d93782SGreg Clayton     }
292744d93782SGreg Clayton 
292844d93782SGreg Clayton     if (highlight)
292944d93782SGreg Clayton       window.AttributeOff(A_REVERSE);
293044d93782SGreg Clayton 
293144d93782SGreg Clayton     return true;
293244d93782SGreg Clayton   }
2933315b6884SEugene Zelenko 
2934b9c1b51eSKate Stone   void DisplayRows(Window &window, std::vector<Row> &rows,
2935b9c1b51eSKate Stone                    DisplayOptions &options) {
293644d93782SGreg Clayton     // >   0x25B7
293744d93782SGreg Clayton     // \/  0x25BD
293844d93782SGreg Clayton 
293944d93782SGreg Clayton     bool window_is_active = window.IsActive();
2940b9c1b51eSKate Stone     for (auto &row : rows) {
294144d93782SGreg Clayton       const bool last_child = row.parent && &rows[rows.size() - 1] == &row;
294244d93782SGreg Clayton       // Save the row index in each Row structure
294344d93782SGreg Clayton       row.row_idx = m_num_rows;
294444d93782SGreg Clayton       if ((m_num_rows >= m_first_visible_row) &&
2945b9c1b51eSKate Stone           ((m_num_rows - m_first_visible_row) <
2946b9c1b51eSKate Stone            static_cast<size_t>(NumVisibleRows()))) {
294744d93782SGreg Clayton         row.x = m_min_x;
294844d93782SGreg Clayton         row.y = m_num_rows - m_first_visible_row + 1;
2949b9c1b51eSKate Stone         if (DisplayRowObject(window, row, options,
2950b9c1b51eSKate Stone                              window_is_active &&
2951b9c1b51eSKate Stone                                  m_num_rows == m_selected_row_idx,
2952b9c1b51eSKate Stone                              last_child)) {
295344d93782SGreg Clayton           ++m_num_rows;
2954b9c1b51eSKate Stone         } else {
295544d93782SGreg Clayton           row.x = 0;
295644d93782SGreg Clayton           row.y = 0;
295744d93782SGreg Clayton         }
2958b9c1b51eSKate Stone       } else {
295944d93782SGreg Clayton         row.x = 0;
296044d93782SGreg Clayton         row.y = 0;
296144d93782SGreg Clayton         ++m_num_rows;
296244d93782SGreg Clayton       }
296344d93782SGreg Clayton 
29648369b28dSGreg Clayton       auto &children = row.GetChildren();
29658369b28dSGreg Clayton       if (row.expanded && !children.empty()) {
29668369b28dSGreg Clayton         DisplayRows(window, children, options);
296744d93782SGreg Clayton       }
296844d93782SGreg Clayton     }
296944d93782SGreg Clayton   }
297044d93782SGreg Clayton 
29718369b28dSGreg Clayton   int CalculateTotalNumberRows(std::vector<Row> &rows) {
297244d93782SGreg Clayton     int row_count = 0;
29738369b28dSGreg Clayton     for (auto &row : rows) {
297444d93782SGreg Clayton       ++row_count;
297544d93782SGreg Clayton       if (row.expanded)
29768369b28dSGreg Clayton         row_count += CalculateTotalNumberRows(row.GetChildren());
297744d93782SGreg Clayton     }
297844d93782SGreg Clayton     return row_count;
297944d93782SGreg Clayton   }
2980315b6884SEugene Zelenko 
2981b9c1b51eSKate Stone   static Row *GetRowForRowIndexImpl(std::vector<Row> &rows, size_t &row_index) {
2982b9c1b51eSKate Stone     for (auto &row : rows) {
298344d93782SGreg Clayton       if (row_index == 0)
298444d93782SGreg Clayton         return &row;
2985b9c1b51eSKate Stone       else {
298644d93782SGreg Clayton         --row_index;
29878369b28dSGreg Clayton         auto &children = row.GetChildren();
29888369b28dSGreg Clayton         if (row.expanded && !children.empty()) {
29898369b28dSGreg Clayton           Row *result = GetRowForRowIndexImpl(children, row_index);
299044d93782SGreg Clayton           if (result)
299144d93782SGreg Clayton             return result;
299244d93782SGreg Clayton         }
299344d93782SGreg Clayton       }
299444d93782SGreg Clayton     }
2995c5dac77aSEugene Zelenko     return nullptr;
299644d93782SGreg Clayton   }
299744d93782SGreg Clayton 
2998b9c1b51eSKate Stone   Row *GetRowForRowIndex(size_t row_index) {
299944d93782SGreg Clayton     return GetRowForRowIndexImpl(m_rows, row_index);
300044d93782SGreg Clayton   }
300144d93782SGreg Clayton 
3002b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
300344d93782SGreg Clayton 
300444d93782SGreg Clayton   static DisplayOptions g_options;
300544d93782SGreg Clayton };
300644d93782SGreg Clayton 
3007b9c1b51eSKate Stone class FrameVariablesWindowDelegate : public ValueObjectListDelegate {
300844d93782SGreg Clayton public:
3009b9c1b51eSKate Stone   FrameVariablesWindowDelegate(Debugger &debugger)
3010b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger),
3011b9c1b51eSKate Stone         m_frame_block(nullptr) {}
301244d93782SGreg Clayton 
3013315b6884SEugene Zelenko   ~FrameVariablesWindowDelegate() override = default;
301444d93782SGreg Clayton 
3015b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
301644d93782SGreg Clayton     return "Frame variable window keyboard shortcuts:";
301744d93782SGreg Clayton   }
301844d93782SGreg Clayton 
3019b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3020b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3021b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
302244d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3023c5dac77aSEugene Zelenko     Block *frame_block = nullptr;
3024c5dac77aSEugene Zelenko     StackFrame *frame = nullptr;
302544d93782SGreg Clayton 
3026b9c1b51eSKate Stone     if (process) {
302744d93782SGreg Clayton       StateType state = process->GetState();
3028b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
302944d93782SGreg Clayton         frame = exe_ctx.GetFramePtr();
303044d93782SGreg Clayton         if (frame)
303144d93782SGreg Clayton           frame_block = frame->GetFrameBlock();
3032b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
303344d93782SGreg Clayton         return true; // Don't do any updating when we are running
303444d93782SGreg Clayton       }
303544d93782SGreg Clayton     }
303644d93782SGreg Clayton 
303744d93782SGreg Clayton     ValueObjectList local_values;
3038b9c1b51eSKate Stone     if (frame_block) {
303944d93782SGreg Clayton       // Only update the variables if they have changed
3040b9c1b51eSKate Stone       if (m_frame_block != frame_block) {
304144d93782SGreg Clayton         m_frame_block = frame_block;
304244d93782SGreg Clayton 
304344d93782SGreg Clayton         VariableList *locals = frame->GetVariableList(true);
3044b9c1b51eSKate Stone         if (locals) {
304544d93782SGreg Clayton           const DynamicValueType use_dynamic = eDynamicDontRunTarget;
304644d93782SGreg Clayton           const size_t num_locals = locals->GetSize();
3047b9c1b51eSKate Stone           for (size_t i = 0; i < num_locals; ++i) {
3048b9c1b51eSKate Stone             ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable(
3049b9c1b51eSKate Stone                 locals->GetVariableAtIndex(i), use_dynamic);
3050b9c1b51eSKate Stone             if (value_sp) {
3051eb72dc7dSGreg Clayton               ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3052eb72dc7dSGreg Clayton               if (synthetic_value_sp)
3053eb72dc7dSGreg Clayton                 local_values.Append(synthetic_value_sp);
3054eb72dc7dSGreg Clayton               else
3055eb72dc7dSGreg Clayton                 local_values.Append(value_sp);
3056eb72dc7dSGreg Clayton             }
3057eb72dc7dSGreg Clayton           }
305844d93782SGreg Clayton           // Update the values
305944d93782SGreg Clayton           SetValues(local_values);
306044d93782SGreg Clayton         }
306144d93782SGreg Clayton       }
3062b9c1b51eSKate Stone     } else {
3063c5dac77aSEugene Zelenko       m_frame_block = nullptr;
306444d93782SGreg Clayton       // Update the values with an empty list if there is no frame
306544d93782SGreg Clayton       SetValues(local_values);
306644d93782SGreg Clayton     }
306744d93782SGreg Clayton 
306844d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
306944d93782SGreg Clayton   }
307044d93782SGreg Clayton 
307144d93782SGreg Clayton protected:
307244d93782SGreg Clayton   Debugger &m_debugger;
307344d93782SGreg Clayton   Block *m_frame_block;
307444d93782SGreg Clayton };
307544d93782SGreg Clayton 
3076b9c1b51eSKate Stone class RegistersWindowDelegate : public ValueObjectListDelegate {
307744d93782SGreg Clayton public:
3078b9c1b51eSKate Stone   RegistersWindowDelegate(Debugger &debugger)
3079b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger) {}
308044d93782SGreg Clayton 
3081315b6884SEugene Zelenko   ~RegistersWindowDelegate() override = default;
308244d93782SGreg Clayton 
3083b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
308444d93782SGreg Clayton     return "Register window keyboard shortcuts:";
308544d93782SGreg Clayton   }
308644d93782SGreg Clayton 
3087b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3088b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3089b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
309044d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
309144d93782SGreg Clayton 
309244d93782SGreg Clayton     ValueObjectList value_list;
3093b9c1b51eSKate Stone     if (frame) {
3094b9c1b51eSKate Stone       if (frame->GetStackID() != m_stack_id) {
309544d93782SGreg Clayton         m_stack_id = frame->GetStackID();
309644d93782SGreg Clayton         RegisterContextSP reg_ctx(frame->GetRegisterContext());
3097b9c1b51eSKate Stone         if (reg_ctx) {
309844d93782SGreg Clayton           const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
3099b9c1b51eSKate Stone           for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) {
3100b9c1b51eSKate Stone             value_list.Append(
3101b9c1b51eSKate Stone                 ValueObjectRegisterSet::Create(frame, reg_ctx, set_idx));
310244d93782SGreg Clayton           }
310344d93782SGreg Clayton         }
310444d93782SGreg Clayton         SetValues(value_list);
310544d93782SGreg Clayton       }
3106b9c1b51eSKate Stone     } else {
310744d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
310844d93782SGreg Clayton       if (process && process->IsAlive())
310944d93782SGreg Clayton         return true; // Don't do any updating if we are running
3110b9c1b51eSKate Stone       else {
311105097246SAdrian Prantl         // Update the values with an empty list if there is no process or the
311205097246SAdrian Prantl         // process isn't alive anymore
311344d93782SGreg Clayton         SetValues(value_list);
311444d93782SGreg Clayton       }
311544d93782SGreg Clayton     }
311644d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
311744d93782SGreg Clayton   }
311844d93782SGreg Clayton 
311944d93782SGreg Clayton protected:
312044d93782SGreg Clayton   Debugger &m_debugger;
312144d93782SGreg Clayton   StackID m_stack_id;
312244d93782SGreg Clayton };
312344d93782SGreg Clayton 
3124b9c1b51eSKate Stone static const char *CursesKeyToCString(int ch) {
312544d93782SGreg Clayton   static char g_desc[32];
3126b9c1b51eSKate Stone   if (ch >= KEY_F0 && ch < KEY_F0 + 64) {
312744d93782SGreg Clayton     snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
312844d93782SGreg Clayton     return g_desc;
312944d93782SGreg Clayton   }
3130b9c1b51eSKate Stone   switch (ch) {
3131b9c1b51eSKate Stone   case KEY_DOWN:
3132b9c1b51eSKate Stone     return "down";
3133b9c1b51eSKate Stone   case KEY_UP:
3134b9c1b51eSKate Stone     return "up";
3135b9c1b51eSKate Stone   case KEY_LEFT:
3136b9c1b51eSKate Stone     return "left";
3137b9c1b51eSKate Stone   case KEY_RIGHT:
3138b9c1b51eSKate Stone     return "right";
3139b9c1b51eSKate Stone   case KEY_HOME:
3140b9c1b51eSKate Stone     return "home";
3141b9c1b51eSKate Stone   case KEY_BACKSPACE:
3142b9c1b51eSKate Stone     return "backspace";
3143b9c1b51eSKate Stone   case KEY_DL:
3144b9c1b51eSKate Stone     return "delete-line";
3145b9c1b51eSKate Stone   case KEY_IL:
3146b9c1b51eSKate Stone     return "insert-line";
3147b9c1b51eSKate Stone   case KEY_DC:
3148b9c1b51eSKate Stone     return "delete-char";
3149b9c1b51eSKate Stone   case KEY_IC:
3150b9c1b51eSKate Stone     return "insert-char";
3151b9c1b51eSKate Stone   case KEY_CLEAR:
3152b9c1b51eSKate Stone     return "clear";
3153b9c1b51eSKate Stone   case KEY_EOS:
3154b9c1b51eSKate Stone     return "clear-to-eos";
3155b9c1b51eSKate Stone   case KEY_EOL:
3156b9c1b51eSKate Stone     return "clear-to-eol";
3157b9c1b51eSKate Stone   case KEY_SF:
3158b9c1b51eSKate Stone     return "scroll-forward";
3159b9c1b51eSKate Stone   case KEY_SR:
3160b9c1b51eSKate Stone     return "scroll-backward";
3161b9c1b51eSKate Stone   case KEY_NPAGE:
3162b9c1b51eSKate Stone     return "page-down";
3163b9c1b51eSKate Stone   case KEY_PPAGE:
3164b9c1b51eSKate Stone     return "page-up";
3165b9c1b51eSKate Stone   case KEY_STAB:
3166b9c1b51eSKate Stone     return "set-tab";
3167b9c1b51eSKate Stone   case KEY_CTAB:
3168b9c1b51eSKate Stone     return "clear-tab";
3169b9c1b51eSKate Stone   case KEY_CATAB:
3170b9c1b51eSKate Stone     return "clear-all-tabs";
3171b9c1b51eSKate Stone   case KEY_ENTER:
3172b9c1b51eSKate Stone     return "enter";
3173b9c1b51eSKate Stone   case KEY_PRINT:
3174b9c1b51eSKate Stone     return "print";
3175b9c1b51eSKate Stone   case KEY_LL:
3176b9c1b51eSKate Stone     return "lower-left key";
3177b9c1b51eSKate Stone   case KEY_A1:
3178b9c1b51eSKate Stone     return "upper left of keypad";
3179b9c1b51eSKate Stone   case KEY_A3:
3180b9c1b51eSKate Stone     return "upper right of keypad";
3181b9c1b51eSKate Stone   case KEY_B2:
3182b9c1b51eSKate Stone     return "center of keypad";
3183b9c1b51eSKate Stone   case KEY_C1:
3184b9c1b51eSKate Stone     return "lower left of keypad";
3185b9c1b51eSKate Stone   case KEY_C3:
3186b9c1b51eSKate Stone     return "lower right of keypad";
3187b9c1b51eSKate Stone   case KEY_BTAB:
3188b9c1b51eSKate Stone     return "back-tab key";
3189b9c1b51eSKate Stone   case KEY_BEG:
3190b9c1b51eSKate Stone     return "begin key";
3191b9c1b51eSKate Stone   case KEY_CANCEL:
3192b9c1b51eSKate Stone     return "cancel key";
3193b9c1b51eSKate Stone   case KEY_CLOSE:
3194b9c1b51eSKate Stone     return "close key";
3195b9c1b51eSKate Stone   case KEY_COMMAND:
3196b9c1b51eSKate Stone     return "command key";
3197b9c1b51eSKate Stone   case KEY_COPY:
3198b9c1b51eSKate Stone     return "copy key";
3199b9c1b51eSKate Stone   case KEY_CREATE:
3200b9c1b51eSKate Stone     return "create key";
3201b9c1b51eSKate Stone   case KEY_END:
3202b9c1b51eSKate Stone     return "end key";
3203b9c1b51eSKate Stone   case KEY_EXIT:
3204b9c1b51eSKate Stone     return "exit key";
3205b9c1b51eSKate Stone   case KEY_FIND:
3206b9c1b51eSKate Stone     return "find key";
3207b9c1b51eSKate Stone   case KEY_HELP:
3208b9c1b51eSKate Stone     return "help key";
3209b9c1b51eSKate Stone   case KEY_MARK:
3210b9c1b51eSKate Stone     return "mark key";
3211b9c1b51eSKate Stone   case KEY_MESSAGE:
3212b9c1b51eSKate Stone     return "message key";
3213b9c1b51eSKate Stone   case KEY_MOVE:
3214b9c1b51eSKate Stone     return "move key";
3215b9c1b51eSKate Stone   case KEY_NEXT:
3216b9c1b51eSKate Stone     return "next key";
3217b9c1b51eSKate Stone   case KEY_OPEN:
3218b9c1b51eSKate Stone     return "open key";
3219b9c1b51eSKate Stone   case KEY_OPTIONS:
3220b9c1b51eSKate Stone     return "options key";
3221b9c1b51eSKate Stone   case KEY_PREVIOUS:
3222b9c1b51eSKate Stone     return "previous key";
3223b9c1b51eSKate Stone   case KEY_REDO:
3224b9c1b51eSKate Stone     return "redo key";
3225b9c1b51eSKate Stone   case KEY_REFERENCE:
3226b9c1b51eSKate Stone     return "reference key";
3227b9c1b51eSKate Stone   case KEY_REFRESH:
3228b9c1b51eSKate Stone     return "refresh key";
3229b9c1b51eSKate Stone   case KEY_REPLACE:
3230b9c1b51eSKate Stone     return "replace key";
3231b9c1b51eSKate Stone   case KEY_RESTART:
3232b9c1b51eSKate Stone     return "restart key";
3233b9c1b51eSKate Stone   case KEY_RESUME:
3234b9c1b51eSKate Stone     return "resume key";
3235b9c1b51eSKate Stone   case KEY_SAVE:
3236b9c1b51eSKate Stone     return "save key";
3237b9c1b51eSKate Stone   case KEY_SBEG:
3238b9c1b51eSKate Stone     return "shifted begin key";
3239b9c1b51eSKate Stone   case KEY_SCANCEL:
3240b9c1b51eSKate Stone     return "shifted cancel key";
3241b9c1b51eSKate Stone   case KEY_SCOMMAND:
3242b9c1b51eSKate Stone     return "shifted command key";
3243b9c1b51eSKate Stone   case KEY_SCOPY:
3244b9c1b51eSKate Stone     return "shifted copy key";
3245b9c1b51eSKate Stone   case KEY_SCREATE:
3246b9c1b51eSKate Stone     return "shifted create key";
3247b9c1b51eSKate Stone   case KEY_SDC:
3248b9c1b51eSKate Stone     return "shifted delete-character key";
3249b9c1b51eSKate Stone   case KEY_SDL:
3250b9c1b51eSKate Stone     return "shifted delete-line key";
3251b9c1b51eSKate Stone   case KEY_SELECT:
3252b9c1b51eSKate Stone     return "select key";
3253b9c1b51eSKate Stone   case KEY_SEND:
3254b9c1b51eSKate Stone     return "shifted end key";
3255b9c1b51eSKate Stone   case KEY_SEOL:
3256b9c1b51eSKate Stone     return "shifted clear-to-end-of-line key";
3257b9c1b51eSKate Stone   case KEY_SEXIT:
3258b9c1b51eSKate Stone     return "shifted exit key";
3259b9c1b51eSKate Stone   case KEY_SFIND:
3260b9c1b51eSKate Stone     return "shifted find key";
3261b9c1b51eSKate Stone   case KEY_SHELP:
3262b9c1b51eSKate Stone     return "shifted help key";
3263b9c1b51eSKate Stone   case KEY_SHOME:
3264b9c1b51eSKate Stone     return "shifted home key";
3265b9c1b51eSKate Stone   case KEY_SIC:
3266b9c1b51eSKate Stone     return "shifted insert-character key";
3267b9c1b51eSKate Stone   case KEY_SLEFT:
3268b9c1b51eSKate Stone     return "shifted left-arrow key";
3269b9c1b51eSKate Stone   case KEY_SMESSAGE:
3270b9c1b51eSKate Stone     return "shifted message key";
3271b9c1b51eSKate Stone   case KEY_SMOVE:
3272b9c1b51eSKate Stone     return "shifted move key";
3273b9c1b51eSKate Stone   case KEY_SNEXT:
3274b9c1b51eSKate Stone     return "shifted next key";
3275b9c1b51eSKate Stone   case KEY_SOPTIONS:
3276b9c1b51eSKate Stone     return "shifted options key";
3277b9c1b51eSKate Stone   case KEY_SPREVIOUS:
3278b9c1b51eSKate Stone     return "shifted previous key";
3279b9c1b51eSKate Stone   case KEY_SPRINT:
3280b9c1b51eSKate Stone     return "shifted print key";
3281b9c1b51eSKate Stone   case KEY_SREDO:
3282b9c1b51eSKate Stone     return "shifted redo key";
3283b9c1b51eSKate Stone   case KEY_SREPLACE:
3284b9c1b51eSKate Stone     return "shifted replace key";
3285b9c1b51eSKate Stone   case KEY_SRIGHT:
3286b9c1b51eSKate Stone     return "shifted right-arrow key";
3287b9c1b51eSKate Stone   case KEY_SRSUME:
3288b9c1b51eSKate Stone     return "shifted resume key";
3289b9c1b51eSKate Stone   case KEY_SSAVE:
3290b9c1b51eSKate Stone     return "shifted save key";
3291b9c1b51eSKate Stone   case KEY_SSUSPEND:
3292b9c1b51eSKate Stone     return "shifted suspend key";
3293b9c1b51eSKate Stone   case KEY_SUNDO:
3294b9c1b51eSKate Stone     return "shifted undo key";
3295b9c1b51eSKate Stone   case KEY_SUSPEND:
3296b9c1b51eSKate Stone     return "suspend key";
3297b9c1b51eSKate Stone   case KEY_UNDO:
3298b9c1b51eSKate Stone     return "undo key";
3299b9c1b51eSKate Stone   case KEY_MOUSE:
3300b9c1b51eSKate Stone     return "Mouse event has occurred";
3301b9c1b51eSKate Stone   case KEY_RESIZE:
3302b9c1b51eSKate Stone     return "Terminal resize event";
330327801f4fSBruce Mitchener #ifdef KEY_EVENT
3304b9c1b51eSKate Stone   case KEY_EVENT:
3305b9c1b51eSKate Stone     return "We were interrupted by an event";
330627801f4fSBruce Mitchener #endif
3307b9c1b51eSKate Stone   case KEY_RETURN:
3308b9c1b51eSKate Stone     return "return";
3309b9c1b51eSKate Stone   case ' ':
3310b9c1b51eSKate Stone     return "space";
3311b9c1b51eSKate Stone   case '\t':
3312b9c1b51eSKate Stone     return "tab";
3313b9c1b51eSKate Stone   case KEY_ESCAPE:
3314b9c1b51eSKate Stone     return "escape";
331544d93782SGreg Clayton   default:
331644d93782SGreg Clayton     if (isprint(ch))
331744d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "%c", ch);
331844d93782SGreg Clayton     else
331944d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
332044d93782SGreg Clayton     return g_desc;
332144d93782SGreg Clayton   }
3322c5dac77aSEugene Zelenko   return nullptr;
332344d93782SGreg Clayton }
332444d93782SGreg Clayton 
3325b9c1b51eSKate Stone HelpDialogDelegate::HelpDialogDelegate(const char *text,
3326b9c1b51eSKate Stone                                        KeyHelp *key_help_array)
3327b9c1b51eSKate Stone     : m_text(), m_first_visible_line(0) {
3328b9c1b51eSKate Stone   if (text && text[0]) {
332944d93782SGreg Clayton     m_text.SplitIntoLines(text);
333044d93782SGreg Clayton     m_text.AppendString("");
333144d93782SGreg Clayton   }
3332b9c1b51eSKate Stone   if (key_help_array) {
3333b9c1b51eSKate Stone     for (KeyHelp *key = key_help_array; key->ch; ++key) {
333444d93782SGreg Clayton       StreamString key_description;
3335b9c1b51eSKate Stone       key_description.Printf("%10s - %s", CursesKeyToCString(key->ch),
3336b9c1b51eSKate Stone                              key->description);
3337c156427dSZachary Turner       m_text.AppendString(key_description.GetString());
333844d93782SGreg Clayton     }
333944d93782SGreg Clayton   }
334044d93782SGreg Clayton }
334144d93782SGreg Clayton 
3342315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
334344d93782SGreg Clayton 
3344b9c1b51eSKate Stone bool HelpDialogDelegate::WindowDelegateDraw(Window &window, bool force) {
334544d93782SGreg Clayton   window.Erase();
334644d93782SGreg Clayton   const int window_height = window.GetHeight();
334744d93782SGreg Clayton   int x = 2;
334844d93782SGreg Clayton   int y = 1;
334944d93782SGreg Clayton   const int min_y = y;
335044d93782SGreg Clayton   const int max_y = window_height - 1 - y;
33513985c8c6SSaleem Abdulrasool   const size_t num_visible_lines = max_y - min_y + 1;
335244d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
335344d93782SGreg Clayton   const char *bottom_message;
335444d93782SGreg Clayton   if (num_lines <= num_visible_lines)
335544d93782SGreg Clayton     bottom_message = "Press any key to exit";
335644d93782SGreg Clayton   else
335744d93782SGreg Clayton     bottom_message = "Use arrows to scroll, any other key to exit";
335844d93782SGreg Clayton   window.DrawTitleBox(window.GetName(), bottom_message);
3359b9c1b51eSKate Stone   while (y <= max_y) {
336044d93782SGreg Clayton     window.MoveCursor(x, y);
3361b9c1b51eSKate Stone     window.PutCStringTruncated(
3362b9c1b51eSKate Stone         m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
336344d93782SGreg Clayton     ++y;
336444d93782SGreg Clayton   }
336544d93782SGreg Clayton   return true;
336644d93782SGreg Clayton }
336744d93782SGreg Clayton 
3368b9c1b51eSKate Stone HandleCharResult HelpDialogDelegate::WindowDelegateHandleChar(Window &window,
3369b9c1b51eSKate Stone                                                               int key) {
337044d93782SGreg Clayton   bool done = false;
337144d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
337244d93782SGreg Clayton   const size_t num_visible_lines = window.GetHeight() - 2;
337344d93782SGreg Clayton 
3374b9c1b51eSKate Stone   if (num_lines <= num_visible_lines) {
337544d93782SGreg Clayton     done = true;
337605097246SAdrian Prantl     // If we have all lines visible and don't need scrolling, then any key
337705097246SAdrian Prantl     // press will cause us to exit
3378b9c1b51eSKate Stone   } else {
3379b9c1b51eSKate Stone     switch (key) {
338044d93782SGreg Clayton     case KEY_UP:
338144d93782SGreg Clayton       if (m_first_visible_line > 0)
338244d93782SGreg Clayton         --m_first_visible_line;
338344d93782SGreg Clayton       break;
338444d93782SGreg Clayton 
338544d93782SGreg Clayton     case KEY_DOWN:
338644d93782SGreg Clayton       if (m_first_visible_line + num_visible_lines < num_lines)
338744d93782SGreg Clayton         ++m_first_visible_line;
338844d93782SGreg Clayton       break;
338944d93782SGreg Clayton 
339044d93782SGreg Clayton     case KEY_PPAGE:
339144d93782SGreg Clayton     case ',':
3392b9c1b51eSKate Stone       if (m_first_visible_line > 0) {
33933985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
339444d93782SGreg Clayton           m_first_visible_line -= num_visible_lines;
339544d93782SGreg Clayton         else
339644d93782SGreg Clayton           m_first_visible_line = 0;
339744d93782SGreg Clayton       }
339844d93782SGreg Clayton       break;
3399315b6884SEugene Zelenko 
340044d93782SGreg Clayton     case KEY_NPAGE:
340144d93782SGreg Clayton     case '.':
3402b9c1b51eSKate Stone       if (m_first_visible_line + num_visible_lines < num_lines) {
340344d93782SGreg Clayton         m_first_visible_line += num_visible_lines;
34043985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > num_lines)
340544d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
340644d93782SGreg Clayton       }
340744d93782SGreg Clayton       break;
3408315b6884SEugene Zelenko 
340944d93782SGreg Clayton     default:
341044d93782SGreg Clayton       done = true;
341144d93782SGreg Clayton       break;
341244d93782SGreg Clayton     }
341344d93782SGreg Clayton   }
341444d93782SGreg Clayton   if (done)
341544d93782SGreg Clayton     window.GetParent()->RemoveSubWindow(&window);
341644d93782SGreg Clayton   return eKeyHandled;
341744d93782SGreg Clayton }
341844d93782SGreg Clayton 
3419b9c1b51eSKate Stone class ApplicationDelegate : public WindowDelegate, public MenuDelegate {
342044d93782SGreg Clayton public:
342144d93782SGreg Clayton   enum {
342244d93782SGreg Clayton     eMenuID_LLDB = 1,
342344d93782SGreg Clayton     eMenuID_LLDBAbout,
342444d93782SGreg Clayton     eMenuID_LLDBExit,
342544d93782SGreg Clayton 
342644d93782SGreg Clayton     eMenuID_Target,
342744d93782SGreg Clayton     eMenuID_TargetCreate,
342844d93782SGreg Clayton     eMenuID_TargetDelete,
342944d93782SGreg Clayton 
343044d93782SGreg Clayton     eMenuID_Process,
343144d93782SGreg Clayton     eMenuID_ProcessAttach,
343244d93782SGreg Clayton     eMenuID_ProcessDetach,
343344d93782SGreg Clayton     eMenuID_ProcessLaunch,
343444d93782SGreg Clayton     eMenuID_ProcessContinue,
343544d93782SGreg Clayton     eMenuID_ProcessHalt,
343644d93782SGreg Clayton     eMenuID_ProcessKill,
343744d93782SGreg Clayton 
343844d93782SGreg Clayton     eMenuID_Thread,
343944d93782SGreg Clayton     eMenuID_ThreadStepIn,
344044d93782SGreg Clayton     eMenuID_ThreadStepOver,
344144d93782SGreg Clayton     eMenuID_ThreadStepOut,
344244d93782SGreg Clayton 
344344d93782SGreg Clayton     eMenuID_View,
344444d93782SGreg Clayton     eMenuID_ViewBacktrace,
344544d93782SGreg Clayton     eMenuID_ViewRegisters,
344644d93782SGreg Clayton     eMenuID_ViewSource,
344744d93782SGreg Clayton     eMenuID_ViewVariables,
344844d93782SGreg Clayton 
344944d93782SGreg Clayton     eMenuID_Help,
345044d93782SGreg Clayton     eMenuID_HelpGUIHelp
345144d93782SGreg Clayton   };
345244d93782SGreg Clayton 
3453b9c1b51eSKate Stone   ApplicationDelegate(Application &app, Debugger &debugger)
3454b9c1b51eSKate Stone       : WindowDelegate(), MenuDelegate(), m_app(app), m_debugger(debugger) {}
345544d93782SGreg Clayton 
3456315b6884SEugene Zelenko   ~ApplicationDelegate() override = default;
3457bd5ae6b4SGreg Clayton 
3458b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
345944d93782SGreg Clayton     return false; // Drawing not handled, let standard window drawing happen
346044d93782SGreg Clayton   }
346144d93782SGreg Clayton 
3462b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override {
3463b9c1b51eSKate Stone     switch (key) {
34645fdb09bbSGreg Clayton     case '\t':
346544d93782SGreg Clayton       window.SelectNextWindowAsActive();
346644d93782SGreg Clayton       return eKeyHandled;
34675fdb09bbSGreg Clayton 
34685fdb09bbSGreg Clayton     case 'h':
34695fdb09bbSGreg Clayton       window.CreateHelpSubwindow();
34705fdb09bbSGreg Clayton       return eKeyHandled;
34715fdb09bbSGreg Clayton 
34725fdb09bbSGreg Clayton     case KEY_ESCAPE:
34735fdb09bbSGreg Clayton       return eQuitApplication;
34745fdb09bbSGreg Clayton 
34755fdb09bbSGreg Clayton     default:
34765fdb09bbSGreg Clayton       break;
347744d93782SGreg Clayton     }
347844d93782SGreg Clayton     return eKeyNotHandled;
347944d93782SGreg Clayton   }
348044d93782SGreg Clayton 
3481b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
34825fdb09bbSGreg Clayton     return "Welcome to the LLDB curses GUI.\n\n"
34835fdb09bbSGreg Clayton            "Press the TAB key to change the selected view.\n"
3484b9c1b51eSKate Stone            "Each view has its own keyboard shortcuts, press 'h' to open a "
3485b9c1b51eSKate Stone            "dialog to display them.\n\n"
34865fdb09bbSGreg Clayton            "Common key bindings for all views:";
34875fdb09bbSGreg Clayton   }
34885fdb09bbSGreg Clayton 
3489b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
34905fdb09bbSGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
34915fdb09bbSGreg Clayton         {'\t', "Select next view"},
34925fdb09bbSGreg Clayton         {'h', "Show help dialog with view specific key bindings"},
34935fdb09bbSGreg Clayton         {',', "Page up"},
34945fdb09bbSGreg Clayton         {'.', "Page down"},
34955fdb09bbSGreg Clayton         {KEY_UP, "Select previous"},
34965fdb09bbSGreg Clayton         {KEY_DOWN, "Select next"},
34975fdb09bbSGreg Clayton         {KEY_LEFT, "Unexpand or select parent"},
34985fdb09bbSGreg Clayton         {KEY_RIGHT, "Expand"},
34995fdb09bbSGreg Clayton         {KEY_PPAGE, "Page up"},
35005fdb09bbSGreg Clayton         {KEY_NPAGE, "Page down"},
3501b9c1b51eSKate Stone         {'\0', nullptr}};
35025fdb09bbSGreg Clayton     return g_source_view_key_help;
35035fdb09bbSGreg Clayton   }
35045fdb09bbSGreg Clayton 
3505b9c1b51eSKate Stone   MenuActionResult MenuDelegateAction(Menu &menu) override {
3506b9c1b51eSKate Stone     switch (menu.GetIdentifier()) {
3507b9c1b51eSKate Stone     case eMenuID_ThreadStepIn: {
3508b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3509b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3510b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
351144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3512b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3513b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
35144b4b2478SJim Ingham           exe_ctx.GetThreadRef().StepIn(true);
351544d93782SGreg Clayton       }
351644d93782SGreg Clayton     }
351744d93782SGreg Clayton       return MenuActionResult::Handled;
351844d93782SGreg Clayton 
3519b9c1b51eSKate Stone     case eMenuID_ThreadStepOut: {
3520b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3521b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3522b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
352344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3524b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3525b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
352644d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
352744d93782SGreg Clayton       }
352844d93782SGreg Clayton     }
352944d93782SGreg Clayton       return MenuActionResult::Handled;
353044d93782SGreg Clayton 
3531b9c1b51eSKate Stone     case eMenuID_ThreadStepOver: {
3532b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3533b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3534b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
353544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3536b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3537b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
353844d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOver(true);
353944d93782SGreg Clayton       }
354044d93782SGreg Clayton     }
354144d93782SGreg Clayton       return MenuActionResult::Handled;
354244d93782SGreg Clayton 
3543b9c1b51eSKate Stone     case eMenuID_ProcessContinue: {
3544b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3545b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3546b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
354744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3548b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3549b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
355044d93782SGreg Clayton           process->Resume();
355144d93782SGreg Clayton       }
355244d93782SGreg Clayton     }
355344d93782SGreg Clayton       return MenuActionResult::Handled;
355444d93782SGreg Clayton 
3555b9c1b51eSKate Stone     case eMenuID_ProcessKill: {
3556b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3557b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3558b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
355944d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
356044d93782SGreg Clayton         if (process && process->IsAlive())
3561ede3193bSJason Molenda           process->Destroy(false);
356244d93782SGreg Clayton       }
356344d93782SGreg Clayton     }
356444d93782SGreg Clayton       return MenuActionResult::Handled;
356544d93782SGreg Clayton 
3566b9c1b51eSKate Stone     case eMenuID_ProcessHalt: {
3567b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3568b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3569b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
357044d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
357144d93782SGreg Clayton         if (process && process->IsAlive())
357244d93782SGreg Clayton           process->Halt();
357344d93782SGreg Clayton       }
357444d93782SGreg Clayton     }
357544d93782SGreg Clayton       return MenuActionResult::Handled;
357644d93782SGreg Clayton 
3577b9c1b51eSKate Stone     case eMenuID_ProcessDetach: {
3578b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3579b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3580b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
358144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
358244d93782SGreg Clayton         if (process && process->IsAlive())
358344d93782SGreg Clayton           process->Detach(false);
358444d93782SGreg Clayton       }
358544d93782SGreg Clayton     }
358644d93782SGreg Clayton       return MenuActionResult::Handled;
358744d93782SGreg Clayton 
3588b9c1b51eSKate Stone     case eMenuID_Process: {
3589b9c1b51eSKate Stone       // Populate the menu with all of the threads if the process is stopped
359005097246SAdrian Prantl       // when the Process menu gets selected and is about to display its
359105097246SAdrian Prantl       // submenu.
359244d93782SGreg Clayton       Menus &submenus = menu.GetSubmenus();
3593b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3594b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
359544d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
3596b9c1b51eSKate Stone       if (process && process->IsAlive() &&
3597b9c1b51eSKate Stone           StateIsStoppedState(process->GetState(), true)) {
359844d93782SGreg Clayton         if (submenus.size() == 7)
359944d93782SGreg Clayton           menu.AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
360044d93782SGreg Clayton         else if (submenus.size() > 8)
360144d93782SGreg Clayton           submenus.erase(submenus.begin() + 8, submenus.end());
360244d93782SGreg Clayton 
360344d93782SGreg Clayton         ThreadList &threads = process->GetThreadList();
3604bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
360544d93782SGreg Clayton         size_t num_threads = threads.GetSize();
3606b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
360744d93782SGreg Clayton           ThreadSP thread_sp = threads.GetThreadAtIndex(i);
360844d93782SGreg Clayton           char menu_char = '\0';
360944d93782SGreg Clayton           if (i < 9)
361044d93782SGreg Clayton             menu_char = '1' + i;
361144d93782SGreg Clayton           StreamString thread_menu_title;
361244d93782SGreg Clayton           thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
361344d93782SGreg Clayton           const char *thread_name = thread_sp->GetName();
361444d93782SGreg Clayton           if (thread_name && thread_name[0])
361544d93782SGreg Clayton             thread_menu_title.Printf(" %s", thread_name);
3616b9c1b51eSKate Stone           else {
361744d93782SGreg Clayton             const char *queue_name = thread_sp->GetQueueName();
361844d93782SGreg Clayton             if (queue_name && queue_name[0])
361944d93782SGreg Clayton               thread_menu_title.Printf(" %s", queue_name);
362044d93782SGreg Clayton           }
3621b9c1b51eSKate Stone           menu.AddSubmenu(
3622c156427dSZachary Turner               MenuSP(new Menu(thread_menu_title.GetString().str().c_str(),
3623c156427dSZachary Turner                               nullptr, menu_char, thread_sp->GetID())));
362444d93782SGreg Clayton         }
3625b9c1b51eSKate Stone       } else if (submenus.size() > 7) {
362605097246SAdrian Prantl         // Remove the separator and any other thread submenu items that were
362705097246SAdrian Prantl         // previously added
362844d93782SGreg Clayton         submenus.erase(submenus.begin() + 7, submenus.end());
362944d93782SGreg Clayton       }
3630b9c1b51eSKate Stone       // Since we are adding and removing items we need to recalculate the name
3631b9c1b51eSKate Stone       // lengths
363244d93782SGreg Clayton       menu.RecalculateNameLengths();
363344d93782SGreg Clayton     }
363444d93782SGreg Clayton       return MenuActionResult::Handled;
363544d93782SGreg Clayton 
3636b9c1b51eSKate Stone     case eMenuID_ViewVariables: {
363744d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
363844d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
363944d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
364044d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
364144d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
364244d93782SGreg Clayton 
3643b9c1b51eSKate Stone       if (variables_window_sp) {
364444d93782SGreg Clayton         const Rect variables_bounds = variables_window_sp->GetBounds();
364544d93782SGreg Clayton 
364644d93782SGreg Clayton         main_window_sp->RemoveSubWindow(variables_window_sp.get());
364744d93782SGreg Clayton 
3648b9c1b51eSKate Stone         if (registers_window_sp) {
3649b9c1b51eSKate Stone           // We have a registers window, so give all the area back to the
3650b9c1b51eSKate Stone           // registers window
365144d93782SGreg Clayton           Rect registers_bounds = variables_bounds;
365244d93782SGreg Clayton           registers_bounds.size.width = source_bounds.size.width;
365344d93782SGreg Clayton           registers_window_sp->SetBounds(registers_bounds);
3654b9c1b51eSKate Stone         } else {
365505097246SAdrian Prantl           // We have no registers window showing so give the bottom area back
365605097246SAdrian Prantl           // to the source view
365744d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3658b9c1b51eSKate Stone                                    source_bounds.size.height +
3659b9c1b51eSKate Stone                                        variables_bounds.size.height);
366044d93782SGreg Clayton         }
3661b9c1b51eSKate Stone       } else {
366244d93782SGreg Clayton         Rect new_variables_rect;
3663b9c1b51eSKate Stone         if (registers_window_sp) {
366444d93782SGreg Clayton           // We have a registers window so split the area of the registers
366544d93782SGreg Clayton           // window into two columns where the left hand side will be the
366644d93782SGreg Clayton           // variables and the right hand side will be the registers
366744d93782SGreg Clayton           const Rect variables_bounds = registers_window_sp->GetBounds();
366844d93782SGreg Clayton           Rect new_registers_rect;
3669b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_variables_rect,
3670b9c1b51eSKate Stone                                                    new_registers_rect);
367144d93782SGreg Clayton           registers_window_sp->SetBounds(new_registers_rect);
3672b9c1b51eSKate Stone         } else {
367344d93782SGreg Clayton           // No variables window, grab the bottom part of the source window
367444d93782SGreg Clayton           Rect new_source_rect;
3675b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3676b9c1b51eSKate Stone                                                   new_variables_rect);
367744d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
367844d93782SGreg Clayton         }
3679b9c1b51eSKate Stone         WindowSP new_window_sp = main_window_sp->CreateSubWindow(
3680b9c1b51eSKate Stone             "Variables", new_variables_rect, false);
3681b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3682b9c1b51eSKate Stone             WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
368344d93782SGreg Clayton       }
368444d93782SGreg Clayton       touchwin(stdscr);
368544d93782SGreg Clayton     }
368644d93782SGreg Clayton       return MenuActionResult::Handled;
368744d93782SGreg Clayton 
3688b9c1b51eSKate Stone     case eMenuID_ViewRegisters: {
368944d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
369044d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
369144d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
369244d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
369344d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
369444d93782SGreg Clayton 
3695b9c1b51eSKate Stone       if (registers_window_sp) {
3696b9c1b51eSKate Stone         if (variables_window_sp) {
369744d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
369844d93782SGreg Clayton 
3699b9c1b51eSKate Stone           // We have a variables window, so give all the area back to the
3700b9c1b51eSKate Stone           // variables window
3701b9c1b51eSKate Stone           variables_window_sp->Resize(variables_bounds.size.width +
3702b9c1b51eSKate Stone                                           registers_window_sp->GetWidth(),
370344d93782SGreg Clayton                                       variables_bounds.size.height);
3704b9c1b51eSKate Stone         } else {
370505097246SAdrian Prantl           // We have no variables window showing so give the bottom area back
370605097246SAdrian Prantl           // to the source view
370744d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3708b9c1b51eSKate Stone                                    source_bounds.size.height +
3709b9c1b51eSKate Stone                                        registers_window_sp->GetHeight());
371044d93782SGreg Clayton         }
371144d93782SGreg Clayton         main_window_sp->RemoveSubWindow(registers_window_sp.get());
3712b9c1b51eSKate Stone       } else {
371344d93782SGreg Clayton         Rect new_regs_rect;
3714b9c1b51eSKate Stone         if (variables_window_sp) {
371505097246SAdrian Prantl           // We have a variables window, split it into two columns where the
371605097246SAdrian Prantl           // left hand side will be the variables and the right hand side will
371705097246SAdrian Prantl           // be the registers
371844d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
371944d93782SGreg Clayton           Rect new_vars_rect;
3720b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_vars_rect,
3721b9c1b51eSKate Stone                                                    new_regs_rect);
372244d93782SGreg Clayton           variables_window_sp->SetBounds(new_vars_rect);
3723b9c1b51eSKate Stone         } else {
372444d93782SGreg Clayton           // No registers window, grab the bottom part of the source window
372544d93782SGreg Clayton           Rect new_source_rect;
3726b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3727b9c1b51eSKate Stone                                                   new_regs_rect);
372844d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
372944d93782SGreg Clayton         }
3730b9c1b51eSKate Stone         WindowSP new_window_sp =
3731b9c1b51eSKate Stone             main_window_sp->CreateSubWindow("Registers", new_regs_rect, false);
3732b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3733b9c1b51eSKate Stone             WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
373444d93782SGreg Clayton       }
373544d93782SGreg Clayton       touchwin(stdscr);
373644d93782SGreg Clayton     }
373744d93782SGreg Clayton       return MenuActionResult::Handled;
373844d93782SGreg Clayton 
373944d93782SGreg Clayton     case eMenuID_HelpGUIHelp:
37405fdb09bbSGreg Clayton       m_app.GetMainWindow()->CreateHelpSubwindow();
374144d93782SGreg Clayton       return MenuActionResult::Handled;
374244d93782SGreg Clayton 
374344d93782SGreg Clayton     default:
374444d93782SGreg Clayton       break;
374544d93782SGreg Clayton     }
374644d93782SGreg Clayton 
374744d93782SGreg Clayton     return MenuActionResult::NotHandled;
374844d93782SGreg Clayton   }
3749b9c1b51eSKate Stone 
375044d93782SGreg Clayton protected:
375144d93782SGreg Clayton   Application &m_app;
375244d93782SGreg Clayton   Debugger &m_debugger;
375344d93782SGreg Clayton };
375444d93782SGreg Clayton 
3755b9c1b51eSKate Stone class StatusBarWindowDelegate : public WindowDelegate {
375644d93782SGreg Clayton public:
3757b9c1b51eSKate Stone   StatusBarWindowDelegate(Debugger &debugger) : m_debugger(debugger) {
3758b9c1b51eSKate Stone     FormatEntity::Parse("Thread: ${thread.id%tid}", m_format);
375944d93782SGreg Clayton   }
376044d93782SGreg Clayton 
3761315b6884SEugene Zelenko   ~StatusBarWindowDelegate() override = default;
3762bd5ae6b4SGreg Clayton 
3763b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3764b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3765b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
376644d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
376744d93782SGreg Clayton     Thread *thread = exe_ctx.GetThreadPtr();
376844d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
376944d93782SGreg Clayton     window.Erase();
377044d93782SGreg Clayton     window.SetBackground(2);
377144d93782SGreg Clayton     window.MoveCursor(0, 0);
3772b9c1b51eSKate Stone     if (process) {
377344d93782SGreg Clayton       const StateType state = process->GetState();
3774b9c1b51eSKate Stone       window.Printf("Process: %5" PRIu64 " %10s", process->GetID(),
3775b9c1b51eSKate Stone                     StateAsCString(state));
377644d93782SGreg Clayton 
3777b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
37785b031ebcSEd Maste         StreamString strm;
3779b9c1b51eSKate Stone         if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx,
3780b9c1b51eSKate Stone                                            nullptr, nullptr, false, false)) {
378144d93782SGreg Clayton           window.MoveCursor(40, 0);
3782c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), 1);
37835b031ebcSEd Maste         }
378444d93782SGreg Clayton 
378544d93782SGreg Clayton         window.MoveCursor(60, 0);
378644d93782SGreg Clayton         if (frame)
3787b9c1b51eSKate Stone           window.Printf("Frame: %3u  PC = 0x%16.16" PRIx64,
3788b9c1b51eSKate Stone                         frame->GetFrameIndex(),
3789b9c1b51eSKate Stone                         frame->GetFrameCodeAddress().GetOpcodeLoadAddress(
3790b9c1b51eSKate Stone                             exe_ctx.GetTargetPtr()));
3791b9c1b51eSKate Stone       } else if (state == eStateExited) {
379244d93782SGreg Clayton         const char *exit_desc = process->GetExitDescription();
379344d93782SGreg Clayton         const int exit_status = process->GetExitStatus();
379444d93782SGreg Clayton         if (exit_desc && exit_desc[0])
379544d93782SGreg Clayton           window.Printf(" with status = %i (%s)", exit_status, exit_desc);
379644d93782SGreg Clayton         else
379744d93782SGreg Clayton           window.Printf(" with status = %i", exit_status);
379844d93782SGreg Clayton       }
379944d93782SGreg Clayton     }
380044d93782SGreg Clayton     window.DeferredRefresh();
380144d93782SGreg Clayton     return true;
380244d93782SGreg Clayton   }
380344d93782SGreg Clayton 
380444d93782SGreg Clayton protected:
380544d93782SGreg Clayton   Debugger &m_debugger;
3806554f68d3SGreg Clayton   FormatEntity::Entry m_format;
380744d93782SGreg Clayton };
380844d93782SGreg Clayton 
3809b9c1b51eSKate Stone class SourceFileWindowDelegate : public WindowDelegate {
381044d93782SGreg Clayton public:
3811b9c1b51eSKate Stone   SourceFileWindowDelegate(Debugger &debugger)
3812b9c1b51eSKate Stone       : WindowDelegate(), m_debugger(debugger), m_sc(), m_file_sp(),
3813b9c1b51eSKate Stone         m_disassembly_scope(nullptr), m_disassembly_sp(), m_disassembly_range(),
3814b9c1b51eSKate Stone         m_title(), m_line_width(4), m_selected_line(0), m_pc_line(0),
3815b9c1b51eSKate Stone         m_stop_id(0), m_frame_idx(UINT32_MAX), m_first_visible_line(0),
3816b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
381744d93782SGreg Clayton 
3818315b6884SEugene Zelenko   ~SourceFileWindowDelegate() override = default;
381944d93782SGreg Clayton 
3820b9c1b51eSKate Stone   void Update(const SymbolContext &sc) { m_sc = sc; }
382144d93782SGreg Clayton 
3822b9c1b51eSKate Stone   uint32_t NumVisibleLines() const { return m_max_y - m_min_y; }
382344d93782SGreg Clayton 
3824b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
382544d93782SGreg Clayton     return "Source/Disassembly window keyboard shortcuts:";
382644d93782SGreg Clayton   }
382744d93782SGreg Clayton 
3828b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
382944d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
383044d93782SGreg Clayton         {KEY_RETURN, "Run to selected line with one shot breakpoint"},
383144d93782SGreg Clayton         {KEY_UP, "Select previous source line"},
383244d93782SGreg Clayton         {KEY_DOWN, "Select next source line"},
383344d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
383444d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
383544d93782SGreg Clayton         {'b', "Set breakpoint on selected source/disassembly line"},
383644d93782SGreg Clayton         {'c', "Continue process"},
383744d93782SGreg Clayton         {'d', "Detach and resume process"},
383844d93782SGreg Clayton         {'D', "Detach with process suspended"},
383944d93782SGreg Clayton         {'h', "Show help dialog"},
384044d93782SGreg Clayton         {'k', "Kill process"},
384144d93782SGreg Clayton         {'n', "Step over (source line)"},
384244d93782SGreg Clayton         {'N', "Step over (single instruction)"},
384344d93782SGreg Clayton         {'o', "Step out"},
384444d93782SGreg Clayton         {'s', "Step in (source line)"},
384544d93782SGreg Clayton         {'S', "Step in (single instruction)"},
384644d93782SGreg Clayton         {',', "Page up"},
384744d93782SGreg Clayton         {'.', "Page down"},
3848b9c1b51eSKate Stone         {'\0', nullptr}};
384944d93782SGreg Clayton     return g_source_view_key_help;
385044d93782SGreg Clayton   }
385144d93782SGreg Clayton 
3852b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3853b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3854b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
385544d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3856c5dac77aSEugene Zelenko     Thread *thread = nullptr;
385744d93782SGreg Clayton 
385844d93782SGreg Clayton     bool update_location = false;
3859b9c1b51eSKate Stone     if (process) {
386044d93782SGreg Clayton       StateType state = process->GetState();
3861b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
386244d93782SGreg Clayton         // We are stopped, so it is ok to
386344d93782SGreg Clayton         update_location = true;
386444d93782SGreg Clayton       }
386544d93782SGreg Clayton     }
386644d93782SGreg Clayton 
386744d93782SGreg Clayton     m_min_x = 1;
3868ec990867SGreg Clayton     m_min_y = 2;
386944d93782SGreg Clayton     m_max_x = window.GetMaxX() - 1;
387044d93782SGreg Clayton     m_max_y = window.GetMaxY() - 1;
387144d93782SGreg Clayton 
387244d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
387344d93782SGreg Clayton     StackFrameSP frame_sp;
387444d93782SGreg Clayton     bool set_selected_line_to_pc = false;
387544d93782SGreg Clayton 
3876b9c1b51eSKate Stone     if (update_location) {
387744d93782SGreg Clayton       const bool process_alive = process ? process->IsAlive() : false;
387844d93782SGreg Clayton       bool thread_changed = false;
3879b9c1b51eSKate Stone       if (process_alive) {
388044d93782SGreg Clayton         thread = exe_ctx.GetThreadPtr();
3881b9c1b51eSKate Stone         if (thread) {
388244d93782SGreg Clayton           frame_sp = thread->GetSelectedFrame();
388344d93782SGreg Clayton           auto tid = thread->GetID();
388444d93782SGreg Clayton           thread_changed = tid != m_tid;
388544d93782SGreg Clayton           m_tid = tid;
3886b9c1b51eSKate Stone         } else {
3887b9c1b51eSKate Stone           if (m_tid != LLDB_INVALID_THREAD_ID) {
388844d93782SGreg Clayton             thread_changed = true;
388944d93782SGreg Clayton             m_tid = LLDB_INVALID_THREAD_ID;
389044d93782SGreg Clayton           }
389144d93782SGreg Clayton         }
389244d93782SGreg Clayton       }
389344d93782SGreg Clayton       const uint32_t stop_id = process ? process->GetStopID() : 0;
389444d93782SGreg Clayton       const bool stop_id_changed = stop_id != m_stop_id;
389544d93782SGreg Clayton       bool frame_changed = false;
389644d93782SGreg Clayton       m_stop_id = stop_id;
3897ec990867SGreg Clayton       m_title.Clear();
3898b9c1b51eSKate Stone       if (frame_sp) {
389944d93782SGreg Clayton         m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
3900b9c1b51eSKate Stone         if (m_sc.module_sp) {
3901b9c1b51eSKate Stone           m_title.Printf(
3902b9c1b51eSKate Stone               "%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
3903ec990867SGreg Clayton           ConstString func_name = m_sc.GetFunctionName();
3904ec990867SGreg Clayton           if (func_name)
3905ec990867SGreg Clayton             m_title.Printf("`%s", func_name.GetCString());
3906ec990867SGreg Clayton         }
390744d93782SGreg Clayton         const uint32_t frame_idx = frame_sp->GetFrameIndex();
390844d93782SGreg Clayton         frame_changed = frame_idx != m_frame_idx;
390944d93782SGreg Clayton         m_frame_idx = frame_idx;
3910b9c1b51eSKate Stone       } else {
391144d93782SGreg Clayton         m_sc.Clear(true);
391244d93782SGreg Clayton         frame_changed = m_frame_idx != UINT32_MAX;
391344d93782SGreg Clayton         m_frame_idx = UINT32_MAX;
391444d93782SGreg Clayton       }
391544d93782SGreg Clayton 
3916b9c1b51eSKate Stone       const bool context_changed =
3917b9c1b51eSKate Stone           thread_changed || frame_changed || stop_id_changed;
391844d93782SGreg Clayton 
3919b9c1b51eSKate Stone       if (process_alive) {
3920b9c1b51eSKate Stone         if (m_sc.line_entry.IsValid()) {
392144d93782SGreg Clayton           m_pc_line = m_sc.line_entry.line;
392244d93782SGreg Clayton           if (m_pc_line != UINT32_MAX)
392344d93782SGreg Clayton             --m_pc_line; // Convert to zero based line number...
392444d93782SGreg Clayton           // Update the selected line if the stop ID changed...
392544d93782SGreg Clayton           if (context_changed)
392644d93782SGreg Clayton             m_selected_line = m_pc_line;
392744d93782SGreg Clayton 
3928b9c1b51eSKate Stone           if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) {
392905097246SAdrian Prantl             // Same file, nothing to do, we should either have the lines or not
393005097246SAdrian Prantl             // (source file missing)
3931b9c1b51eSKate Stone             if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) {
393244d93782SGreg Clayton               if (m_selected_line >= m_first_visible_line + num_visible_lines)
393344d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
3934b9c1b51eSKate Stone             } else {
393544d93782SGreg Clayton               if (m_selected_line > 10)
393644d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
393744d93782SGreg Clayton               else
393844d93782SGreg Clayton                 m_first_visible_line = 0;
393944d93782SGreg Clayton             }
3940b9c1b51eSKate Stone           } else {
394144d93782SGreg Clayton             // File changed, set selected line to the line with the PC
394244d93782SGreg Clayton             m_selected_line = m_pc_line;
3943b9c1b51eSKate Stone             m_file_sp =
3944b9c1b51eSKate Stone                 m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
3945b9c1b51eSKate Stone             if (m_file_sp) {
394644d93782SGreg Clayton               const size_t num_lines = m_file_sp->GetNumLines();
3947eebf32faSPavel Labath               m_line_width = 1;
394844d93782SGreg Clayton               for (size_t n = num_lines; n >= 10; n = n / 10)
394944d93782SGreg Clayton                 ++m_line_width;
395044d93782SGreg Clayton 
3951b9c1b51eSKate Stone               if (num_lines < num_visible_lines ||
3952b9c1b51eSKate Stone                   m_selected_line < num_visible_lines)
395344d93782SGreg Clayton                 m_first_visible_line = 0;
395444d93782SGreg Clayton               else
395544d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
395644d93782SGreg Clayton             }
395744d93782SGreg Clayton           }
3958b9c1b51eSKate Stone         } else {
395944d93782SGreg Clayton           m_file_sp.reset();
396044d93782SGreg Clayton         }
396144d93782SGreg Clayton 
3962b9c1b51eSKate Stone         if (!m_file_sp || m_file_sp->GetNumLines() == 0) {
396344d93782SGreg Clayton           // Show disassembly
396444d93782SGreg Clayton           bool prefer_file_cache = false;
3965b9c1b51eSKate Stone           if (m_sc.function) {
3966b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.function) {
396744d93782SGreg Clayton               m_disassembly_scope = m_sc.function;
3968b9c1b51eSKate Stone               m_disassembly_sp = m_sc.function->GetInstructions(
3969b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3970b9c1b51eSKate Stone               if (m_disassembly_sp) {
397144d93782SGreg Clayton                 set_selected_line_to_pc = true;
397244d93782SGreg Clayton                 m_disassembly_range = m_sc.function->GetAddressRange();
3973b9c1b51eSKate Stone               } else {
397444d93782SGreg Clayton                 m_disassembly_range.Clear();
397544d93782SGreg Clayton               }
3976b9c1b51eSKate Stone             } else {
397744d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
397844d93782SGreg Clayton             }
3979b9c1b51eSKate Stone           } else if (m_sc.symbol) {
3980b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.symbol) {
398144d93782SGreg Clayton               m_disassembly_scope = m_sc.symbol;
3982b9c1b51eSKate Stone               m_disassembly_sp = m_sc.symbol->GetInstructions(
3983b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3984b9c1b51eSKate Stone               if (m_disassembly_sp) {
398544d93782SGreg Clayton                 set_selected_line_to_pc = true;
3986b9c1b51eSKate Stone                 m_disassembly_range.GetBaseAddress() =
3987b9c1b51eSKate Stone                     m_sc.symbol->GetAddress();
398844d93782SGreg Clayton                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
3989b9c1b51eSKate Stone               } else {
399044d93782SGreg Clayton                 m_disassembly_range.Clear();
399144d93782SGreg Clayton               }
3992b9c1b51eSKate Stone             } else {
399344d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
399444d93782SGreg Clayton             }
399544d93782SGreg Clayton           }
399644d93782SGreg Clayton         }
3997b9c1b51eSKate Stone       } else {
399844d93782SGreg Clayton         m_pc_line = UINT32_MAX;
399944d93782SGreg Clayton       }
400044d93782SGreg Clayton     }
400144d93782SGreg Clayton 
4002ec990867SGreg Clayton     const int window_width = window.GetWidth();
400344d93782SGreg Clayton     window.Erase();
400444d93782SGreg Clayton     window.DrawTitleBox("Sources");
4005b9c1b51eSKate Stone     if (!m_title.GetString().empty()) {
4006ec990867SGreg Clayton       window.AttributeOn(A_REVERSE);
4007ec990867SGreg Clayton       window.MoveCursor(1, 1);
4008ec990867SGreg Clayton       window.PutChar(' ');
4009c156427dSZachary Turner       window.PutCStringTruncated(m_title.GetString().str().c_str(), 1);
4010ec990867SGreg Clayton       int x = window.GetCursorX();
4011b9c1b51eSKate Stone       if (x < window_width - 1) {
4012ec990867SGreg Clayton         window.Printf("%*s", window_width - x - 1, "");
4013ec990867SGreg Clayton       }
4014ec990867SGreg Clayton       window.AttributeOff(A_REVERSE);
4015ec990867SGreg Clayton     }
401644d93782SGreg Clayton 
401744d93782SGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
401844d93782SGreg Clayton     const size_t num_source_lines = GetNumSourceLines();
4019b9c1b51eSKate Stone     if (num_source_lines > 0) {
402044d93782SGreg Clayton       // Display source
402144d93782SGreg Clayton       BreakpointLines bp_lines;
4022b9c1b51eSKate Stone       if (target) {
402344d93782SGreg Clayton         BreakpointList &bp_list = target->GetBreakpointList();
402444d93782SGreg Clayton         const size_t num_bps = bp_list.GetSize();
4025b9c1b51eSKate Stone         for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
402644d93782SGreg Clayton           BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
402744d93782SGreg Clayton           const size_t num_bps_locs = bp_sp->GetNumLocations();
4028b9c1b51eSKate Stone           for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs; ++bp_loc_idx) {
4029b9c1b51eSKate Stone             BreakpointLocationSP bp_loc_sp =
4030b9c1b51eSKate Stone                 bp_sp->GetLocationAtIndex(bp_loc_idx);
403144d93782SGreg Clayton             LineEntry bp_loc_line_entry;
4032b9c1b51eSKate Stone             if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry(
4033b9c1b51eSKate Stone                     bp_loc_line_entry)) {
4034b9c1b51eSKate Stone               if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) {
403544d93782SGreg Clayton                 bp_lines.insert(bp_loc_line_entry.line);
403644d93782SGreg Clayton               }
403744d93782SGreg Clayton             }
403844d93782SGreg Clayton           }
403944d93782SGreg Clayton         }
404044d93782SGreg Clayton       }
404144d93782SGreg Clayton 
404244d93782SGreg Clayton       const attr_t selected_highlight_attr = A_REVERSE;
404344d93782SGreg Clayton       const attr_t pc_highlight_attr = COLOR_PAIR(1);
404444d93782SGreg Clayton 
4045b9c1b51eSKate Stone       for (size_t i = 0; i < num_visible_lines; ++i) {
404644d93782SGreg Clayton         const uint32_t curr_line = m_first_visible_line + i;
4047b9c1b51eSKate Stone         if (curr_line < num_source_lines) {
4048ec990867SGreg Clayton           const int line_y = m_min_y + i;
404944d93782SGreg Clayton           window.MoveCursor(1, line_y);
405044d93782SGreg Clayton           const bool is_pc_line = curr_line == m_pc_line;
405144d93782SGreg Clayton           const bool line_is_selected = m_selected_line == curr_line;
405244d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
405344d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
405444d93782SGreg Clayton           attr_t highlight_attr = 0;
405544d93782SGreg Clayton           attr_t bp_attr = 0;
405644d93782SGreg Clayton           if (is_pc_line)
405744d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
405844d93782SGreg Clayton           else if (line_is_selected)
405944d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
406044d93782SGreg Clayton 
406144d93782SGreg Clayton           if (bp_lines.find(curr_line + 1) != bp_lines.end())
406244d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
406344d93782SGreg Clayton 
406444d93782SGreg Clayton           if (bp_attr)
406544d93782SGreg Clayton             window.AttributeOn(bp_attr);
406644d93782SGreg Clayton 
4067eebf32faSPavel Labath           window.Printf(" %*u ", m_line_width, curr_line + 1);
406844d93782SGreg Clayton 
406944d93782SGreg Clayton           if (bp_attr)
407044d93782SGreg Clayton             window.AttributeOff(bp_attr);
407144d93782SGreg Clayton 
407244d93782SGreg Clayton           window.PutChar(ACS_VLINE);
407344d93782SGreg Clayton           // Mark the line with the PC with a diamond
407444d93782SGreg Clayton           if (is_pc_line)
407544d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
407644d93782SGreg Clayton           else
407744d93782SGreg Clayton             window.PutChar(' ');
407844d93782SGreg Clayton 
407944d93782SGreg Clayton           if (highlight_attr)
408044d93782SGreg Clayton             window.AttributeOn(highlight_attr);
4081b9c1b51eSKate Stone           const uint32_t line_len =
4082b9c1b51eSKate Stone               m_file_sp->GetLineLength(curr_line + 1, false);
408344d93782SGreg Clayton           if (line_len > 0)
408444d93782SGreg Clayton             window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
408544d93782SGreg Clayton 
4086b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4087b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
408844d93782SGreg Clayton             StopInfoSP stop_info_sp;
408944d93782SGreg Clayton             if (thread)
409044d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4091b9c1b51eSKate Stone             if (stop_info_sp) {
409244d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4093b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
409444d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4095ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
409644d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4097b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4098b9c1b51eSKate Stone                 // line_y);
4099b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4100b9c1b51eSKate Stone                               stop_description);
410144d93782SGreg Clayton               }
4102b9c1b51eSKate Stone             } else {
4103ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
410444d93782SGreg Clayton             }
410544d93782SGreg Clayton           }
410644d93782SGreg Clayton           if (highlight_attr)
410744d93782SGreg Clayton             window.AttributeOff(highlight_attr);
4108b9c1b51eSKate Stone         } else {
410944d93782SGreg Clayton           break;
411044d93782SGreg Clayton         }
411144d93782SGreg Clayton       }
4112b9c1b51eSKate Stone     } else {
411344d93782SGreg Clayton       size_t num_disassembly_lines = GetNumDisassemblyLines();
4114b9c1b51eSKate Stone       if (num_disassembly_lines > 0) {
411544d93782SGreg Clayton         // Display disassembly
411644d93782SGreg Clayton         BreakpointAddrs bp_file_addrs;
411744d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
4118b9c1b51eSKate Stone         if (target) {
411944d93782SGreg Clayton           BreakpointList &bp_list = target->GetBreakpointList();
412044d93782SGreg Clayton           const size_t num_bps = bp_list.GetSize();
4121b9c1b51eSKate Stone           for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
412244d93782SGreg Clayton             BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
412344d93782SGreg Clayton             const size_t num_bps_locs = bp_sp->GetNumLocations();
4124b9c1b51eSKate Stone             for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs;
4125b9c1b51eSKate Stone                  ++bp_loc_idx) {
4126b9c1b51eSKate Stone               BreakpointLocationSP bp_loc_sp =
4127b9c1b51eSKate Stone                   bp_sp->GetLocationAtIndex(bp_loc_idx);
412844d93782SGreg Clayton               LineEntry bp_loc_line_entry;
4129b9c1b51eSKate Stone               const lldb::addr_t file_addr =
4130b9c1b51eSKate Stone                   bp_loc_sp->GetAddress().GetFileAddress();
4131b9c1b51eSKate Stone               if (file_addr != LLDB_INVALID_ADDRESS) {
413244d93782SGreg Clayton                 if (m_disassembly_range.ContainsFileAddress(file_addr))
413344d93782SGreg Clayton                   bp_file_addrs.insert(file_addr);
413444d93782SGreg Clayton               }
413544d93782SGreg Clayton             }
413644d93782SGreg Clayton           }
413744d93782SGreg Clayton         }
413844d93782SGreg Clayton 
413944d93782SGreg Clayton         const attr_t selected_highlight_attr = A_REVERSE;
414044d93782SGreg Clayton         const attr_t pc_highlight_attr = COLOR_PAIR(1);
414144d93782SGreg Clayton 
414244d93782SGreg Clayton         StreamString strm;
414344d93782SGreg Clayton 
414444d93782SGreg Clayton         InstructionList &insts = m_disassembly_sp->GetInstructionList();
414544d93782SGreg Clayton         Address pc_address;
414644d93782SGreg Clayton 
414744d93782SGreg Clayton         if (frame_sp)
414844d93782SGreg Clayton           pc_address = frame_sp->GetFrameCodeAddress();
4149b9c1b51eSKate Stone         const uint32_t pc_idx =
4150b9c1b51eSKate Stone             pc_address.IsValid()
4151b9c1b51eSKate Stone                 ? insts.GetIndexOfInstructionAtAddress(pc_address)
4152b9c1b51eSKate Stone                 : UINT32_MAX;
4153b9c1b51eSKate Stone         if (set_selected_line_to_pc) {
415444d93782SGreg Clayton           m_selected_line = pc_idx;
415544d93782SGreg Clayton         }
415644d93782SGreg Clayton 
415744d93782SGreg Clayton         const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
41583985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
415944d93782SGreg Clayton           m_first_visible_line = 0;
416044d93782SGreg Clayton 
4161b9c1b51eSKate Stone         if (pc_idx < num_disassembly_lines) {
41623985c8c6SSaleem Abdulrasool           if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
416344d93782SGreg Clayton               pc_idx >= m_first_visible_line + num_visible_lines)
416444d93782SGreg Clayton             m_first_visible_line = pc_idx - non_visible_pc_offset;
416544d93782SGreg Clayton         }
416644d93782SGreg Clayton 
4167b9c1b51eSKate Stone         for (size_t i = 0; i < num_visible_lines; ++i) {
416844d93782SGreg Clayton           const uint32_t inst_idx = m_first_visible_line + i;
416944d93782SGreg Clayton           Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
417044d93782SGreg Clayton           if (!inst)
417144d93782SGreg Clayton             break;
417244d93782SGreg Clayton 
4173ec990867SGreg Clayton           const int line_y = m_min_y + i;
4174ec990867SGreg Clayton           window.MoveCursor(1, line_y);
417544d93782SGreg Clayton           const bool is_pc_line = frame_sp && inst_idx == pc_idx;
417644d93782SGreg Clayton           const bool line_is_selected = m_selected_line == inst_idx;
417744d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
417844d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
417944d93782SGreg Clayton           attr_t highlight_attr = 0;
418044d93782SGreg Clayton           attr_t bp_attr = 0;
418144d93782SGreg Clayton           if (is_pc_line)
418244d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
418344d93782SGreg Clayton           else if (line_is_selected)
418444d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
418544d93782SGreg Clayton 
4186b9c1b51eSKate Stone           if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) !=
4187b9c1b51eSKate Stone               bp_file_addrs.end())
418844d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
418944d93782SGreg Clayton 
419044d93782SGreg Clayton           if (bp_attr)
419144d93782SGreg Clayton             window.AttributeOn(bp_attr);
419244d93782SGreg Clayton 
4193324a1036SSaleem Abdulrasool           window.Printf(" 0x%16.16llx ",
4194b9c1b51eSKate Stone                         static_cast<unsigned long long>(
4195b9c1b51eSKate Stone                             inst->GetAddress().GetLoadAddress(target)));
419644d93782SGreg Clayton 
419744d93782SGreg Clayton           if (bp_attr)
419844d93782SGreg Clayton             window.AttributeOff(bp_attr);
419944d93782SGreg Clayton 
420044d93782SGreg Clayton           window.PutChar(ACS_VLINE);
420144d93782SGreg Clayton           // Mark the line with the PC with a diamond
420244d93782SGreg Clayton           if (is_pc_line)
420344d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
420444d93782SGreg Clayton           else
420544d93782SGreg Clayton             window.PutChar(' ');
420644d93782SGreg Clayton 
420744d93782SGreg Clayton           if (highlight_attr)
420844d93782SGreg Clayton             window.AttributeOn(highlight_attr);
420944d93782SGreg Clayton 
421044d93782SGreg Clayton           const char *mnemonic = inst->GetMnemonic(&exe_ctx);
421144d93782SGreg Clayton           const char *operands = inst->GetOperands(&exe_ctx);
421244d93782SGreg Clayton           const char *comment = inst->GetComment(&exe_ctx);
421344d93782SGreg Clayton 
4214c5dac77aSEugene Zelenko           if (mnemonic != nullptr && mnemonic[0] == '\0')
4215c5dac77aSEugene Zelenko             mnemonic = nullptr;
4216c5dac77aSEugene Zelenko           if (operands != nullptr && operands[0] == '\0')
4217c5dac77aSEugene Zelenko             operands = nullptr;
4218c5dac77aSEugene Zelenko           if (comment != nullptr && comment[0] == '\0')
4219c5dac77aSEugene Zelenko             comment = nullptr;
422044d93782SGreg Clayton 
422144d93782SGreg Clayton           strm.Clear();
422244d93782SGreg Clayton 
4223c5dac77aSEugene Zelenko           if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
422444d93782SGreg Clayton             strm.Printf("%-8s %-25s ; %s", mnemonic, operands, comment);
4225c5dac77aSEugene Zelenko           else if (mnemonic != nullptr && operands != nullptr)
422644d93782SGreg Clayton             strm.Printf("%-8s %s", mnemonic, operands);
4227c5dac77aSEugene Zelenko           else if (mnemonic != nullptr)
422844d93782SGreg Clayton             strm.Printf("%s", mnemonic);
422944d93782SGreg Clayton 
423044d93782SGreg Clayton           int right_pad = 1;
4231c156427dSZachary Turner           window.PutCStringTruncated(strm.GetData(), right_pad);
423244d93782SGreg Clayton 
4233b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4234b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
423544d93782SGreg Clayton             StopInfoSP stop_info_sp;
423644d93782SGreg Clayton             if (thread)
423744d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4238b9c1b51eSKate Stone             if (stop_info_sp) {
423944d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4240b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
424144d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4242ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
424344d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4244b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4245b9c1b51eSKate Stone                 // line_y);
4246b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4247b9c1b51eSKate Stone                               stop_description);
424844d93782SGreg Clayton               }
4249b9c1b51eSKate Stone             } else {
4250ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
425144d93782SGreg Clayton             }
425244d93782SGreg Clayton           }
425344d93782SGreg Clayton           if (highlight_attr)
425444d93782SGreg Clayton             window.AttributeOff(highlight_attr);
425544d93782SGreg Clayton         }
425644d93782SGreg Clayton       }
425744d93782SGreg Clayton     }
425844d93782SGreg Clayton     window.DeferredRefresh();
425944d93782SGreg Clayton     return true; // Drawing handled
426044d93782SGreg Clayton   }
426144d93782SGreg Clayton 
4262b9c1b51eSKate Stone   size_t GetNumLines() {
426344d93782SGreg Clayton     size_t num_lines = GetNumSourceLines();
426444d93782SGreg Clayton     if (num_lines == 0)
426544d93782SGreg Clayton       num_lines = GetNumDisassemblyLines();
426644d93782SGreg Clayton     return num_lines;
426744d93782SGreg Clayton   }
426844d93782SGreg Clayton 
4269b9c1b51eSKate Stone   size_t GetNumSourceLines() const {
427044d93782SGreg Clayton     if (m_file_sp)
427144d93782SGreg Clayton       return m_file_sp->GetNumLines();
427244d93782SGreg Clayton     return 0;
427344d93782SGreg Clayton   }
4274315b6884SEugene Zelenko 
4275b9c1b51eSKate Stone   size_t GetNumDisassemblyLines() const {
427644d93782SGreg Clayton     if (m_disassembly_sp)
427744d93782SGreg Clayton       return m_disassembly_sp->GetInstructionList().GetSize();
427844d93782SGreg Clayton     return 0;
427944d93782SGreg Clayton   }
428044d93782SGreg Clayton 
4281b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
428244d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
428344d93782SGreg Clayton     const size_t num_lines = GetNumLines();
428444d93782SGreg Clayton 
4285b9c1b51eSKate Stone     switch (c) {
428644d93782SGreg Clayton     case ',':
428744d93782SGreg Clayton     case KEY_PPAGE:
428844d93782SGreg Clayton       // Page up key
42893985c8c6SSaleem Abdulrasool       if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
429044d93782SGreg Clayton         m_first_visible_line -= num_visible_lines;
429144d93782SGreg Clayton       else
429244d93782SGreg Clayton         m_first_visible_line = 0;
429344d93782SGreg Clayton       m_selected_line = m_first_visible_line;
429444d93782SGreg Clayton       return eKeyHandled;
429544d93782SGreg Clayton 
429644d93782SGreg Clayton     case '.':
429744d93782SGreg Clayton     case KEY_NPAGE:
429844d93782SGreg Clayton       // Page down key
429944d93782SGreg Clayton       {
430044d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < num_lines)
430144d93782SGreg Clayton           m_first_visible_line += num_visible_lines;
430244d93782SGreg Clayton         else if (num_lines < num_visible_lines)
430344d93782SGreg Clayton           m_first_visible_line = 0;
430444d93782SGreg Clayton         else
430544d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
430644d93782SGreg Clayton         m_selected_line = m_first_visible_line;
430744d93782SGreg Clayton       }
430844d93782SGreg Clayton       return eKeyHandled;
430944d93782SGreg Clayton 
431044d93782SGreg Clayton     case KEY_UP:
4311b9c1b51eSKate Stone       if (m_selected_line > 0) {
431244d93782SGreg Clayton         m_selected_line--;
43133985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
431444d93782SGreg Clayton           m_first_visible_line = m_selected_line;
431544d93782SGreg Clayton       }
431644d93782SGreg Clayton       return eKeyHandled;
431744d93782SGreg Clayton 
431844d93782SGreg Clayton     case KEY_DOWN:
4319b9c1b51eSKate Stone       if (m_selected_line + 1 < num_lines) {
432044d93782SGreg Clayton         m_selected_line++;
432144d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < m_selected_line)
432244d93782SGreg Clayton           m_first_visible_line++;
432344d93782SGreg Clayton       }
432444d93782SGreg Clayton       return eKeyHandled;
432544d93782SGreg Clayton 
432644d93782SGreg Clayton     case '\r':
432744d93782SGreg Clayton     case '\n':
432844d93782SGreg Clayton     case KEY_ENTER:
432944d93782SGreg Clayton       // Set a breakpoint and run to the line using a one shot breakpoint
4330b9c1b51eSKate Stone       if (GetNumSourceLines() > 0) {
4331b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4332b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4333b9c1b51eSKate Stone         if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) {
4334b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4335b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
433644d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4337b9c1b51eSKate Stone               m_selected_line +
4338b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4339431b1584SAdrian Prantl               0,     // Unspecified column.
43402411167fSJim Ingham               0,     // No offset
434144d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
434244d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
434344d93782SGreg Clayton               false,               // internal
4344055ad9beSIlia K               false,               // request_hardware
4345055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
434644d93782SGreg Clayton           // Make breakpoint one shot
434744d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
434844d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
434944d93782SGreg Clayton         }
4350b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4351b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4352b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4353b9c1b51eSKate Stone                                       .get();
4354b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4355b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4356b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
435744d93782SGreg Clayton           Address addr = inst->GetAddress();
4358b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4359b9c1b51eSKate Stone               addr,   // lldb_private::Address
436044d93782SGreg Clayton               false,  // internal
436144d93782SGreg Clayton               false); // request_hardware
436244d93782SGreg Clayton           // Make breakpoint one shot
436344d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
436444d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
436544d93782SGreg Clayton         }
436644d93782SGreg Clayton       }
436744d93782SGreg Clayton       return eKeyHandled;
436844d93782SGreg Clayton 
436944d93782SGreg Clayton     case 'b': // 'b' == toggle breakpoint on currently selected line
4370b9c1b51eSKate Stone       if (m_selected_line < GetNumSourceLines()) {
4371b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4372b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4373b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
4374b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4375b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
437644d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4377b9c1b51eSKate Stone               m_selected_line +
4378b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4379431b1584SAdrian Prantl               0,     // No column specified.
43802411167fSJim Ingham               0,     // No offset
438144d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
438244d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
438344d93782SGreg Clayton               false,               // internal
4384055ad9beSIlia K               false,               // request_hardware
4385055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
438644d93782SGreg Clayton         }
4387b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4388b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4389b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4390b9c1b51eSKate Stone                                       .get();
4391b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4392b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4393b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
439444d93782SGreg Clayton           Address addr = inst->GetAddress();
4395b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4396b9c1b51eSKate Stone               addr,   // lldb_private::Address
439744d93782SGreg Clayton               false,  // internal
439844d93782SGreg Clayton               false); // request_hardware
439944d93782SGreg Clayton         }
440044d93782SGreg Clayton       }
440144d93782SGreg Clayton       return eKeyHandled;
440244d93782SGreg Clayton 
440344d93782SGreg Clayton     case 'd': // 'd' == detach and let run
440444d93782SGreg Clayton     case 'D': // 'D' == detach and keep stopped
440544d93782SGreg Clayton     {
4406b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4407b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
440844d93782SGreg Clayton       if (exe_ctx.HasProcessScope())
440944d93782SGreg Clayton         exe_ctx.GetProcessRef().Detach(c == 'D');
441044d93782SGreg Clayton     }
441144d93782SGreg Clayton       return eKeyHandled;
441244d93782SGreg Clayton 
441344d93782SGreg Clayton     case 'k':
441444d93782SGreg Clayton       // 'k' == kill
441544d93782SGreg Clayton       {
4416b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4417b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
441844d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
4419ede3193bSJason Molenda           exe_ctx.GetProcessRef().Destroy(false);
442044d93782SGreg Clayton       }
442144d93782SGreg Clayton       return eKeyHandled;
442244d93782SGreg Clayton 
442344d93782SGreg Clayton     case 'c':
442444d93782SGreg Clayton       // 'c' == continue
442544d93782SGreg Clayton       {
4426b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4427b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
442844d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
442944d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
443044d93782SGreg Clayton       }
443144d93782SGreg Clayton       return eKeyHandled;
443244d93782SGreg Clayton 
443344d93782SGreg Clayton     case 'o':
443444d93782SGreg Clayton       // 'o' == step out
443544d93782SGreg Clayton       {
4436b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4437b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4438b9c1b51eSKate Stone         if (exe_ctx.HasThreadScope() &&
4439b9c1b51eSKate Stone             StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
444044d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
444144d93782SGreg Clayton         }
444244d93782SGreg Clayton       }
444344d93782SGreg Clayton       return eKeyHandled;
4444315b6884SEugene Zelenko 
444544d93782SGreg Clayton     case 'n': // 'n' == step over
444644d93782SGreg Clayton     case 'N': // 'N' == step over instruction
444744d93782SGreg Clayton     {
4448b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4449b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4450b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4451b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
445244d93782SGreg Clayton         bool source_step = (c == 'n');
445344d93782SGreg Clayton         exe_ctx.GetThreadRef().StepOver(source_step);
445444d93782SGreg Clayton       }
445544d93782SGreg Clayton     }
445644d93782SGreg Clayton       return eKeyHandled;
4457315b6884SEugene Zelenko 
445844d93782SGreg Clayton     case 's': // 's' == step into
445944d93782SGreg Clayton     case 'S': // 'S' == step into instruction
446044d93782SGreg Clayton     {
4461b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4462b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4463b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4464b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
446544d93782SGreg Clayton         bool source_step = (c == 's');
44664b4b2478SJim Ingham         exe_ctx.GetThreadRef().StepIn(source_step);
446744d93782SGreg Clayton       }
446844d93782SGreg Clayton     }
446944d93782SGreg Clayton       return eKeyHandled;
447044d93782SGreg Clayton 
447144d93782SGreg Clayton     case 'h':
447244d93782SGreg Clayton       window.CreateHelpSubwindow();
447344d93782SGreg Clayton       return eKeyHandled;
447444d93782SGreg Clayton 
447544d93782SGreg Clayton     default:
447644d93782SGreg Clayton       break;
447744d93782SGreg Clayton     }
447844d93782SGreg Clayton     return eKeyNotHandled;
447944d93782SGreg Clayton   }
448044d93782SGreg Clayton 
448144d93782SGreg Clayton protected:
448244d93782SGreg Clayton   typedef std::set<uint32_t> BreakpointLines;
448344d93782SGreg Clayton   typedef std::set<lldb::addr_t> BreakpointAddrs;
448444d93782SGreg Clayton 
448544d93782SGreg Clayton   Debugger &m_debugger;
448644d93782SGreg Clayton   SymbolContext m_sc;
448744d93782SGreg Clayton   SourceManager::FileSP m_file_sp;
448844d93782SGreg Clayton   SymbolContextScope *m_disassembly_scope;
448944d93782SGreg Clayton   lldb::DisassemblerSP m_disassembly_sp;
449044d93782SGreg Clayton   AddressRange m_disassembly_range;
4491ec990867SGreg Clayton   StreamString m_title;
449244d93782SGreg Clayton   lldb::user_id_t m_tid;
449344d93782SGreg Clayton   int m_line_width;
449444d93782SGreg Clayton   uint32_t m_selected_line; // The selected line
449544d93782SGreg Clayton   uint32_t m_pc_line;       // The line with the PC
449644d93782SGreg Clayton   uint32_t m_stop_id;
449744d93782SGreg Clayton   uint32_t m_frame_idx;
449844d93782SGreg Clayton   int m_first_visible_line;
449944d93782SGreg Clayton   int m_min_x;
450044d93782SGreg Clayton   int m_min_y;
450144d93782SGreg Clayton   int m_max_x;
450244d93782SGreg Clayton   int m_max_y;
450344d93782SGreg Clayton };
450444d93782SGreg Clayton 
450544d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = {true};
450644d93782SGreg Clayton 
4507b9c1b51eSKate Stone IOHandlerCursesGUI::IOHandlerCursesGUI(Debugger &debugger)
4508b9c1b51eSKate Stone     : IOHandler(debugger, IOHandler::Type::Curses) {}
450944d93782SGreg Clayton 
4510b9c1b51eSKate Stone void IOHandlerCursesGUI::Activate() {
451144d93782SGreg Clayton   IOHandler::Activate();
4512b9c1b51eSKate Stone   if (!m_app_ap) {
451344d93782SGreg Clayton     m_app_ap.reset(new Application(GetInputFILE(), GetOutputFILE()));
451444d93782SGreg Clayton 
451544d93782SGreg Clayton     // This is both a window and a menu delegate
4516b9c1b51eSKate Stone     std::shared_ptr<ApplicationDelegate> app_delegate_sp(
4517b9c1b51eSKate Stone         new ApplicationDelegate(*m_app_ap, m_debugger));
451844d93782SGreg Clayton 
4519b9c1b51eSKate Stone     MenuDelegateSP app_menu_delegate_sp =
4520b9c1b51eSKate Stone         std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
4521b9c1b51eSKate Stone     MenuSP lldb_menu_sp(
4522b9c1b51eSKate Stone         new Menu("LLDB", "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
4523b9c1b51eSKate Stone     MenuSP exit_menuitem_sp(
4524b9c1b51eSKate Stone         new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
452544d93782SGreg Clayton     exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
4526b9c1b51eSKate Stone     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(
4527b9c1b51eSKate Stone         "About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
452844d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
452944d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(exit_menuitem_sp);
453044d93782SGreg Clayton 
4531b9c1b51eSKate Stone     MenuSP target_menu_sp(new Menu("Target", "F2", KEY_F(2),
4532b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Target));
4533b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4534b9c1b51eSKate Stone         "Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
4535b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4536b9c1b51eSKate Stone         "Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
453744d93782SGreg Clayton 
4538b9c1b51eSKate Stone     MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3),
4539b9c1b51eSKate Stone                                     ApplicationDelegate::eMenuID_Process));
4540b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4541b9c1b51eSKate Stone         "Attach", nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
4542b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4543b9c1b51eSKate Stone         "Detach", nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
4544b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4545b9c1b51eSKate Stone         "Launch", nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
454644d93782SGreg Clayton     process_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
4547b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(
4548b9c1b51eSKate Stone         MenuSP(new Menu("Continue", nullptr, 'c',
4549b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ProcessContinue)));
4550b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4551b9c1b51eSKate Stone         "Halt", nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
4552b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4553b9c1b51eSKate Stone         "Kill", nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
455444d93782SGreg Clayton 
4555b9c1b51eSKate Stone     MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4),
4556b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Thread));
4557b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4558b9c1b51eSKate Stone         "Step In", nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
4559b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(
4560b9c1b51eSKate Stone         MenuSP(new Menu("Step Over", nullptr, 'v',
4561b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ThreadStepOver)));
4562b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4563b9c1b51eSKate Stone         "Step Out", nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
456444d93782SGreg Clayton 
4565b9c1b51eSKate Stone     MenuSP view_menu_sp(
4566b9c1b51eSKate Stone         new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
4567b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4568b9c1b51eSKate Stone         MenuSP(new Menu("Backtrace", nullptr, 'b',
4569b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewBacktrace)));
4570b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4571b9c1b51eSKate Stone         MenuSP(new Menu("Registers", nullptr, 'r',
4572b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewRegisters)));
4573b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(MenuSP(new Menu(
4574b9c1b51eSKate Stone         "Source", nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
4575b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4576b9c1b51eSKate Stone         MenuSP(new Menu("Variables", nullptr, 'v',
4577b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewVariables)));
457844d93782SGreg Clayton 
4579b9c1b51eSKate Stone     MenuSP help_menu_sp(
4580b9c1b51eSKate Stone         new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
4581b9c1b51eSKate Stone     help_menu_sp->AddSubmenu(MenuSP(new Menu(
4582b9c1b51eSKate Stone         "GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
458344d93782SGreg Clayton 
458444d93782SGreg Clayton     m_app_ap->Initialize();
458544d93782SGreg Clayton     WindowSP &main_window_sp = m_app_ap->GetMainWindow();
458644d93782SGreg Clayton 
458744d93782SGreg Clayton     MenuSP menubar_sp(new Menu(Menu::Type::Bar));
458844d93782SGreg Clayton     menubar_sp->AddSubmenu(lldb_menu_sp);
458944d93782SGreg Clayton     menubar_sp->AddSubmenu(target_menu_sp);
459044d93782SGreg Clayton     menubar_sp->AddSubmenu(process_menu_sp);
459144d93782SGreg Clayton     menubar_sp->AddSubmenu(thread_menu_sp);
459244d93782SGreg Clayton     menubar_sp->AddSubmenu(view_menu_sp);
459344d93782SGreg Clayton     menubar_sp->AddSubmenu(help_menu_sp);
459444d93782SGreg Clayton     menubar_sp->SetDelegate(app_menu_delegate_sp);
459544d93782SGreg Clayton 
459644d93782SGreg Clayton     Rect content_bounds = main_window_sp->GetFrame();
459744d93782SGreg Clayton     Rect menubar_bounds = content_bounds.MakeMenuBar();
459844d93782SGreg Clayton     Rect status_bounds = content_bounds.MakeStatusBar();
459944d93782SGreg Clayton     Rect source_bounds;
460044d93782SGreg Clayton     Rect variables_bounds;
460144d93782SGreg Clayton     Rect threads_bounds;
460244d93782SGreg Clayton     Rect source_variables_bounds;
4603b9c1b51eSKate Stone     content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds,
4604b9c1b51eSKate Stone                                            threads_bounds);
4605b9c1b51eSKate Stone     source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds,
4606b9c1b51eSKate Stone                                                       variables_bounds);
460744d93782SGreg Clayton 
4608b9c1b51eSKate Stone     WindowSP menubar_window_sp =
4609b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
461005097246SAdrian Prantl     // Let the menubar get keys if the active window doesn't handle the keys
461105097246SAdrian Prantl     // that are typed so it can respond to menubar key presses.
4612b9c1b51eSKate Stone     menubar_window_sp->SetCanBeActive(
4613b9c1b51eSKate Stone         false); // Don't let the menubar become the active window
461444d93782SGreg Clayton     menubar_window_sp->SetDelegate(menubar_sp);
461544d93782SGreg Clayton 
4616b9c1b51eSKate Stone     WindowSP source_window_sp(
4617b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Source", source_bounds, true));
4618b9c1b51eSKate Stone     WindowSP variables_window_sp(
4619b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Variables", variables_bounds, false));
4620b9c1b51eSKate Stone     WindowSP threads_window_sp(
4621b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Threads", threads_bounds, false));
4622b9c1b51eSKate Stone     WindowSP status_window_sp(
46236bb7e21fSPavel Labath         main_window_sp->CreateSubWindow("Status", status_bounds, false));
4624b9c1b51eSKate Stone     status_window_sp->SetCanBeActive(
4625b9c1b51eSKate Stone         false); // Don't let the status bar become the active window
4626b9c1b51eSKate Stone     main_window_sp->SetDelegate(
4627b9c1b51eSKate Stone         std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
4628b9c1b51eSKate Stone     source_window_sp->SetDelegate(
4629b9c1b51eSKate Stone         WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
4630b9c1b51eSKate Stone     variables_window_sp->SetDelegate(
4631b9c1b51eSKate Stone         WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
4632ec990867SGreg Clayton     TreeDelegateSP thread_delegate_sp(new ThreadsTreeDelegate(m_debugger));
4633b9c1b51eSKate Stone     threads_window_sp->SetDelegate(WindowDelegateSP(
4634b9c1b51eSKate Stone         new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
4635b9c1b51eSKate Stone     status_window_sp->SetDelegate(
4636b9c1b51eSKate Stone         WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
463744d93782SGreg Clayton 
46385fdb09bbSGreg Clayton     // Show the main help window once the first time the curses GUI is launched
46395fdb09bbSGreg Clayton     static bool g_showed_help = false;
4640b9c1b51eSKate Stone     if (!g_showed_help) {
46415fdb09bbSGreg Clayton       g_showed_help = true;
46425fdb09bbSGreg Clayton       main_window_sp->CreateHelpSubwindow();
46435fdb09bbSGreg Clayton     }
46445fdb09bbSGreg Clayton 
464544d93782SGreg Clayton     init_pair(1, COLOR_WHITE, COLOR_BLUE);
464644d93782SGreg Clayton     init_pair(2, COLOR_BLACK, COLOR_WHITE);
464744d93782SGreg Clayton     init_pair(3, COLOR_MAGENTA, COLOR_WHITE);
464844d93782SGreg Clayton     init_pair(4, COLOR_MAGENTA, COLOR_BLACK);
464944d93782SGreg Clayton     init_pair(5, COLOR_RED, COLOR_BLACK);
465044d93782SGreg Clayton   }
465144d93782SGreg Clayton }
465244d93782SGreg Clayton 
4653b9c1b51eSKate Stone void IOHandlerCursesGUI::Deactivate() { m_app_ap->Terminate(); }
465444d93782SGreg Clayton 
4655b9c1b51eSKate Stone void IOHandlerCursesGUI::Run() {
465644d93782SGreg Clayton   m_app_ap->Run(m_debugger);
465744d93782SGreg Clayton   SetIsDone(true);
465844d93782SGreg Clayton }
465944d93782SGreg Clayton 
4660315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
466144d93782SGreg Clayton 
4662b9c1b51eSKate Stone void IOHandlerCursesGUI::Cancel() {}
466344d93782SGreg Clayton 
4664b9c1b51eSKate Stone bool IOHandlerCursesGUI::Interrupt() { return false; }
466544d93782SGreg Clayton 
4666b9c1b51eSKate Stone void IOHandlerCursesGUI::GotEOF() {}
466744d93782SGreg Clayton 
4668315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
4669