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 
555c82608dSHaibo Huang #ifdef _WIN32
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,
767ca15ba7SLawrence D'Anna                 FileSP(),       // 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,
847ca15ba7SLawrence D'Anna                      const lldb::FileSP &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() {
1017ca15ba7SLawrence D'Anna   return (m_input_sp ? m_input_sp->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() {
1137ca15ba7SLawrence D'Anna   return (m_input_sp ? m_input_sp->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 
1247ca15ba7SLawrence D'Anna FileSP &IOHandler::GetInputFileSP() { return m_input_sp; }
12544d93782SGreg Clayton 
1267ca15ba7SLawrence D'Anna StreamFileSP &IOHandler::GetOutputStreamFileSP() { return m_output_sp; }
12744d93782SGreg Clayton 
1287ca15ba7SLawrence D'Anna StreamFileSP &IOHandler::GetErrorStreamFileSP() { return m_error_sp; }
12944d93782SGreg Clayton 
130b9c1b51eSKate Stone bool IOHandler::GetIsInteractive() {
1317ca15ba7SLawrence D'Anna   return GetInputFileSP() ? GetInputFileSP()->GetIsInteractive() : false;
132340b0309SGreg Clayton }
133340b0309SGreg Clayton 
134b9c1b51eSKate Stone bool IOHandler::GetIsRealTerminal() {
1357ca15ba7SLawrence D'Anna   return GetInputFileSP() ? GetInputFileSP()->GetIsRealTerminal() : false;
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 
173ae34ed2cSRaphael Isemann void IOHandlerConfirm::IOHandlerComplete(IOHandler &io_handler,
1742fc20f65SRaphael Isemann                                          CompletionRequest &request) {
1751153dc96SRaphael Isemann   if (request.GetRawCursorPos() != 0)
1761153dc96SRaphael Isemann     return;
1771153dc96SRaphael Isemann   request.AddCompletion(m_default_response ? "y" : "n");
17844d93782SGreg Clayton }
17944d93782SGreg Clayton 
180b9c1b51eSKate Stone void IOHandlerConfirm::IOHandlerInputComplete(IOHandler &io_handler,
181b9c1b51eSKate Stone                                               std::string &line) {
182b9c1b51eSKate Stone   if (line.empty()) {
18344d93782SGreg Clayton     // User just hit enter, set the response to the default
18444d93782SGreg Clayton     m_user_response = m_default_response;
18544d93782SGreg Clayton     io_handler.SetIsDone(true);
18644d93782SGreg Clayton     return;
18744d93782SGreg Clayton   }
18844d93782SGreg Clayton 
189b9c1b51eSKate Stone   if (line.size() == 1) {
190b9c1b51eSKate Stone     switch (line[0]) {
19144d93782SGreg Clayton     case 'y':
19244d93782SGreg Clayton     case 'Y':
19344d93782SGreg Clayton       m_user_response = true;
19444d93782SGreg Clayton       io_handler.SetIsDone(true);
19544d93782SGreg Clayton       return;
19644d93782SGreg Clayton     case 'n':
19744d93782SGreg Clayton     case 'N':
19844d93782SGreg Clayton       m_user_response = false;
19944d93782SGreg Clayton       io_handler.SetIsDone(true);
20044d93782SGreg Clayton       return;
20144d93782SGreg Clayton     default:
20244d93782SGreg Clayton       break;
20344d93782SGreg Clayton     }
20444d93782SGreg Clayton   }
20544d93782SGreg Clayton 
206b9c1b51eSKate Stone   if (line == "yes" || line == "YES" || line == "Yes") {
20744d93782SGreg Clayton     m_user_response = true;
20844d93782SGreg Clayton     io_handler.SetIsDone(true);
209b9c1b51eSKate Stone   } else if (line == "no" || line == "NO" || line == "No") {
21044d93782SGreg Clayton     m_user_response = false;
21144d93782SGreg Clayton     io_handler.SetIsDone(true);
21244d93782SGreg Clayton   }
21344d93782SGreg Clayton }
21444d93782SGreg Clayton 
215ae34ed2cSRaphael Isemann void IOHandlerDelegate::IOHandlerComplete(IOHandler &io_handler,
2162fc20f65SRaphael Isemann                                           CompletionRequest &request) {
217b9c1b51eSKate Stone   switch (m_completion) {
21844d93782SGreg Clayton   case Completion::None:
21944d93782SGreg Clayton     break;
22044d93782SGreg Clayton   case Completion::LLDBCommand:
221ae34ed2cSRaphael Isemann     io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion(request);
222ae34ed2cSRaphael Isemann     break;
223ae34ed2cSRaphael Isemann   case Completion::Expression:
224b9c1b51eSKate Stone     CommandCompletions::InvokeCommonCompletionCallbacks(
225b9c1b51eSKate Stone         io_handler.GetDebugger().GetCommandInterpreter(),
226ae34ed2cSRaphael Isemann         CommandCompletions::eVariablePathCompletion, request, nullptr);
227ae34ed2cSRaphael Isemann     break;
22844d93782SGreg Clayton   }
22944d93782SGreg Clayton }
23044d93782SGreg Clayton 
231b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
232b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
23344d93782SGreg Clayton     const char *editline_name, // Used for saving history files
234514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
235514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
236d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
237b9c1b51eSKate Stone     : IOHandlerEditline(debugger, type,
2387ca15ba7SLawrence D'Anna                         FileSP(),       // Inherit input from top input reader
23944d93782SGreg Clayton                         StreamFileSP(), // Inherit output from top input reader
24044d93782SGreg Clayton                         StreamFileSP(), // Inherit error from top input reader
241340b0309SGreg Clayton                         0,              // Flags
24244d93782SGreg Clayton                         editline_name,  // Used for saving history files
243b9c1b51eSKate Stone                         prompt, continuation_prompt, multi_line, color_prompts,
244d77c2e09SJonas Devlieghere                         line_number_start, delegate, data_recorder) {}
24544d93782SGreg Clayton 
246b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
2477ca15ba7SLawrence D'Anna     Debugger &debugger, IOHandler::Type type, const lldb::FileSP &input_sp,
2487ca15ba7SLawrence D'Anna     const lldb::StreamFileSP &output_sp, const lldb::StreamFileSP &error_sp,
2497ca15ba7SLawrence D'Anna     uint32_t flags,
25044d93782SGreg Clayton     const char *editline_name, // Used for saving history files
251514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
252514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
253d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
254d77c2e09SJonas Devlieghere     : IOHandler(debugger, type, input_sp, output_sp, error_sp, flags,
255d77c2e09SJonas Devlieghere                 data_recorder),
256cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
257d5b44036SJonas Devlieghere       m_editline_up(),
258cacde7dfSTodd Fiala #endif
259b9c1b51eSKate Stone       m_delegate(delegate), m_prompt(), m_continuation_prompt(),
260b9c1b51eSKate Stone       m_current_lines_ptr(nullptr), m_base_line_number(line_number_start),
261b9c1b51eSKate Stone       m_curr_line_idx(UINT32_MAX), m_multi_line(multi_line),
262b9c1b51eSKate Stone       m_color_prompts(color_prompts), m_interrupt_exits(true),
263b9c1b51eSKate Stone       m_editing(false) {
26444d93782SGreg Clayton   SetPrompt(prompt);
26544d93782SGreg Clayton 
266cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
267914b8d98SDeepak Panickal   bool use_editline = false;
268340b0309SGreg Clayton 
269609010d0SLawrence D'Anna   use_editline = GetInputFILE() && GetOutputFILE() && GetErrorFILE() &&
270609010d0SLawrence D'Anna                  m_input_sp && m_input_sp->GetIsRealTerminal();
27144d93782SGreg Clayton 
272b9c1b51eSKate Stone   if (use_editline) {
273d5b44036SJonas Devlieghere     m_editline_up.reset(new Editline(editline_name, GetInputFILE(),
274b9c1b51eSKate Stone                                      GetOutputFILE(), GetErrorFILE(),
275e30f11d9SKate Stone                                      m_color_prompts));
276d5b44036SJonas Devlieghere     m_editline_up->SetIsInputCompleteCallback(IsInputCompleteCallback, this);
277d5b44036SJonas Devlieghere     m_editline_up->SetAutoCompleteCallback(AutoCompleteCallback, this);
278e30f11d9SKate Stone     // See if the delegate supports fixing indentation
279e30f11d9SKate Stone     const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
280b9c1b51eSKate Stone     if (indent_chars) {
281b9c1b51eSKate Stone       // The delegate does support indentation, hook it up so when any
28205097246SAdrian Prantl       // indentation character is typed, the delegate gets a chance to fix it
283d5b44036SJonas Devlieghere       m_editline_up->SetFixIndentationCallback(FixIndentationCallback, this,
284b9c1b51eSKate Stone                                                indent_chars);
285e30f11d9SKate Stone     }
28644d93782SGreg Clayton   }
287cacde7dfSTodd Fiala #endif
288e30f11d9SKate Stone   SetBaseLineNumber(m_base_line_number);
289514d8cd8SZachary Turner   SetPrompt(prompt);
290e30f11d9SKate Stone   SetContinuationPrompt(continuation_prompt);
29144d93782SGreg Clayton }
29244d93782SGreg Clayton 
293b9c1b51eSKate Stone IOHandlerEditline::~IOHandlerEditline() {
294cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
295d5b44036SJonas Devlieghere   m_editline_up.reset();
296cacde7dfSTodd Fiala #endif
29744d93782SGreg Clayton }
29844d93782SGreg Clayton 
299b9c1b51eSKate Stone void IOHandlerEditline::Activate() {
300e30f11d9SKate Stone   IOHandler::Activate();
3010affb582SDave Lee   m_delegate.IOHandlerActivated(*this, GetIsInteractive());
302e30f11d9SKate Stone }
303e30f11d9SKate Stone 
304b9c1b51eSKate Stone void IOHandlerEditline::Deactivate() {
305e30f11d9SKate Stone   IOHandler::Deactivate();
306e30f11d9SKate Stone   m_delegate.IOHandlerDeactivated(*this);
307e30f11d9SKate Stone }
308e30f11d9SKate Stone 
309b9c1b51eSKate Stone bool IOHandlerEditline::GetLine(std::string &line, bool &interrupted) {
310cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
311d5b44036SJonas Devlieghere   if (m_editline_up) {
312d77c2e09SJonas Devlieghere     bool b = m_editline_up->GetLine(line, interrupted);
313d77c2e09SJonas Devlieghere     if (m_data_recorder)
314d77c2e09SJonas Devlieghere       m_data_recorder->Record(line, true);
315d77c2e09SJonas Devlieghere     return b;
316b9c1b51eSKate Stone   } else {
317cacde7dfSTodd Fiala #endif
31844d93782SGreg Clayton     line.clear();
31944d93782SGreg Clayton 
32044d93782SGreg Clayton     FILE *in = GetInputFILE();
321b9c1b51eSKate Stone     if (in) {
322b9c1b51eSKate Stone       if (GetIsInteractive()) {
323c5dac77aSEugene Zelenko         const char *prompt = nullptr;
324e30f11d9SKate Stone 
325e30f11d9SKate Stone         if (m_multi_line && m_curr_line_idx > 0)
326e30f11d9SKate Stone           prompt = GetContinuationPrompt();
327e30f11d9SKate Stone 
328c5dac77aSEugene Zelenko         if (prompt == nullptr)
329e30f11d9SKate Stone           prompt = GetPrompt();
330e30f11d9SKate Stone 
331b9c1b51eSKate Stone         if (prompt && prompt[0]) {
332*5da2bc22SLawrence D'Anna           if (m_output_sp) {
333*5da2bc22SLawrence D'Anna             m_output_sp->Printf("%s", prompt);
334*5da2bc22SLawrence D'Anna             m_output_sp->Flush();
33544d93782SGreg Clayton           }
33644d93782SGreg Clayton         }
33744d93782SGreg Clayton       }
33844d93782SGreg Clayton       char buffer[256];
33944d93782SGreg Clayton       bool done = false;
3400f86e6e7SGreg Clayton       bool got_line = false;
341e034a04eSGreg Clayton       m_editing = true;
342b9c1b51eSKate Stone       while (!done) {
343e7167908SNathan Lanza #ifdef _WIN32
344e7167908SNathan Lanza         // ReadFile on Windows is supposed to set ERROR_OPERATION_ABORTED
345e7167908SNathan Lanza         // according to the docs on MSDN. However, this has evidently been a
346e7167908SNathan Lanza         // known bug since Windows 8. Therefore, we can't detect if a signal
347e7167908SNathan Lanza         // interrupted in the fgets. So pressing ctrl-c causes the repl to end
348e7167908SNathan Lanza         // and the process to exit. A temporary workaround is just to attempt to
349e7167908SNathan Lanza         // fgets twice until this bug is fixed.
350e7167908SNathan Lanza         if (fgets(buffer, sizeof(buffer), in) == nullptr &&
351e7167908SNathan Lanza             fgets(buffer, sizeof(buffer), in) == nullptr) {
352cb305205SNathan Lanza           // this is the equivalent of EINTR for Windows
353cb305205SNathan Lanza           if (GetLastError() == ERROR_OPERATION_ABORTED)
354cb305205SNathan Lanza             continue;
355e7167908SNathan Lanza #else
356b9c1b51eSKate Stone       if (fgets(buffer, sizeof(buffer), in) == nullptr) {
357e7167908SNathan Lanza #endif
358c7797accSGreg Clayton           const int saved_errno = errno;
359c9cf5798SGreg Clayton           if (feof(in))
36044d93782SGreg Clayton             done = true;
361b9c1b51eSKate Stone           else if (ferror(in)) {
362c7797accSGreg Clayton             if (saved_errno != EINTR)
363c7797accSGreg Clayton               done = true;
364c7797accSGreg Clayton           }
365b9c1b51eSKate Stone         } else {
3660f86e6e7SGreg Clayton           got_line = true;
36744d93782SGreg Clayton           size_t buffer_len = strlen(buffer);
36844d93782SGreg Clayton           assert(buffer[buffer_len] == '\0');
36944d93782SGreg Clayton           char last_char = buffer[buffer_len - 1];
370b9c1b51eSKate Stone           if (last_char == '\r' || last_char == '\n') {
37144d93782SGreg Clayton             done = true;
37244d93782SGreg Clayton             // Strip trailing newlines
373b9c1b51eSKate Stone             while (last_char == '\r' || last_char == '\n') {
37444d93782SGreg Clayton               --buffer_len;
37544d93782SGreg Clayton               if (buffer_len == 0)
37644d93782SGreg Clayton                 break;
37744d93782SGreg Clayton               last_char = buffer[buffer_len - 1];
37844d93782SGreg Clayton             }
37944d93782SGreg Clayton           }
38044d93782SGreg Clayton           line.append(buffer, buffer_len);
38144d93782SGreg Clayton         }
38244d93782SGreg Clayton       }
383e034a04eSGreg Clayton       m_editing = false;
384d77c2e09SJonas Devlieghere       if (m_data_recorder && got_line)
385d77c2e09SJonas Devlieghere         m_data_recorder->Record(line, true);
38605097246SAdrian Prantl       // We might have gotten a newline on a line by itself make sure to return
38705097246SAdrian Prantl       // true in this case.
3880f86e6e7SGreg Clayton       return got_line;
389b9c1b51eSKate Stone     } else {
39044d93782SGreg Clayton       // No more input file, we are done...
39144d93782SGreg Clayton       SetIsDone(true);
39244d93782SGreg Clayton     }
393340b0309SGreg Clayton     return false;
394cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
39544d93782SGreg Clayton   }
396cacde7dfSTodd Fiala #endif
39744d93782SGreg Clayton }
39844d93782SGreg Clayton 
399cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
400b9c1b51eSKate Stone bool IOHandlerEditline::IsInputCompleteCallback(Editline *editline,
40144d93782SGreg Clayton                                                 StringList &lines,
402b9c1b51eSKate Stone                                                 void *baton) {
40344d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
404b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader,
405b9c1b51eSKate Stone                                                               lines);
406e30f11d9SKate Stone }
407e30f11d9SKate Stone 
408b9c1b51eSKate Stone int IOHandlerEditline::FixIndentationCallback(Editline *editline,
409e30f11d9SKate Stone                                               const StringList &lines,
410e30f11d9SKate Stone                                               int cursor_position,
411b9c1b51eSKate Stone                                               void *baton) {
412e30f11d9SKate Stone   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
413b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerFixIndentation(
414b9c1b51eSKate Stone       *editline_reader, lines, cursor_position);
41544d93782SGreg Clayton }
41644d93782SGreg Clayton 
417ae34ed2cSRaphael Isemann void IOHandlerEditline::AutoCompleteCallback(CompletionRequest &request,
4182fc20f65SRaphael Isemann                                              void *baton) {
41944d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
42044d93782SGreg Clayton   if (editline_reader)
421ae34ed2cSRaphael Isemann     editline_reader->m_delegate.IOHandlerComplete(*editline_reader, request);
42244d93782SGreg Clayton }
423cacde7dfSTodd Fiala #endif
42444d93782SGreg Clayton 
425b9c1b51eSKate Stone const char *IOHandlerEditline::GetPrompt() {
426cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
427d5b44036SJonas Devlieghere   if (m_editline_up) {
428d5b44036SJonas Devlieghere     return m_editline_up->GetPrompt();
429b9c1b51eSKate Stone   } else {
430cacde7dfSTodd Fiala #endif
431cacde7dfSTodd Fiala     if (m_prompt.empty())
432c5dac77aSEugene Zelenko       return nullptr;
433cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
434cacde7dfSTodd Fiala   }
435cacde7dfSTodd Fiala #endif
43644d93782SGreg Clayton   return m_prompt.c_str();
43744d93782SGreg Clayton }
43844d93782SGreg Clayton 
439514d8cd8SZachary Turner bool IOHandlerEditline::SetPrompt(llvm::StringRef prompt) {
440514d8cd8SZachary Turner   m_prompt = prompt;
441514d8cd8SZachary Turner 
442cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
443d5b44036SJonas Devlieghere   if (m_editline_up)
444d5b44036SJonas Devlieghere     m_editline_up->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
445cacde7dfSTodd Fiala #endif
44644d93782SGreg Clayton   return true;
44744d93782SGreg Clayton }
44844d93782SGreg Clayton 
449b9c1b51eSKate Stone const char *IOHandlerEditline::GetContinuationPrompt() {
450b9c1b51eSKate Stone   return (m_continuation_prompt.empty() ? nullptr
451b9c1b51eSKate Stone                                         : m_continuation_prompt.c_str());
452e30f11d9SKate Stone }
453e30f11d9SKate Stone 
454514d8cd8SZachary Turner void IOHandlerEditline::SetContinuationPrompt(llvm::StringRef prompt) {
455514d8cd8SZachary Turner   m_continuation_prompt = prompt;
456e30f11d9SKate Stone 
457d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
458d5b44036SJonas Devlieghere   if (m_editline_up)
459d5b44036SJonas Devlieghere     m_editline_up->SetContinuationPrompt(m_continuation_prompt.empty()
460b9c1b51eSKate Stone                                              ? nullptr
461b9c1b51eSKate Stone                                              : m_continuation_prompt.c_str());
462d553d00cSZachary Turner #endif
463e30f11d9SKate Stone }
464e30f11d9SKate Stone 
465b9c1b51eSKate Stone void IOHandlerEditline::SetBaseLineNumber(uint32_t line) {
466f6913cd7SGreg Clayton   m_base_line_number = line;
467f6913cd7SGreg Clayton }
468e30f11d9SKate Stone 
469b9c1b51eSKate Stone uint32_t IOHandlerEditline::GetCurrentLineIndex() const {
470d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
471d5b44036SJonas Devlieghere   if (m_editline_up)
472d5b44036SJonas Devlieghere     return m_editline_up->GetCurrentLine();
473e30f11d9SKate Stone #endif
474e30f11d9SKate Stone   return m_curr_line_idx;
475e30f11d9SKate Stone }
476e30f11d9SKate Stone 
477b9c1b51eSKate Stone bool IOHandlerEditline::GetLines(StringList &lines, bool &interrupted) {
478e30f11d9SKate Stone   m_current_lines_ptr = &lines;
479e30f11d9SKate Stone 
48044d93782SGreg Clayton   bool success = false;
481cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
482d5b44036SJonas Devlieghere   if (m_editline_up) {
483d5b44036SJonas Devlieghere     return m_editline_up->GetLines(m_base_line_number, lines, interrupted);
484b9c1b51eSKate Stone   } else {
485cacde7dfSTodd Fiala #endif
486e30f11d9SKate Stone     bool done = false;
48797206d57SZachary Turner     Status error;
48844d93782SGreg Clayton 
489b9c1b51eSKate Stone     while (!done) {
490f6913cd7SGreg Clayton       // Show line numbers if we are asked to
49144d93782SGreg Clayton       std::string line;
492b9c1b51eSKate Stone       if (m_base_line_number > 0 && GetIsInteractive()) {
493*5da2bc22SLawrence D'Anna         if (m_output_sp) {
494*5da2bc22SLawrence D'Anna           m_output_sp->Printf("%u%s",
495*5da2bc22SLawrence D'Anna                               m_base_line_number + (uint32_t)lines.GetSize(),
496b9c1b51eSKate Stone                               GetPrompt() == nullptr ? " " : "");
497f6913cd7SGreg Clayton         }
498*5da2bc22SLawrence D'Anna       }
499f6913cd7SGreg Clayton 
500e30f11d9SKate Stone       m_curr_line_idx = lines.GetSize();
501e30f11d9SKate Stone 
502f0066ad0SGreg Clayton       bool interrupted = false;
503b9c1b51eSKate Stone       if (GetLine(line, interrupted) && !interrupted) {
50444d93782SGreg Clayton         lines.AppendString(line);
505e30f11d9SKate Stone         done = m_delegate.IOHandlerIsInputComplete(*this, lines);
506b9c1b51eSKate Stone       } else {
507e30f11d9SKate Stone         done = true;
50844d93782SGreg Clayton       }
50944d93782SGreg Clayton     }
51044d93782SGreg Clayton     success = lines.GetSize() > 0;
511cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
51244d93782SGreg Clayton   }
513cacde7dfSTodd Fiala #endif
51444d93782SGreg Clayton   return success;
51544d93782SGreg Clayton }
51644d93782SGreg Clayton 
51705097246SAdrian Prantl // Each IOHandler gets to run until it is done. It should read data from the
51805097246SAdrian Prantl // "in" and place output into "out" and "err and return when done.
519b9c1b51eSKate Stone void IOHandlerEditline::Run() {
52044d93782SGreg Clayton   std::string line;
521b9c1b51eSKate Stone   while (IsActive()) {
522f0066ad0SGreg Clayton     bool interrupted = false;
523b9c1b51eSKate Stone     if (m_multi_line) {
52444d93782SGreg Clayton       StringList lines;
525b9c1b51eSKate Stone       if (GetLines(lines, interrupted)) {
526b9c1b51eSKate Stone         if (interrupted) {
527e30f11d9SKate Stone           m_done = m_interrupt_exits;
528e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
529e30f11d9SKate Stone 
530b9c1b51eSKate Stone         } else {
53144d93782SGreg Clayton           line = lines.CopyList();
53244d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
53344d93782SGreg Clayton         }
534b9c1b51eSKate Stone       } else {
53544d93782SGreg Clayton         m_done = true;
53644d93782SGreg Clayton       }
537b9c1b51eSKate Stone     } else {
538b9c1b51eSKate Stone       if (GetLine(line, interrupted)) {
539e30f11d9SKate Stone         if (interrupted)
540e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
541e30f11d9SKate Stone         else
54244d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
543b9c1b51eSKate Stone       } else {
54444d93782SGreg Clayton         m_done = true;
54544d93782SGreg Clayton       }
54644d93782SGreg Clayton     }
54744d93782SGreg Clayton   }
54844d93782SGreg Clayton }
54944d93782SGreg Clayton 
550b9c1b51eSKate Stone void IOHandlerEditline::Cancel() {
551cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
552d5b44036SJonas Devlieghere   if (m_editline_up)
553d5b44036SJonas Devlieghere     m_editline_up->Cancel();
554cacde7dfSTodd Fiala #endif
555e68f5d6bSGreg Clayton }
556e68f5d6bSGreg Clayton 
557b9c1b51eSKate Stone bool IOHandlerEditline::Interrupt() {
558f0066ad0SGreg Clayton   // Let the delgate handle it first
559f0066ad0SGreg Clayton   if (m_delegate.IOHandlerInterrupt(*this))
560f0066ad0SGreg Clayton     return true;
561f0066ad0SGreg Clayton 
562cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
563d5b44036SJonas Devlieghere   if (m_editline_up)
564d5b44036SJonas Devlieghere     return m_editline_up->Interrupt();
565cacde7dfSTodd Fiala #endif
566f0066ad0SGreg Clayton   return false;
56744d93782SGreg Clayton }
56844d93782SGreg Clayton 
569b9c1b51eSKate Stone void IOHandlerEditline::GotEOF() {
570cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
571d5b44036SJonas Devlieghere   if (m_editline_up)
572d5b44036SJonas Devlieghere     m_editline_up->Interrupt();
573cacde7dfSTodd Fiala #endif
57444d93782SGreg Clayton }
57544d93782SGreg Clayton 
576b9c1b51eSKate Stone void IOHandlerEditline::PrintAsync(Stream *stream, const char *s, size_t len) {
5774446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
578d5b44036SJonas Devlieghere   if (m_editline_up)
579d5b44036SJonas Devlieghere     m_editline_up->PrintAsync(stream, s, len);
5804446487dSPavel Labath   else
5814446487dSPavel Labath #endif
582fab31220STed Woodward   {
5838b98f12aSMartin Storsjo #ifdef _WIN32
584341e4789SDawn Perchik     const char *prompt = GetPrompt();
585b9c1b51eSKate Stone     if (prompt) {
586fab31220STed Woodward       // Back up over previous prompt using Windows API
587fab31220STed Woodward       CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
588fab31220STed Woodward       HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
589fab31220STed Woodward       GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
590fab31220STed Woodward       COORD coord = screen_buffer_info.dwCursorPosition;
591fab31220STed Woodward       coord.X -= strlen(prompt);
592fab31220STed Woodward       if (coord.X < 0)
593fab31220STed Woodward         coord.X = 0;
594fab31220STed Woodward       SetConsoleCursorPosition(console_handle, coord);
595fab31220STed Woodward     }
596fab31220STed Woodward #endif
5974446487dSPavel Labath     IOHandler::PrintAsync(stream, s, len);
5988b98f12aSMartin Storsjo #ifdef _WIN32
599fab31220STed Woodward     if (prompt)
6007ca15ba7SLawrence D'Anna       IOHandler::PrintAsync(GetOutputStreamFileSP().get(), prompt,
601b9c1b51eSKate Stone                             strlen(prompt));
602341e4789SDawn Perchik #endif
603fab31220STed Woodward   }
6044446487dSPavel Labath }
6054446487dSPavel Labath 
60605097246SAdrian Prantl // we may want curses to be disabled for some builds for instance, windows
607914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
608914b8d98SDeepak Panickal 
60944d93782SGreg Clayton #define KEY_RETURN 10
61044d93782SGreg Clayton #define KEY_ESCAPE 27
61144d93782SGreg Clayton 
612b9c1b51eSKate Stone namespace curses {
61344d93782SGreg Clayton class Menu;
61444d93782SGreg Clayton class MenuDelegate;
61544d93782SGreg Clayton class Window;
61644d93782SGreg Clayton class WindowDelegate;
61744d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP;
61844d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
61944d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP;
62044d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
62144d93782SGreg Clayton typedef std::vector<MenuSP> Menus;
62244d93782SGreg Clayton typedef std::vector<WindowSP> Windows;
62344d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates;
62444d93782SGreg Clayton 
62544d93782SGreg Clayton #if 0
62644d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
62744d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
62844d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
62944d93782SGreg Clayton #endif
630315b6884SEugene Zelenko 
631b9c1b51eSKate Stone struct Point {
63244d93782SGreg Clayton   int x;
63344d93782SGreg Clayton   int y;
63444d93782SGreg Clayton 
635b9c1b51eSKate Stone   Point(int _x = 0, int _y = 0) : x(_x), y(_y) {}
63644d93782SGreg Clayton 
637b9c1b51eSKate Stone   void Clear() {
63844d93782SGreg Clayton     x = 0;
63944d93782SGreg Clayton     y = 0;
64044d93782SGreg Clayton   }
64144d93782SGreg Clayton 
642b9c1b51eSKate Stone   Point &operator+=(const Point &rhs) {
64344d93782SGreg Clayton     x += rhs.x;
64444d93782SGreg Clayton     y += rhs.y;
64544d93782SGreg Clayton     return *this;
64644d93782SGreg Clayton   }
64744d93782SGreg Clayton 
648b9c1b51eSKate Stone   void Dump() { printf("(x=%i, y=%i)\n", x, y); }
64944d93782SGreg Clayton };
65044d93782SGreg Clayton 
651b9c1b51eSKate Stone bool operator==(const Point &lhs, const Point &rhs) {
65244d93782SGreg Clayton   return lhs.x == rhs.x && lhs.y == rhs.y;
65344d93782SGreg Clayton }
654315b6884SEugene Zelenko 
655b9c1b51eSKate Stone bool operator!=(const Point &lhs, const Point &rhs) {
65644d93782SGreg Clayton   return lhs.x != rhs.x || lhs.y != rhs.y;
65744d93782SGreg Clayton }
65844d93782SGreg Clayton 
659b9c1b51eSKate Stone struct Size {
66044d93782SGreg Clayton   int width;
66144d93782SGreg Clayton   int height;
662b9c1b51eSKate Stone   Size(int w = 0, int h = 0) : width(w), height(h) {}
66344d93782SGreg Clayton 
664b9c1b51eSKate Stone   void Clear() {
66544d93782SGreg Clayton     width = 0;
66644d93782SGreg Clayton     height = 0;
66744d93782SGreg Clayton   }
66844d93782SGreg Clayton 
669b9c1b51eSKate Stone   void Dump() { printf("(w=%i, h=%i)\n", width, height); }
67044d93782SGreg Clayton };
67144d93782SGreg Clayton 
672b9c1b51eSKate Stone bool operator==(const Size &lhs, const Size &rhs) {
67344d93782SGreg Clayton   return lhs.width == rhs.width && lhs.height == rhs.height;
67444d93782SGreg Clayton }
675315b6884SEugene Zelenko 
676b9c1b51eSKate Stone bool operator!=(const Size &lhs, const Size &rhs) {
67744d93782SGreg Clayton   return lhs.width != rhs.width || lhs.height != rhs.height;
67844d93782SGreg Clayton }
67944d93782SGreg Clayton 
680b9c1b51eSKate Stone struct Rect {
68144d93782SGreg Clayton   Point origin;
68244d93782SGreg Clayton   Size size;
68344d93782SGreg Clayton 
684b9c1b51eSKate Stone   Rect() : origin(), size() {}
68544d93782SGreg Clayton 
686b9c1b51eSKate Stone   Rect(const Point &p, const Size &s) : origin(p), size(s) {}
68744d93782SGreg Clayton 
688b9c1b51eSKate Stone   void Clear() {
68944d93782SGreg Clayton     origin.Clear();
69044d93782SGreg Clayton     size.Clear();
69144d93782SGreg Clayton   }
69244d93782SGreg Clayton 
693b9c1b51eSKate Stone   void Dump() {
694b9c1b51eSKate Stone     printf("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width,
695b9c1b51eSKate Stone            size.height);
69644d93782SGreg Clayton   }
69744d93782SGreg Clayton 
698b9c1b51eSKate Stone   void Inset(int w, int h) {
69944d93782SGreg Clayton     if (size.width > w * 2)
70044d93782SGreg Clayton       size.width -= w * 2;
70144d93782SGreg Clayton     origin.x += w;
70244d93782SGreg Clayton 
70344d93782SGreg Clayton     if (size.height > h * 2)
70444d93782SGreg Clayton       size.height -= h * 2;
70544d93782SGreg Clayton     origin.y += h;
70644d93782SGreg Clayton   }
707315b6884SEugene Zelenko 
70805097246SAdrian Prantl   // Return a status bar rectangle which is the last line of this rectangle.
70905097246SAdrian Prantl   // This rectangle will be modified to not include the status bar area.
710b9c1b51eSKate Stone   Rect MakeStatusBar() {
71144d93782SGreg Clayton     Rect status_bar;
712b9c1b51eSKate Stone     if (size.height > 1) {
71344d93782SGreg Clayton       status_bar.origin.x = origin.x;
71444d93782SGreg Clayton       status_bar.origin.y = size.height;
71544d93782SGreg Clayton       status_bar.size.width = size.width;
71644d93782SGreg Clayton       status_bar.size.height = 1;
71744d93782SGreg Clayton       --size.height;
71844d93782SGreg Clayton     }
71944d93782SGreg Clayton     return status_bar;
72044d93782SGreg Clayton   }
72144d93782SGreg Clayton 
72205097246SAdrian Prantl   // Return a menubar rectangle which is the first line of this rectangle. This
72305097246SAdrian Prantl   // rectangle will be modified to not include the menubar area.
724b9c1b51eSKate Stone   Rect MakeMenuBar() {
72544d93782SGreg Clayton     Rect menubar;
726b9c1b51eSKate Stone     if (size.height > 1) {
72744d93782SGreg Clayton       menubar.origin.x = origin.x;
72844d93782SGreg Clayton       menubar.origin.y = origin.y;
72944d93782SGreg Clayton       menubar.size.width = size.width;
73044d93782SGreg Clayton       menubar.size.height = 1;
73144d93782SGreg Clayton       ++origin.y;
73244d93782SGreg Clayton       --size.height;
73344d93782SGreg Clayton     }
73444d93782SGreg Clayton     return menubar;
73544d93782SGreg Clayton   }
73644d93782SGreg Clayton 
737b9c1b51eSKate Stone   void HorizontalSplitPercentage(float top_percentage, Rect &top,
738b9c1b51eSKate Stone                                  Rect &bottom) const {
73944d93782SGreg Clayton     float top_height = top_percentage * size.height;
74044d93782SGreg Clayton     HorizontalSplit(top_height, top, bottom);
74144d93782SGreg Clayton   }
74244d93782SGreg Clayton 
743b9c1b51eSKate Stone   void HorizontalSplit(int top_height, Rect &top, Rect &bottom) const {
74444d93782SGreg Clayton     top = *this;
745b9c1b51eSKate Stone     if (top_height < size.height) {
74644d93782SGreg Clayton       top.size.height = top_height;
74744d93782SGreg Clayton       bottom.origin.x = origin.x;
74844d93782SGreg Clayton       bottom.origin.y = origin.y + top.size.height;
74944d93782SGreg Clayton       bottom.size.width = size.width;
75044d93782SGreg Clayton       bottom.size.height = size.height - top.size.height;
751b9c1b51eSKate Stone     } else {
75244d93782SGreg Clayton       bottom.Clear();
75344d93782SGreg Clayton     }
75444d93782SGreg Clayton   }
75544d93782SGreg Clayton 
756b9c1b51eSKate Stone   void VerticalSplitPercentage(float left_percentage, Rect &left,
757b9c1b51eSKate Stone                                Rect &right) const {
75844d93782SGreg Clayton     float left_width = left_percentage * size.width;
75944d93782SGreg Clayton     VerticalSplit(left_width, left, right);
76044d93782SGreg Clayton   }
76144d93782SGreg Clayton 
762b9c1b51eSKate Stone   void VerticalSplit(int left_width, Rect &left, Rect &right) const {
76344d93782SGreg Clayton     left = *this;
764b9c1b51eSKate Stone     if (left_width < size.width) {
76544d93782SGreg Clayton       left.size.width = left_width;
76644d93782SGreg Clayton       right.origin.x = origin.x + left.size.width;
76744d93782SGreg Clayton       right.origin.y = origin.y;
76844d93782SGreg Clayton       right.size.width = size.width - left.size.width;
76944d93782SGreg Clayton       right.size.height = size.height;
770b9c1b51eSKate Stone     } else {
77144d93782SGreg Clayton       right.Clear();
77244d93782SGreg Clayton     }
77344d93782SGreg Clayton   }
77444d93782SGreg Clayton };
77544d93782SGreg Clayton 
776b9c1b51eSKate Stone bool operator==(const Rect &lhs, const Rect &rhs) {
77744d93782SGreg Clayton   return lhs.origin == rhs.origin && lhs.size == rhs.size;
77844d93782SGreg Clayton }
779315b6884SEugene Zelenko 
780b9c1b51eSKate Stone bool operator!=(const Rect &lhs, const Rect &rhs) {
78144d93782SGreg Clayton   return lhs.origin != rhs.origin || lhs.size != rhs.size;
78244d93782SGreg Clayton }
78344d93782SGreg Clayton 
784b9c1b51eSKate Stone enum HandleCharResult {
78544d93782SGreg Clayton   eKeyNotHandled = 0,
78644d93782SGreg Clayton   eKeyHandled = 1,
78744d93782SGreg Clayton   eQuitApplication = 2
78844d93782SGreg Clayton };
78944d93782SGreg Clayton 
790b9c1b51eSKate Stone enum class MenuActionResult {
79144d93782SGreg Clayton   Handled,
79244d93782SGreg Clayton   NotHandled,
79344d93782SGreg Clayton   Quit // Exit all menus and quit
79444d93782SGreg Clayton };
79544d93782SGreg Clayton 
796b9c1b51eSKate Stone struct KeyHelp {
79744d93782SGreg Clayton   int ch;
79844d93782SGreg Clayton   const char *description;
79944d93782SGreg Clayton };
80044d93782SGreg Clayton 
801b9c1b51eSKate Stone class WindowDelegate {
80244d93782SGreg Clayton public:
803b9c1b51eSKate Stone   virtual ~WindowDelegate() = default;
80444d93782SGreg Clayton 
805b9c1b51eSKate Stone   virtual bool WindowDelegateDraw(Window &window, bool force) {
80644d93782SGreg Clayton     return false; // Drawing not handled
80744d93782SGreg Clayton   }
80844d93782SGreg Clayton 
809b9c1b51eSKate Stone   virtual HandleCharResult WindowDelegateHandleChar(Window &window, int key) {
81044d93782SGreg Clayton     return eKeyNotHandled;
81144d93782SGreg Clayton   }
81244d93782SGreg Clayton 
813b9c1b51eSKate Stone   virtual const char *WindowDelegateGetHelpText() { return nullptr; }
81444d93782SGreg Clayton 
815b9c1b51eSKate Stone   virtual KeyHelp *WindowDelegateGetKeyHelp() { return nullptr; }
81644d93782SGreg Clayton };
81744d93782SGreg Clayton 
818b9c1b51eSKate Stone class HelpDialogDelegate : public WindowDelegate {
81944d93782SGreg Clayton public:
82044d93782SGreg Clayton   HelpDialogDelegate(const char *text, KeyHelp *key_help_array);
82144d93782SGreg Clayton 
822bd5ae6b4SGreg Clayton   ~HelpDialogDelegate() override;
82344d93782SGreg Clayton 
824b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
82544d93782SGreg Clayton 
826b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
82744d93782SGreg Clayton 
828b9c1b51eSKate Stone   size_t GetNumLines() const { return m_text.GetSize(); }
82944d93782SGreg Clayton 
830b9c1b51eSKate Stone   size_t GetMaxLineLength() const { return m_text.GetMaxStringLength(); }
83144d93782SGreg Clayton 
83244d93782SGreg Clayton protected:
83344d93782SGreg Clayton   StringList m_text;
83444d93782SGreg Clayton   int m_first_visible_line;
83544d93782SGreg Clayton };
83644d93782SGreg Clayton 
837b9c1b51eSKate Stone class Window {
83844d93782SGreg Clayton public:
839b9c1b51eSKate Stone   Window(const char *name)
840b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
841b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
842b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(false),
843b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {}
84444d93782SGreg Clayton 
845b9c1b51eSKate Stone   Window(const char *name, WINDOW *w, bool del = true)
846b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
847b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
848b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(del),
849b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
85044d93782SGreg Clayton     if (w)
85144d93782SGreg Clayton       Reset(w);
85244d93782SGreg Clayton   }
85344d93782SGreg Clayton 
854b9c1b51eSKate Stone   Window(const char *name, const Rect &bounds)
855b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_parent(nullptr), m_subwindows(),
856b9c1b51eSKate Stone         m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
857b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(true),
858b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
859b9c1b51eSKate Stone     Reset(::newwin(bounds.size.height, bounds.size.width, bounds.origin.y,
860b9c1b51eSKate Stone                    bounds.origin.y));
86144d93782SGreg Clayton   }
86244d93782SGreg Clayton 
863b9c1b51eSKate Stone   virtual ~Window() {
86444d93782SGreg Clayton     RemoveSubWindows();
86544d93782SGreg Clayton     Reset();
86644d93782SGreg Clayton   }
86744d93782SGreg Clayton 
868b9c1b51eSKate Stone   void Reset(WINDOW *w = nullptr, bool del = true) {
86944d93782SGreg Clayton     if (m_window == w)
87044d93782SGreg Clayton       return;
87144d93782SGreg Clayton 
872b9c1b51eSKate Stone     if (m_panel) {
87344d93782SGreg Clayton       ::del_panel(m_panel);
874c5dac77aSEugene Zelenko       m_panel = nullptr;
87544d93782SGreg Clayton     }
876b9c1b51eSKate Stone     if (m_window && m_delete) {
87744d93782SGreg Clayton       ::delwin(m_window);
878c5dac77aSEugene Zelenko       m_window = nullptr;
87944d93782SGreg Clayton       m_delete = false;
88044d93782SGreg Clayton     }
881b9c1b51eSKate Stone     if (w) {
88244d93782SGreg Clayton       m_window = w;
88344d93782SGreg Clayton       m_panel = ::new_panel(m_window);
88444d93782SGreg Clayton       m_delete = del;
88544d93782SGreg Clayton     }
88644d93782SGreg Clayton   }
88744d93782SGreg Clayton 
88844d93782SGreg Clayton   void AttributeOn(attr_t attr) { ::wattron(m_window, attr); }
88944d93782SGreg Clayton   void AttributeOff(attr_t attr) { ::wattroff(m_window, attr); }
890b9c1b51eSKate Stone   void Box(chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) {
891b9c1b51eSKate Stone     ::box(m_window, v_char, h_char);
892b9c1b51eSKate Stone   }
89344d93782SGreg Clayton   void Clear() { ::wclear(m_window); }
89444d93782SGreg Clayton   void Erase() { ::werase(m_window); }
895b9c1b51eSKate Stone   Rect GetBounds() {
896b9c1b51eSKate Stone     return Rect(GetParentOrigin(), GetSize());
897b9c1b51eSKate Stone   } // Get the rectangle in our parent window
89844d93782SGreg Clayton   int GetChar() { return ::wgetch(m_window); }
89944d93782SGreg Clayton   int GetCursorX() { return getcurx(m_window); }
90044d93782SGreg Clayton   int GetCursorY() { return getcury(m_window); }
901b9c1b51eSKate Stone   Rect GetFrame() {
902b9c1b51eSKate Stone     return Rect(Point(), GetSize());
903b9c1b51eSKate Stone   } // Get our rectangle in our own coordinate system
90444d93782SGreg Clayton   Point GetParentOrigin() { return Point(GetParentX(), GetParentY()); }
90544d93782SGreg Clayton   Size GetSize() { return Size(GetWidth(), GetHeight()); }
90644d93782SGreg Clayton   int GetParentX() { return getparx(m_window); }
90744d93782SGreg Clayton   int GetParentY() { return getpary(m_window); }
90844d93782SGreg Clayton   int GetMaxX() { return getmaxx(m_window); }
90944d93782SGreg Clayton   int GetMaxY() { return getmaxy(m_window); }
91044d93782SGreg Clayton   int GetWidth() { return GetMaxX(); }
91144d93782SGreg Clayton   int GetHeight() { return GetMaxY(); }
91244d93782SGreg Clayton   void MoveCursor(int x, int y) { ::wmove(m_window, y, x); }
91344d93782SGreg Clayton   void MoveWindow(int x, int y) { MoveWindow(Point(x, y)); }
91444d93782SGreg Clayton   void Resize(int w, int h) { ::wresize(m_window, h, w); }
915b9c1b51eSKate Stone   void Resize(const Size &size) {
916b9c1b51eSKate Stone     ::wresize(m_window, size.height, size.width);
917b9c1b51eSKate Stone   }
91844d93782SGreg Clayton   void PutChar(int ch) { ::waddch(m_window, ch); }
91944d93782SGreg Clayton   void PutCString(const char *s, int len = -1) { ::waddnstr(m_window, s, len); }
920b9c1b51eSKate Stone   void SetBackground(int color_pair_idx) {
921b9c1b51eSKate Stone     ::wbkgd(m_window, COLOR_PAIR(color_pair_idx));
922b9c1b51eSKate Stone   }
92344d93782SGreg Clayton 
924b9c1b51eSKate Stone   void PutCStringTruncated(const char *s, int right_pad) {
92544d93782SGreg Clayton     int bytes_left = GetWidth() - GetCursorX();
926b9c1b51eSKate Stone     if (bytes_left > right_pad) {
92744d93782SGreg Clayton       bytes_left -= right_pad;
92844d93782SGreg Clayton       ::waddnstr(m_window, s, bytes_left);
92944d93782SGreg Clayton     }
93044d93782SGreg Clayton   }
93144d93782SGreg Clayton 
932b9c1b51eSKate Stone   void MoveWindow(const Point &origin) {
93344d93782SGreg Clayton     const bool moving_window = origin != GetParentOrigin();
934b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
93544d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
93644d93782SGreg Clayton       Size size = GetSize();
937b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, size.height, size.width, origin.y,
938b9c1b51eSKate Stone                      origin.x),
939b9c1b51eSKate Stone             true);
940b9c1b51eSKate Stone     } else {
94144d93782SGreg Clayton       ::mvwin(m_window, origin.y, origin.x);
94244d93782SGreg Clayton     }
94344d93782SGreg Clayton   }
94444d93782SGreg Clayton 
945b9c1b51eSKate Stone   void SetBounds(const Rect &bounds) {
94644d93782SGreg Clayton     const bool moving_window = bounds.origin != GetParentOrigin();
947b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
94844d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
949b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, bounds.size.height, bounds.size.width,
950b9c1b51eSKate Stone                      bounds.origin.y, bounds.origin.x),
951b9c1b51eSKate Stone             true);
952b9c1b51eSKate Stone     } else {
95344d93782SGreg Clayton       if (moving_window)
95444d93782SGreg Clayton         MoveWindow(bounds.origin);
95544d93782SGreg Clayton       Resize(bounds.size);
95644d93782SGreg Clayton     }
95744d93782SGreg Clayton   }
95844d93782SGreg Clayton 
959b9c1b51eSKate Stone   void Printf(const char *format, ...) __attribute__((format(printf, 2, 3))) {
96044d93782SGreg Clayton     va_list args;
96144d93782SGreg Clayton     va_start(args, format);
96244d93782SGreg Clayton     vwprintw(m_window, format, args);
96344d93782SGreg Clayton     va_end(args);
96444d93782SGreg Clayton   }
96544d93782SGreg Clayton 
966b9c1b51eSKate Stone   void Touch() {
96744d93782SGreg Clayton     ::touchwin(m_window);
96844d93782SGreg Clayton     if (m_parent)
96944d93782SGreg Clayton       m_parent->Touch();
97044d93782SGreg Clayton   }
97144d93782SGreg Clayton 
972b9c1b51eSKate Stone   WindowSP CreateSubWindow(const char *name, const Rect &bounds,
973b9c1b51eSKate Stone                            bool make_active) {
974c6091d2bSJonas Devlieghere     auto get_window = [this, &bounds]() {
975c6091d2bSJonas Devlieghere       return m_window
976c6091d2bSJonas Devlieghere                  ? ::subwin(m_window, bounds.size.height, bounds.size.width,
977c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x)
978c6091d2bSJonas Devlieghere                  : ::newwin(bounds.size.height, bounds.size.width,
979c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x);
980c6091d2bSJonas Devlieghere     };
981c6091d2bSJonas Devlieghere     WindowSP subwindow_sp = std::make_shared<Window>(name, get_window(), true);
982c6091d2bSJonas Devlieghere     subwindow_sp->m_is_subwin = subwindow_sp.operator bool();
98344d93782SGreg Clayton     subwindow_sp->m_parent = this;
984b9c1b51eSKate Stone     if (make_active) {
98544d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
98644d93782SGreg Clayton       m_curr_active_window_idx = m_subwindows.size();
98744d93782SGreg Clayton     }
98844d93782SGreg Clayton     m_subwindows.push_back(subwindow_sp);
98944d93782SGreg Clayton     ::top_panel(subwindow_sp->m_panel);
99044d93782SGreg Clayton     m_needs_update = true;
99144d93782SGreg Clayton     return subwindow_sp;
99244d93782SGreg Clayton   }
99344d93782SGreg Clayton 
994b9c1b51eSKate Stone   bool RemoveSubWindow(Window *window) {
99544d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
99644d93782SGreg Clayton     size_t i = 0;
997b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
998b9c1b51eSKate Stone       if ((*pos).get() == window) {
99944d93782SGreg Clayton         if (m_prev_active_window_idx == i)
100044d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1001b9c1b51eSKate Stone         else if (m_prev_active_window_idx != UINT32_MAX &&
1002b9c1b51eSKate Stone                  m_prev_active_window_idx > i)
100344d93782SGreg Clayton           --m_prev_active_window_idx;
100444d93782SGreg Clayton 
100544d93782SGreg Clayton         if (m_curr_active_window_idx == i)
100644d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
1007b9c1b51eSKate Stone         else if (m_curr_active_window_idx != UINT32_MAX &&
1008b9c1b51eSKate Stone                  m_curr_active_window_idx > i)
100944d93782SGreg Clayton           --m_curr_active_window_idx;
101044d93782SGreg Clayton         window->Erase();
101144d93782SGreg Clayton         m_subwindows.erase(pos);
101244d93782SGreg Clayton         m_needs_update = true;
101344d93782SGreg Clayton         if (m_parent)
101444d93782SGreg Clayton           m_parent->Touch();
101544d93782SGreg Clayton         else
101644d93782SGreg Clayton           ::touchwin(stdscr);
101744d93782SGreg Clayton         return true;
101844d93782SGreg Clayton       }
101944d93782SGreg Clayton     }
102044d93782SGreg Clayton     return false;
102144d93782SGreg Clayton   }
102244d93782SGreg Clayton 
1023b9c1b51eSKate Stone   WindowSP FindSubWindow(const char *name) {
102444d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
102544d93782SGreg Clayton     size_t i = 0;
1026b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
10278d20cfdfSJonas Devlieghere       if ((*pos)->m_name == name)
102844d93782SGreg Clayton         return *pos;
102944d93782SGreg Clayton     }
103044d93782SGreg Clayton     return WindowSP();
103144d93782SGreg Clayton   }
103244d93782SGreg Clayton 
1033b9c1b51eSKate Stone   void RemoveSubWindows() {
103444d93782SGreg Clayton     m_curr_active_window_idx = UINT32_MAX;
103544d93782SGreg Clayton     m_prev_active_window_idx = UINT32_MAX;
103644d93782SGreg Clayton     for (Windows::iterator pos = m_subwindows.begin();
1037b9c1b51eSKate Stone          pos != m_subwindows.end(); pos = m_subwindows.erase(pos)) {
103844d93782SGreg Clayton       (*pos)->Erase();
103944d93782SGreg Clayton     }
104044d93782SGreg Clayton     if (m_parent)
104144d93782SGreg Clayton       m_parent->Touch();
104244d93782SGreg Clayton     else
104344d93782SGreg Clayton       ::touchwin(stdscr);
104444d93782SGreg Clayton   }
104544d93782SGreg Clayton 
1046b9c1b51eSKate Stone   WINDOW *get() { return m_window; }
104744d93782SGreg Clayton 
1048b9c1b51eSKate Stone   operator WINDOW *() { return m_window; }
104944d93782SGreg Clayton 
105044d93782SGreg Clayton   // Window drawing utilities
1051b9c1b51eSKate Stone   void DrawTitleBox(const char *title, const char *bottom_message = nullptr) {
105244d93782SGreg Clayton     attr_t attr = 0;
105344d93782SGreg Clayton     if (IsActive())
105444d93782SGreg Clayton       attr = A_BOLD | COLOR_PAIR(2);
105544d93782SGreg Clayton     else
105644d93782SGreg Clayton       attr = 0;
105744d93782SGreg Clayton     if (attr)
105844d93782SGreg Clayton       AttributeOn(attr);
105944d93782SGreg Clayton 
106044d93782SGreg Clayton     Box();
106144d93782SGreg Clayton     MoveCursor(3, 0);
106244d93782SGreg Clayton 
1063b9c1b51eSKate Stone     if (title && title[0]) {
106444d93782SGreg Clayton       PutChar('<');
106544d93782SGreg Clayton       PutCString(title);
106644d93782SGreg Clayton       PutChar('>');
106744d93782SGreg Clayton     }
106844d93782SGreg Clayton 
1069b9c1b51eSKate Stone     if (bottom_message && bottom_message[0]) {
107044d93782SGreg Clayton       int bottom_message_length = strlen(bottom_message);
107144d93782SGreg Clayton       int x = GetWidth() - 3 - (bottom_message_length + 2);
107244d93782SGreg Clayton 
1073b9c1b51eSKate Stone       if (x > 0) {
107444d93782SGreg Clayton         MoveCursor(x, GetHeight() - 1);
107544d93782SGreg Clayton         PutChar('[');
107644d93782SGreg Clayton         PutCString(bottom_message);
107744d93782SGreg Clayton         PutChar(']');
1078b9c1b51eSKate Stone       } else {
107944d93782SGreg Clayton         MoveCursor(1, GetHeight() - 1);
108044d93782SGreg Clayton         PutChar('[');
108144d93782SGreg Clayton         PutCStringTruncated(bottom_message, 1);
108244d93782SGreg Clayton       }
108344d93782SGreg Clayton     }
108444d93782SGreg Clayton     if (attr)
108544d93782SGreg Clayton       AttributeOff(attr);
108644d93782SGreg Clayton   }
108744d93782SGreg Clayton 
1088b9c1b51eSKate Stone   virtual void Draw(bool force) {
108944d93782SGreg Clayton     if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw(*this, force))
109044d93782SGreg Clayton       return;
109144d93782SGreg Clayton 
109244d93782SGreg Clayton     for (auto &subwindow_sp : m_subwindows)
109344d93782SGreg Clayton       subwindow_sp->Draw(force);
109444d93782SGreg Clayton   }
109544d93782SGreg Clayton 
1096b9c1b51eSKate Stone   bool CreateHelpSubwindow() {
1097b9c1b51eSKate Stone     if (m_delegate_sp) {
109844d93782SGreg Clayton       const char *text = m_delegate_sp->WindowDelegateGetHelpText();
109944d93782SGreg Clayton       KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp();
1100b9c1b51eSKate Stone       if ((text && text[0]) || key_help) {
1101d5b44036SJonas Devlieghere         std::unique_ptr<HelpDialogDelegate> help_delegate_up(
1102b9c1b51eSKate Stone             new HelpDialogDelegate(text, key_help));
1103d5b44036SJonas Devlieghere         const size_t num_lines = help_delegate_up->GetNumLines();
1104d5b44036SJonas Devlieghere         const size_t max_length = help_delegate_up->GetMaxLineLength();
110544d93782SGreg Clayton         Rect bounds = GetBounds();
110644d93782SGreg Clayton         bounds.Inset(1, 1);
1107b9c1b51eSKate Stone         if (max_length + 4 < static_cast<size_t>(bounds.size.width)) {
110844d93782SGreg Clayton           bounds.origin.x += (bounds.size.width - max_length + 4) / 2;
110944d93782SGreg Clayton           bounds.size.width = max_length + 4;
1110b9c1b51eSKate Stone         } else {
1111b9c1b51eSKate Stone           if (bounds.size.width > 100) {
111244d93782SGreg Clayton             const int inset_w = bounds.size.width / 4;
111344d93782SGreg Clayton             bounds.origin.x += inset_w;
111444d93782SGreg Clayton             bounds.size.width -= 2 * inset_w;
111544d93782SGreg Clayton           }
111644d93782SGreg Clayton         }
111744d93782SGreg Clayton 
1118b9c1b51eSKate Stone         if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) {
111944d93782SGreg Clayton           bounds.origin.y += (bounds.size.height - num_lines + 2) / 2;
112044d93782SGreg Clayton           bounds.size.height = num_lines + 2;
1121b9c1b51eSKate Stone         } else {
1122b9c1b51eSKate Stone           if (bounds.size.height > 100) {
112344d93782SGreg Clayton             const int inset_h = bounds.size.height / 4;
112444d93782SGreg Clayton             bounds.origin.y += inset_h;
112544d93782SGreg Clayton             bounds.size.height -= 2 * inset_h;
112644d93782SGreg Clayton           }
112744d93782SGreg Clayton         }
11285fdb09bbSGreg Clayton         WindowSP help_window_sp;
11295fdb09bbSGreg Clayton         Window *parent_window = GetParent();
11305fdb09bbSGreg Clayton         if (parent_window)
11315fdb09bbSGreg Clayton           help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
11325fdb09bbSGreg Clayton         else
11335fdb09bbSGreg Clayton           help_window_sp = CreateSubWindow("Help", bounds, true);
1134b9c1b51eSKate Stone         help_window_sp->SetDelegate(
1135d5b44036SJonas Devlieghere             WindowDelegateSP(help_delegate_up.release()));
113644d93782SGreg Clayton         return true;
113744d93782SGreg Clayton       }
113844d93782SGreg Clayton     }
113944d93782SGreg Clayton     return false;
114044d93782SGreg Clayton   }
114144d93782SGreg Clayton 
1142b9c1b51eSKate Stone   virtual HandleCharResult HandleChar(int key) {
114344d93782SGreg Clayton     // Always check the active window first
114444d93782SGreg Clayton     HandleCharResult result = eKeyNotHandled;
114544d93782SGreg Clayton     WindowSP active_window_sp = GetActiveWindow();
1146b9c1b51eSKate Stone     if (active_window_sp) {
114744d93782SGreg Clayton       result = active_window_sp->HandleChar(key);
114844d93782SGreg Clayton       if (result != eKeyNotHandled)
114944d93782SGreg Clayton         return result;
115044d93782SGreg Clayton     }
115144d93782SGreg Clayton 
1152b9c1b51eSKate Stone     if (m_delegate_sp) {
115344d93782SGreg Clayton       result = m_delegate_sp->WindowDelegateHandleChar(*this, key);
115444d93782SGreg Clayton       if (result != eKeyNotHandled)
115544d93782SGreg Clayton         return result;
115644d93782SGreg Clayton     }
115744d93782SGreg Clayton 
115805097246SAdrian Prantl     // Then check for any windows that want any keys that weren't handled. This
115905097246SAdrian Prantl     // is typically only for a menubar. Make a copy of the subwindows in case
116005097246SAdrian Prantl     // any HandleChar() functions muck with the subwindows. If we don't do
116105097246SAdrian Prantl     // this, we can crash when iterating over the subwindows.
116244d93782SGreg Clayton     Windows subwindows(m_subwindows);
1163b9c1b51eSKate Stone     for (auto subwindow_sp : subwindows) {
1164b9c1b51eSKate Stone       if (!subwindow_sp->m_can_activate) {
116544d93782SGreg Clayton         HandleCharResult result = subwindow_sp->HandleChar(key);
116644d93782SGreg Clayton         if (result != eKeyNotHandled)
116744d93782SGreg Clayton           return result;
116844d93782SGreg Clayton       }
116944d93782SGreg Clayton     }
117044d93782SGreg Clayton 
117144d93782SGreg Clayton     return eKeyNotHandled;
117244d93782SGreg Clayton   }
117344d93782SGreg Clayton 
1174b9c1b51eSKate Stone   WindowSP GetActiveWindow() {
1175b9c1b51eSKate Stone     if (!m_subwindows.empty()) {
1176b9c1b51eSKate Stone       if (m_curr_active_window_idx >= m_subwindows.size()) {
1177b9c1b51eSKate Stone         if (m_prev_active_window_idx < m_subwindows.size()) {
117844d93782SGreg Clayton           m_curr_active_window_idx = m_prev_active_window_idx;
117944d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1180b9c1b51eSKate Stone         } else if (IsActive()) {
118144d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
118244d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
118344d93782SGreg Clayton 
118444d93782SGreg Clayton           // Find first window that wants to be active if this window is active
118544d93782SGreg Clayton           const size_t num_subwindows = m_subwindows.size();
1186b9c1b51eSKate Stone           for (size_t i = 0; i < num_subwindows; ++i) {
1187b9c1b51eSKate Stone             if (m_subwindows[i]->GetCanBeActive()) {
118844d93782SGreg Clayton               m_curr_active_window_idx = i;
118944d93782SGreg Clayton               break;
119044d93782SGreg Clayton             }
119144d93782SGreg Clayton           }
119244d93782SGreg Clayton         }
119344d93782SGreg Clayton       }
119444d93782SGreg Clayton 
119544d93782SGreg Clayton       if (m_curr_active_window_idx < m_subwindows.size())
119644d93782SGreg Clayton         return m_subwindows[m_curr_active_window_idx];
119744d93782SGreg Clayton     }
119844d93782SGreg Clayton     return WindowSP();
119944d93782SGreg Clayton   }
120044d93782SGreg Clayton 
1201b9c1b51eSKate Stone   bool GetCanBeActive() const { return m_can_activate; }
120244d93782SGreg Clayton 
1203b9c1b51eSKate Stone   void SetCanBeActive(bool b) { m_can_activate = b; }
120444d93782SGreg Clayton 
1205b9c1b51eSKate Stone   void SetDelegate(const WindowDelegateSP &delegate_sp) {
120644d93782SGreg Clayton     m_delegate_sp = delegate_sp;
120744d93782SGreg Clayton   }
120844d93782SGreg Clayton 
1209b9c1b51eSKate Stone   Window *GetParent() const { return m_parent; }
121044d93782SGreg Clayton 
1211b9c1b51eSKate Stone   bool IsActive() const {
121244d93782SGreg Clayton     if (m_parent)
121344d93782SGreg Clayton       return m_parent->GetActiveWindow().get() == this;
121444d93782SGreg Clayton     else
121544d93782SGreg Clayton       return true; // Top level window is always active
121644d93782SGreg Clayton   }
121744d93782SGreg Clayton 
1218b9c1b51eSKate Stone   void SelectNextWindowAsActive() {
121944d93782SGreg Clayton     // Move active focus to next window
122044d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1221b9c1b51eSKate Stone     if (m_curr_active_window_idx == UINT32_MAX) {
122244d93782SGreg Clayton       uint32_t idx = 0;
1223b9c1b51eSKate Stone       for (auto subwindow_sp : m_subwindows) {
1224b9c1b51eSKate Stone         if (subwindow_sp->GetCanBeActive()) {
122544d93782SGreg Clayton           m_curr_active_window_idx = idx;
122644d93782SGreg Clayton           break;
122744d93782SGreg Clayton         }
122844d93782SGreg Clayton         ++idx;
122944d93782SGreg Clayton       }
1230b9c1b51eSKate Stone     } else if (m_curr_active_window_idx + 1 < num_subwindows) {
123144d93782SGreg Clayton       bool handled = false;
123244d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1233b9c1b51eSKate Stone       for (size_t idx = m_curr_active_window_idx + 1; idx < num_subwindows;
1234b9c1b51eSKate Stone            ++idx) {
1235b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
123644d93782SGreg Clayton           m_curr_active_window_idx = idx;
123744d93782SGreg Clayton           handled = true;
123844d93782SGreg Clayton           break;
123944d93782SGreg Clayton         }
124044d93782SGreg Clayton       }
1241b9c1b51eSKate Stone       if (!handled) {
1242b9c1b51eSKate Stone         for (size_t idx = 0; idx <= m_prev_active_window_idx; ++idx) {
1243b9c1b51eSKate Stone           if (m_subwindows[idx]->GetCanBeActive()) {
124444d93782SGreg Clayton             m_curr_active_window_idx = idx;
124544d93782SGreg Clayton             break;
124644d93782SGreg Clayton           }
124744d93782SGreg Clayton         }
124844d93782SGreg Clayton       }
1249b9c1b51eSKate Stone     } else {
125044d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1251b9c1b51eSKate Stone       for (size_t idx = 0; idx < num_subwindows; ++idx) {
1252b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
125344d93782SGreg Clayton           m_curr_active_window_idx = idx;
125444d93782SGreg Clayton           break;
125544d93782SGreg Clayton         }
125644d93782SGreg Clayton       }
125744d93782SGreg Clayton     }
125844d93782SGreg Clayton   }
125944d93782SGreg Clayton 
1260b9c1b51eSKate Stone   const char *GetName() const { return m_name.c_str(); }
1261315b6884SEugene Zelenko 
126244d93782SGreg Clayton protected:
126344d93782SGreg Clayton   std::string m_name;
126444d93782SGreg Clayton   WINDOW *m_window;
126544d93782SGreg Clayton   PANEL *m_panel;
126644d93782SGreg Clayton   Window *m_parent;
126744d93782SGreg Clayton   Windows m_subwindows;
126844d93782SGreg Clayton   WindowDelegateSP m_delegate_sp;
126944d93782SGreg Clayton   uint32_t m_curr_active_window_idx;
127044d93782SGreg Clayton   uint32_t m_prev_active_window_idx;
127144d93782SGreg Clayton   bool m_delete;
127244d93782SGreg Clayton   bool m_needs_update;
127344d93782SGreg Clayton   bool m_can_activate;
127444d93782SGreg Clayton   bool m_is_subwin;
127544d93782SGreg Clayton 
127644d93782SGreg Clayton private:
127744d93782SGreg Clayton   DISALLOW_COPY_AND_ASSIGN(Window);
127844d93782SGreg Clayton };
127944d93782SGreg Clayton 
1280b9c1b51eSKate Stone class MenuDelegate {
128144d93782SGreg Clayton public:
1282315b6884SEugene Zelenko   virtual ~MenuDelegate() = default;
128344d93782SGreg Clayton 
1284b9c1b51eSKate Stone   virtual MenuActionResult MenuDelegateAction(Menu &menu) = 0;
128544d93782SGreg Clayton };
128644d93782SGreg Clayton 
1287b9c1b51eSKate Stone class Menu : public WindowDelegate {
128844d93782SGreg Clayton public:
1289b9c1b51eSKate Stone   enum class Type { Invalid, Bar, Item, Separator };
129044d93782SGreg Clayton 
129144d93782SGreg Clayton   // Menubar or separator constructor
129244d93782SGreg Clayton   Menu(Type type);
129344d93782SGreg Clayton 
129444d93782SGreg Clayton   // Menuitem constructor
1295b9c1b51eSKate Stone   Menu(const char *name, const char *key_name, int key_value,
129644d93782SGreg Clayton        uint64_t identifier);
129744d93782SGreg Clayton 
1298315b6884SEugene Zelenko   ~Menu() override = default;
129944d93782SGreg Clayton 
1300b9c1b51eSKate Stone   const MenuDelegateSP &GetDelegate() const { return m_delegate_sp; }
130144d93782SGreg Clayton 
1302b9c1b51eSKate Stone   void SetDelegate(const MenuDelegateSP &delegate_sp) {
130344d93782SGreg Clayton     m_delegate_sp = delegate_sp;
130444d93782SGreg Clayton   }
130544d93782SGreg Clayton 
1306b9c1b51eSKate Stone   void RecalculateNameLengths();
130744d93782SGreg Clayton 
1308b9c1b51eSKate Stone   void AddSubmenu(const MenuSP &menu_sp);
130944d93782SGreg Clayton 
1310b9c1b51eSKate Stone   int DrawAndRunMenu(Window &window);
131144d93782SGreg Clayton 
1312b9c1b51eSKate Stone   void DrawMenuTitle(Window &window, bool highlight);
131344d93782SGreg Clayton 
1314b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
131544d93782SGreg Clayton 
1316b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
131744d93782SGreg Clayton 
1318b9c1b51eSKate Stone   MenuActionResult ActionPrivate(Menu &menu) {
131944d93782SGreg Clayton     MenuActionResult result = MenuActionResult::NotHandled;
1320b9c1b51eSKate Stone     if (m_delegate_sp) {
132144d93782SGreg Clayton       result = m_delegate_sp->MenuDelegateAction(menu);
132244d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
132344d93782SGreg Clayton         return result;
1324b9c1b51eSKate Stone     } else if (m_parent) {
132544d93782SGreg Clayton       result = m_parent->ActionPrivate(menu);
132644d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
132744d93782SGreg Clayton         return result;
132844d93782SGreg Clayton     }
132944d93782SGreg Clayton     return m_canned_result;
133044d93782SGreg Clayton   }
133144d93782SGreg Clayton 
1332b9c1b51eSKate Stone   MenuActionResult Action() {
133305097246SAdrian Prantl     // Call the recursive action so it can try to handle it with the menu
133405097246SAdrian Prantl     // delegate, and if not, try our parent menu
133544d93782SGreg Clayton     return ActionPrivate(*this);
133644d93782SGreg Clayton   }
133744d93782SGreg Clayton 
1338b9c1b51eSKate Stone   void SetCannedResult(MenuActionResult result) { m_canned_result = result; }
133944d93782SGreg Clayton 
1340b9c1b51eSKate Stone   Menus &GetSubmenus() { return m_submenus; }
134144d93782SGreg Clayton 
1342b9c1b51eSKate Stone   const Menus &GetSubmenus() const { return m_submenus; }
134344d93782SGreg Clayton 
1344b9c1b51eSKate Stone   int GetSelectedSubmenuIndex() const { return m_selected; }
134544d93782SGreg Clayton 
1346b9c1b51eSKate Stone   void SetSelectedSubmenuIndex(int idx) { m_selected = idx; }
134744d93782SGreg Clayton 
1348b9c1b51eSKate Stone   Type GetType() const { return m_type; }
134944d93782SGreg Clayton 
1350b9c1b51eSKate Stone   int GetStartingColumn() const { return m_start_col; }
135144d93782SGreg Clayton 
1352b9c1b51eSKate Stone   void SetStartingColumn(int col) { m_start_col = col; }
135344d93782SGreg Clayton 
1354b9c1b51eSKate Stone   int GetKeyValue() const { return m_key_value; }
135544d93782SGreg Clayton 
1356b9c1b51eSKate Stone   std::string &GetName() { return m_name; }
135744d93782SGreg Clayton 
1358b9c1b51eSKate Stone   int GetDrawWidth() const {
135944d93782SGreg Clayton     return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
136044d93782SGreg Clayton   }
136144d93782SGreg Clayton 
1362b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
136344d93782SGreg Clayton 
1364b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
136544d93782SGreg Clayton 
136644d93782SGreg Clayton protected:
136744d93782SGreg Clayton   std::string m_name;
136844d93782SGreg Clayton   std::string m_key_name;
136944d93782SGreg Clayton   uint64_t m_identifier;
137044d93782SGreg Clayton   Type m_type;
137144d93782SGreg Clayton   int m_key_value;
137244d93782SGreg Clayton   int m_start_col;
137344d93782SGreg Clayton   int m_max_submenu_name_length;
137444d93782SGreg Clayton   int m_max_submenu_key_name_length;
137544d93782SGreg Clayton   int m_selected;
137644d93782SGreg Clayton   Menu *m_parent;
137744d93782SGreg Clayton   Menus m_submenus;
137844d93782SGreg Clayton   WindowSP m_menu_window_sp;
137944d93782SGreg Clayton   MenuActionResult m_canned_result;
138044d93782SGreg Clayton   MenuDelegateSP m_delegate_sp;
138144d93782SGreg Clayton };
138244d93782SGreg Clayton 
138344d93782SGreg Clayton // Menubar or separator constructor
1384b9c1b51eSKate Stone Menu::Menu(Type type)
1385b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(0), m_type(type), m_key_value(0),
1386b9c1b51eSKate Stone       m_start_col(0), m_max_submenu_name_length(0),
1387b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1388b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1389b9c1b51eSKate Stone       m_delegate_sp() {}
139044d93782SGreg Clayton 
139144d93782SGreg Clayton // Menuitem constructor
1392b9c1b51eSKate Stone Menu::Menu(const char *name, const char *key_name, int key_value,
1393b9c1b51eSKate Stone            uint64_t identifier)
1394b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(identifier), m_type(Type::Invalid),
1395b9c1b51eSKate Stone       m_key_value(key_value), m_start_col(0), m_max_submenu_name_length(0),
1396b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1397b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1398b9c1b51eSKate Stone       m_delegate_sp() {
1399b9c1b51eSKate Stone   if (name && name[0]) {
140044d93782SGreg Clayton     m_name = name;
140144d93782SGreg Clayton     m_type = Type::Item;
140244d93782SGreg Clayton     if (key_name && key_name[0])
140344d93782SGreg Clayton       m_key_name = key_name;
1404b9c1b51eSKate Stone   } else {
140544d93782SGreg Clayton     m_type = Type::Separator;
140644d93782SGreg Clayton   }
140744d93782SGreg Clayton }
140844d93782SGreg Clayton 
1409b9c1b51eSKate Stone void Menu::RecalculateNameLengths() {
141044d93782SGreg Clayton   m_max_submenu_name_length = 0;
141144d93782SGreg Clayton   m_max_submenu_key_name_length = 0;
141244d93782SGreg Clayton   Menus &submenus = GetSubmenus();
141344d93782SGreg Clayton   const size_t num_submenus = submenus.size();
1414b9c1b51eSKate Stone   for (size_t i = 0; i < num_submenus; ++i) {
141544d93782SGreg Clayton     Menu *submenu = submenus[i].get();
14163985c8c6SSaleem Abdulrasool     if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
141744d93782SGreg Clayton       m_max_submenu_name_length = submenu->m_name.size();
1418b9c1b51eSKate Stone     if (static_cast<size_t>(m_max_submenu_key_name_length) <
1419b9c1b51eSKate Stone         submenu->m_key_name.size())
142044d93782SGreg Clayton       m_max_submenu_key_name_length = submenu->m_key_name.size();
142144d93782SGreg Clayton   }
142244d93782SGreg Clayton }
142344d93782SGreg Clayton 
1424b9c1b51eSKate Stone void Menu::AddSubmenu(const MenuSP &menu_sp) {
142544d93782SGreg Clayton   menu_sp->m_parent = this;
14263985c8c6SSaleem Abdulrasool   if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
142744d93782SGreg Clayton     m_max_submenu_name_length = menu_sp->m_name.size();
1428b9c1b51eSKate Stone   if (static_cast<size_t>(m_max_submenu_key_name_length) <
1429b9c1b51eSKate Stone       menu_sp->m_key_name.size())
143044d93782SGreg Clayton     m_max_submenu_key_name_length = menu_sp->m_key_name.size();
143144d93782SGreg Clayton   m_submenus.push_back(menu_sp);
143244d93782SGreg Clayton }
143344d93782SGreg Clayton 
1434b9c1b51eSKate Stone void Menu::DrawMenuTitle(Window &window, bool highlight) {
1435b9c1b51eSKate Stone   if (m_type == Type::Separator) {
143644d93782SGreg Clayton     window.MoveCursor(0, window.GetCursorY());
143744d93782SGreg Clayton     window.PutChar(ACS_LTEE);
143844d93782SGreg Clayton     int width = window.GetWidth();
1439b9c1b51eSKate Stone     if (width > 2) {
144044d93782SGreg Clayton       width -= 2;
14413985c8c6SSaleem Abdulrasool       for (int i = 0; i < width; ++i)
144244d93782SGreg Clayton         window.PutChar(ACS_HLINE);
144344d93782SGreg Clayton     }
144444d93782SGreg Clayton     window.PutChar(ACS_RTEE);
1445b9c1b51eSKate Stone   } else {
144644d93782SGreg Clayton     const int shortcut_key = m_key_value;
144744d93782SGreg Clayton     bool underlined_shortcut = false;
144844d93782SGreg Clayton     const attr_t hilgight_attr = A_REVERSE;
144944d93782SGreg Clayton     if (highlight)
145044d93782SGreg Clayton       window.AttributeOn(hilgight_attr);
1451b9c1b51eSKate Stone     if (isprint(shortcut_key)) {
145244d93782SGreg Clayton       size_t lower_pos = m_name.find(tolower(shortcut_key));
145344d93782SGreg Clayton       size_t upper_pos = m_name.find(toupper(shortcut_key));
145444d93782SGreg Clayton       const char *name = m_name.c_str();
145544d93782SGreg Clayton       size_t pos = std::min<size_t>(lower_pos, upper_pos);
1456b9c1b51eSKate Stone       if (pos != std::string::npos) {
145744d93782SGreg Clayton         underlined_shortcut = true;
1458b9c1b51eSKate Stone         if (pos > 0) {
145944d93782SGreg Clayton           window.PutCString(name, pos);
146044d93782SGreg Clayton           name += pos;
146144d93782SGreg Clayton         }
146244d93782SGreg Clayton         const attr_t shortcut_attr = A_UNDERLINE | A_BOLD;
146344d93782SGreg Clayton         window.AttributeOn(shortcut_attr);
146444d93782SGreg Clayton         window.PutChar(name[0]);
146544d93782SGreg Clayton         window.AttributeOff(shortcut_attr);
146644d93782SGreg Clayton         name++;
146744d93782SGreg Clayton         if (name[0])
146844d93782SGreg Clayton           window.PutCString(name);
146944d93782SGreg Clayton       }
147044d93782SGreg Clayton     }
147144d93782SGreg Clayton 
1472b9c1b51eSKate Stone     if (!underlined_shortcut) {
147344d93782SGreg Clayton       window.PutCString(m_name.c_str());
147444d93782SGreg Clayton     }
147544d93782SGreg Clayton 
147644d93782SGreg Clayton     if (highlight)
147744d93782SGreg Clayton       window.AttributeOff(hilgight_attr);
147844d93782SGreg Clayton 
1479b9c1b51eSKate Stone     if (m_key_name.empty()) {
1480b9c1b51eSKate Stone       if (!underlined_shortcut && isprint(m_key_value)) {
148144d93782SGreg Clayton         window.AttributeOn(COLOR_PAIR(3));
148244d93782SGreg Clayton         window.Printf(" (%c)", m_key_value);
148344d93782SGreg Clayton         window.AttributeOff(COLOR_PAIR(3));
148444d93782SGreg Clayton       }
1485b9c1b51eSKate Stone     } else {
148644d93782SGreg Clayton       window.AttributeOn(COLOR_PAIR(3));
148744d93782SGreg Clayton       window.Printf(" (%s)", m_key_name.c_str());
148844d93782SGreg Clayton       window.AttributeOff(COLOR_PAIR(3));
148944d93782SGreg Clayton     }
149044d93782SGreg Clayton   }
149144d93782SGreg Clayton }
149244d93782SGreg Clayton 
1493b9c1b51eSKate Stone bool Menu::WindowDelegateDraw(Window &window, bool force) {
149444d93782SGreg Clayton   Menus &submenus = GetSubmenus();
149544d93782SGreg Clayton   const size_t num_submenus = submenus.size();
149644d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
149744d93782SGreg Clayton   Menu::Type menu_type = GetType();
1498b9c1b51eSKate Stone   switch (menu_type) {
1499b9c1b51eSKate Stone   case Menu::Type::Bar: {
150044d93782SGreg Clayton     window.SetBackground(2);
150144d93782SGreg Clayton     window.MoveCursor(0, 0);
1502b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
150344d93782SGreg Clayton       Menu *menu = submenus[i].get();
150444d93782SGreg Clayton       if (i > 0)
150544d93782SGreg Clayton         window.PutChar(' ');
150644d93782SGreg Clayton       menu->SetStartingColumn(window.GetCursorX());
150744d93782SGreg Clayton       window.PutCString("| ");
150844d93782SGreg Clayton       menu->DrawMenuTitle(window, false);
150944d93782SGreg Clayton     }
151044d93782SGreg Clayton     window.PutCString(" |");
1511b9c1b51eSKate Stone   } break;
151244d93782SGreg Clayton 
1513b9c1b51eSKate Stone   case Menu::Type::Item: {
151444d93782SGreg Clayton     int y = 1;
151544d93782SGreg Clayton     int x = 3;
151644d93782SGreg Clayton     // Draw the menu
151744d93782SGreg Clayton     int cursor_x = 0;
151844d93782SGreg Clayton     int cursor_y = 0;
151944d93782SGreg Clayton     window.Erase();
152044d93782SGreg Clayton     window.SetBackground(2);
152144d93782SGreg Clayton     window.Box();
1522b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
1523b9c1b51eSKate Stone       const bool is_selected = (i == static_cast<size_t>(selected_idx));
152444d93782SGreg Clayton       window.MoveCursor(x, y + i);
1525b9c1b51eSKate Stone       if (is_selected) {
152644d93782SGreg Clayton         // Remember where we want the cursor to be
152744d93782SGreg Clayton         cursor_x = x - 1;
152844d93782SGreg Clayton         cursor_y = y + i;
152944d93782SGreg Clayton       }
153044d93782SGreg Clayton       submenus[i]->DrawMenuTitle(window, is_selected);
153144d93782SGreg Clayton     }
153244d93782SGreg Clayton     window.MoveCursor(cursor_x, cursor_y);
1533b9c1b51eSKate Stone   } break;
153444d93782SGreg Clayton 
153544d93782SGreg Clayton   default:
153644d93782SGreg Clayton   case Menu::Type::Separator:
153744d93782SGreg Clayton     break;
153844d93782SGreg Clayton   }
153944d93782SGreg Clayton   return true; // Drawing handled...
154044d93782SGreg Clayton }
154144d93782SGreg Clayton 
1542b9c1b51eSKate Stone HandleCharResult Menu::WindowDelegateHandleChar(Window &window, int key) {
154344d93782SGreg Clayton   HandleCharResult result = eKeyNotHandled;
154444d93782SGreg Clayton 
154544d93782SGreg Clayton   Menus &submenus = GetSubmenus();
154644d93782SGreg Clayton   const size_t num_submenus = submenus.size();
154744d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
154844d93782SGreg Clayton   Menu::Type menu_type = GetType();
1549b9c1b51eSKate Stone   if (menu_type == Menu::Type::Bar) {
155044d93782SGreg Clayton     MenuSP run_menu_sp;
1551b9c1b51eSKate Stone     switch (key) {
155244d93782SGreg Clayton     case KEY_DOWN:
155344d93782SGreg Clayton     case KEY_UP:
155444d93782SGreg Clayton       // Show last menu or first menu
15553985c8c6SSaleem Abdulrasool       if (selected_idx < static_cast<int>(num_submenus))
155644d93782SGreg Clayton         run_menu_sp = submenus[selected_idx];
155744d93782SGreg Clayton       else if (!submenus.empty())
155844d93782SGreg Clayton         run_menu_sp = submenus.front();
155944d93782SGreg Clayton       result = eKeyHandled;
156044d93782SGreg Clayton       break;
156144d93782SGreg Clayton 
156244d93782SGreg Clayton     case KEY_RIGHT:
156344d93782SGreg Clayton       ++m_selected;
15643985c8c6SSaleem Abdulrasool       if (m_selected >= static_cast<int>(num_submenus))
156544d93782SGreg Clayton         m_selected = 0;
15663985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
156744d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
156844d93782SGreg Clayton       else if (!submenus.empty())
156944d93782SGreg Clayton         run_menu_sp = submenus.front();
157044d93782SGreg Clayton       result = eKeyHandled;
157144d93782SGreg Clayton       break;
157244d93782SGreg Clayton 
157344d93782SGreg Clayton     case KEY_LEFT:
157444d93782SGreg Clayton       --m_selected;
157544d93782SGreg Clayton       if (m_selected < 0)
157644d93782SGreg Clayton         m_selected = num_submenus - 1;
15773985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
157844d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
157944d93782SGreg Clayton       else if (!submenus.empty())
158044d93782SGreg Clayton         run_menu_sp = submenus.front();
158144d93782SGreg Clayton       result = eKeyHandled;
158244d93782SGreg Clayton       break;
158344d93782SGreg Clayton 
158444d93782SGreg Clayton     default:
1585b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
1586b9c1b51eSKate Stone         if (submenus[i]->GetKeyValue() == key) {
158744d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
158844d93782SGreg Clayton           run_menu_sp = submenus[i];
158944d93782SGreg Clayton           result = eKeyHandled;
159044d93782SGreg Clayton           break;
159144d93782SGreg Clayton         }
159244d93782SGreg Clayton       }
159344d93782SGreg Clayton       break;
159444d93782SGreg Clayton     }
159544d93782SGreg Clayton 
1596b9c1b51eSKate Stone     if (run_menu_sp) {
159705097246SAdrian Prantl       // Run the action on this menu in case we need to populate the menu with
159805097246SAdrian Prantl       // dynamic content and also in case check marks, and any other menu
159905097246SAdrian Prantl       // decorations need to be calculated
160044d93782SGreg Clayton       if (run_menu_sp->Action() == MenuActionResult::Quit)
160144d93782SGreg Clayton         return eQuitApplication;
160244d93782SGreg Clayton 
160344d93782SGreg Clayton       Rect menu_bounds;
160444d93782SGreg Clayton       menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
160544d93782SGreg Clayton       menu_bounds.origin.y = 1;
160644d93782SGreg Clayton       menu_bounds.size.width = run_menu_sp->GetDrawWidth();
160744d93782SGreg Clayton       menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
160844d93782SGreg Clayton       if (m_menu_window_sp)
160944d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
161044d93782SGreg Clayton 
1611b9c1b51eSKate Stone       m_menu_window_sp = window.GetParent()->CreateSubWindow(
1612b9c1b51eSKate Stone           run_menu_sp->GetName().c_str(), menu_bounds, true);
161344d93782SGreg Clayton       m_menu_window_sp->SetDelegate(run_menu_sp);
161444d93782SGreg Clayton     }
1615b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Item) {
1616b9c1b51eSKate Stone     switch (key) {
161744d93782SGreg Clayton     case KEY_DOWN:
1618b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
161944d93782SGreg Clayton         const int start_select = m_selected;
1620b9c1b51eSKate Stone         while (++m_selected != start_select) {
16213985c8c6SSaleem Abdulrasool           if (static_cast<size_t>(m_selected) >= num_submenus)
162244d93782SGreg Clayton             m_selected = 0;
162344d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
162444d93782SGreg Clayton             continue;
162544d93782SGreg Clayton           else
162644d93782SGreg Clayton             break;
162744d93782SGreg Clayton         }
162844d93782SGreg Clayton         return eKeyHandled;
162944d93782SGreg Clayton       }
163044d93782SGreg Clayton       break;
163144d93782SGreg Clayton 
163244d93782SGreg Clayton     case KEY_UP:
1633b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
163444d93782SGreg Clayton         const int start_select = m_selected;
1635b9c1b51eSKate Stone         while (--m_selected != start_select) {
16363985c8c6SSaleem Abdulrasool           if (m_selected < static_cast<int>(0))
163744d93782SGreg Clayton             m_selected = num_submenus - 1;
163844d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
163944d93782SGreg Clayton             continue;
164044d93782SGreg Clayton           else
164144d93782SGreg Clayton             break;
164244d93782SGreg Clayton         }
164344d93782SGreg Clayton         return eKeyHandled;
164444d93782SGreg Clayton       }
164544d93782SGreg Clayton       break;
164644d93782SGreg Clayton 
164744d93782SGreg Clayton     case KEY_RETURN:
1648b9c1b51eSKate Stone       if (static_cast<size_t>(selected_idx) < num_submenus) {
164944d93782SGreg Clayton         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
165044d93782SGreg Clayton           return eQuitApplication;
165144d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
165244d93782SGreg Clayton         return eKeyHandled;
165344d93782SGreg Clayton       }
165444d93782SGreg Clayton       break;
165544d93782SGreg Clayton 
1656b9c1b51eSKate Stone     case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in
1657b9c1b51eSKate Stone                      // case other chars are entered for escaped sequences
165844d93782SGreg Clayton       window.GetParent()->RemoveSubWindow(&window);
165944d93782SGreg Clayton       return eKeyHandled;
166044d93782SGreg Clayton 
166144d93782SGreg Clayton     default:
1662b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
166344d93782SGreg Clayton         Menu *menu = submenus[i].get();
1664b9c1b51eSKate Stone         if (menu->GetKeyValue() == key) {
166544d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
166644d93782SGreg Clayton           window.GetParent()->RemoveSubWindow(&window);
166744d93782SGreg Clayton           if (menu->Action() == MenuActionResult::Quit)
166844d93782SGreg Clayton             return eQuitApplication;
166944d93782SGreg Clayton           return eKeyHandled;
167044d93782SGreg Clayton         }
167144d93782SGreg Clayton       }
167244d93782SGreg Clayton       break;
167344d93782SGreg Clayton     }
1674b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Separator) {
167544d93782SGreg Clayton   }
167644d93782SGreg Clayton   return result;
167744d93782SGreg Clayton }
167844d93782SGreg Clayton 
1679b9c1b51eSKate Stone class Application {
168044d93782SGreg Clayton public:
1681b9c1b51eSKate Stone   Application(FILE *in, FILE *out)
1682b9c1b51eSKate Stone       : m_window_sp(), m_screen(nullptr), m_in(in), m_out(out) {}
168344d93782SGreg Clayton 
1684b9c1b51eSKate Stone   ~Application() {
168544d93782SGreg Clayton     m_window_delegates.clear();
168644d93782SGreg Clayton     m_window_sp.reset();
1687b9c1b51eSKate Stone     if (m_screen) {
168844d93782SGreg Clayton       ::delscreen(m_screen);
1689c5dac77aSEugene Zelenko       m_screen = nullptr;
169044d93782SGreg Clayton     }
169144d93782SGreg Clayton   }
169244d93782SGreg Clayton 
1693b9c1b51eSKate Stone   void Initialize() {
169444d93782SGreg Clayton     ::setlocale(LC_ALL, "");
169544d93782SGreg Clayton     ::setlocale(LC_CTYPE, "");
1696c5dac77aSEugene Zelenko     m_screen = ::newterm(nullptr, m_out, m_in);
169744d93782SGreg Clayton     ::start_color();
169844d93782SGreg Clayton     ::curs_set(0);
169944d93782SGreg Clayton     ::noecho();
170044d93782SGreg Clayton     ::keypad(stdscr, TRUE);
170144d93782SGreg Clayton   }
170244d93782SGreg Clayton 
1703b9c1b51eSKate Stone   void Terminate() { ::endwin(); }
170444d93782SGreg Clayton 
1705b9c1b51eSKate Stone   void Run(Debugger &debugger) {
170644d93782SGreg Clayton     bool done = false;
170744d93782SGreg Clayton     int delay_in_tenths_of_a_second = 1;
170844d93782SGreg Clayton 
170905097246SAdrian Prantl     // Alas the threading model in curses is a bit lame so we need to resort to
171005097246SAdrian Prantl     // polling every 0.5 seconds. We could poll for stdin ourselves and then
171105097246SAdrian Prantl     // pass the keys down but then we need to translate all of the escape
171205097246SAdrian Prantl     // sequences ourselves. So we resort to polling for input because we need
171305097246SAdrian Prantl     // to receive async process events while in this loop.
171444d93782SGreg Clayton 
1715b9c1b51eSKate Stone     halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths
1716b9c1b51eSKate Stone                                             // of seconds seconds when calling
1717b9c1b51eSKate Stone                                             // Window::GetChar()
171844d93782SGreg Clayton 
1719b9c1b51eSKate Stone     ListenerSP listener_sp(
1720b9c1b51eSKate Stone         Listener::MakeListener("lldb.IOHandler.curses.Application"));
172144d93782SGreg Clayton     ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
172244d93782SGreg Clayton     ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
172344d93782SGreg Clayton     ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
172444d93782SGreg Clayton     debugger.EnableForwardEvents(listener_sp);
172544d93782SGreg Clayton 
172644d93782SGreg Clayton     bool update = true;
172744d93782SGreg Clayton #if defined(__APPLE__)
172844d93782SGreg Clayton     std::deque<int> escape_chars;
172944d93782SGreg Clayton #endif
173044d93782SGreg Clayton 
1731b9c1b51eSKate Stone     while (!done) {
1732b9c1b51eSKate Stone       if (update) {
173344d93782SGreg Clayton         m_window_sp->Draw(false);
173405097246SAdrian Prantl         // All windows should be calling Window::DeferredRefresh() instead of
173505097246SAdrian Prantl         // Window::Refresh() so we can do a single update and avoid any screen
173605097246SAdrian Prantl         // blinking
173744d93782SGreg Clayton         update_panels();
173844d93782SGreg Clayton 
1739b9c1b51eSKate Stone         // Cursor hiding isn't working on MacOSX, so hide it in the top left
1740b9c1b51eSKate Stone         // corner
174144d93782SGreg Clayton         m_window_sp->MoveCursor(0, 0);
174244d93782SGreg Clayton 
174344d93782SGreg Clayton         doupdate();
174444d93782SGreg Clayton         update = false;
174544d93782SGreg Clayton       }
174644d93782SGreg Clayton 
174744d93782SGreg Clayton #if defined(__APPLE__)
174805097246SAdrian Prantl       // Terminal.app doesn't map its function keys correctly, F1-F4 default
174905097246SAdrian Prantl       // to: \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if
1750b9c1b51eSKate Stone       // possible
175144d93782SGreg Clayton       int ch;
175244d93782SGreg Clayton       if (escape_chars.empty())
175344d93782SGreg Clayton         ch = m_window_sp->GetChar();
1754b9c1b51eSKate Stone       else {
175544d93782SGreg Clayton         ch = escape_chars.front();
175644d93782SGreg Clayton         escape_chars.pop_front();
175744d93782SGreg Clayton       }
1758b9c1b51eSKate Stone       if (ch == KEY_ESCAPE) {
175944d93782SGreg Clayton         int ch2 = m_window_sp->GetChar();
1760b9c1b51eSKate Stone         if (ch2 == 'O') {
176144d93782SGreg Clayton           int ch3 = m_window_sp->GetChar();
1762b9c1b51eSKate Stone           switch (ch3) {
1763b9c1b51eSKate Stone           case 'P':
1764b9c1b51eSKate Stone             ch = KEY_F(1);
1765b9c1b51eSKate Stone             break;
1766b9c1b51eSKate Stone           case 'Q':
1767b9c1b51eSKate Stone             ch = KEY_F(2);
1768b9c1b51eSKate Stone             break;
1769b9c1b51eSKate Stone           case 'R':
1770b9c1b51eSKate Stone             ch = KEY_F(3);
1771b9c1b51eSKate Stone             break;
1772b9c1b51eSKate Stone           case 'S':
1773b9c1b51eSKate Stone             ch = KEY_F(4);
1774b9c1b51eSKate Stone             break;
177544d93782SGreg Clayton           default:
177644d93782SGreg Clayton             escape_chars.push_back(ch2);
177744d93782SGreg Clayton             if (ch3 != -1)
177844d93782SGreg Clayton               escape_chars.push_back(ch3);
177944d93782SGreg Clayton             break;
178044d93782SGreg Clayton           }
1781b9c1b51eSKate Stone         } else if (ch2 != -1)
178244d93782SGreg Clayton           escape_chars.push_back(ch2);
178344d93782SGreg Clayton       }
178444d93782SGreg Clayton #else
178544d93782SGreg Clayton       int ch = m_window_sp->GetChar();
178644d93782SGreg Clayton 
178744d93782SGreg Clayton #endif
1788b9c1b51eSKate Stone       if (ch == -1) {
1789b9c1b51eSKate Stone         if (feof(m_in) || ferror(m_in)) {
179044d93782SGreg Clayton           done = true;
1791b9c1b51eSKate Stone         } else {
179244d93782SGreg Clayton           // Just a timeout from using halfdelay(), check for events
179344d93782SGreg Clayton           EventSP event_sp;
1794b9c1b51eSKate Stone           while (listener_sp->PeekAtNextEvent()) {
1795d35031e1SPavel Labath             listener_sp->GetEvent(event_sp, std::chrono::seconds(0));
179644d93782SGreg Clayton 
1797b9c1b51eSKate Stone             if (event_sp) {
179844d93782SGreg Clayton               Broadcaster *broadcaster = event_sp->GetBroadcaster();
1799b9c1b51eSKate Stone               if (broadcaster) {
180044d93782SGreg Clayton                 // uint32_t event_type = event_sp->GetType();
1801b9c1b51eSKate Stone                 ConstString broadcaster_class(
1802b9c1b51eSKate Stone                     broadcaster->GetBroadcasterClass());
1803b9c1b51eSKate Stone                 if (broadcaster_class == broadcaster_class_process) {
1804b9c1b51eSKate Stone                   debugger.GetCommandInterpreter().UpdateExecutionContext(
1805b9c1b51eSKate Stone                       nullptr);
180644d93782SGreg Clayton                   update = true;
180744d93782SGreg Clayton                   continue; // Don't get any key, just update our view
180844d93782SGreg Clayton                 }
180944d93782SGreg Clayton               }
181044d93782SGreg Clayton             }
181144d93782SGreg Clayton           }
181244d93782SGreg Clayton         }
1813b9c1b51eSKate Stone       } else {
181444d93782SGreg Clayton         HandleCharResult key_result = m_window_sp->HandleChar(ch);
1815b9c1b51eSKate Stone         switch (key_result) {
181644d93782SGreg Clayton         case eKeyHandled:
1817c5dac77aSEugene Zelenko           debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
181844d93782SGreg Clayton           update = true;
181944d93782SGreg Clayton           break;
182044d93782SGreg Clayton         case eKeyNotHandled:
182144d93782SGreg Clayton           break;
182244d93782SGreg Clayton         case eQuitApplication:
182344d93782SGreg Clayton           done = true;
182444d93782SGreg Clayton           break;
182544d93782SGreg Clayton         }
182644d93782SGreg Clayton       }
182744d93782SGreg Clayton     }
182844d93782SGreg Clayton 
182944d93782SGreg Clayton     debugger.CancelForwardEvents(listener_sp);
183044d93782SGreg Clayton   }
183144d93782SGreg Clayton 
1832b9c1b51eSKate Stone   WindowSP &GetMainWindow() {
183344d93782SGreg Clayton     if (!m_window_sp)
1834796ac80bSJonas Devlieghere       m_window_sp = std::make_shared<Window>("main", stdscr, false);
183544d93782SGreg Clayton     return m_window_sp;
183644d93782SGreg Clayton   }
183744d93782SGreg Clayton 
183844d93782SGreg Clayton protected:
183944d93782SGreg Clayton   WindowSP m_window_sp;
184044d93782SGreg Clayton   WindowDelegates m_window_delegates;
184144d93782SGreg Clayton   SCREEN *m_screen;
184244d93782SGreg Clayton   FILE *m_in;
184344d93782SGreg Clayton   FILE *m_out;
184444d93782SGreg Clayton };
184544d93782SGreg Clayton 
184644d93782SGreg Clayton } // namespace curses
184744d93782SGreg Clayton 
184844d93782SGreg Clayton using namespace curses;
184944d93782SGreg Clayton 
1850b9c1b51eSKate Stone struct Row {
18518369b28dSGreg Clayton   ValueObjectManager value;
185244d93782SGreg Clayton   Row *parent;
18538369b28dSGreg Clayton   // The process stop ID when the children were calculated.
18548369b28dSGreg Clayton   uint32_t children_stop_id;
185544d93782SGreg Clayton   int row_idx;
185644d93782SGreg Clayton   int x;
185744d93782SGreg Clayton   int y;
185844d93782SGreg Clayton   bool might_have_children;
185944d93782SGreg Clayton   bool expanded;
186044d93782SGreg Clayton   bool calculated_children;
186144d93782SGreg Clayton   std::vector<Row> children;
186244d93782SGreg Clayton 
1863b9c1b51eSKate Stone   Row(const ValueObjectSP &v, Row *p)
18648369b28dSGreg Clayton       : value(v, lldb::eDynamicDontRunTarget, true), parent(p), row_idx(0),
18658369b28dSGreg Clayton         x(1), y(1), might_have_children(v ? v->MightHaveChildren() : false),
1866b9c1b51eSKate Stone         expanded(false), calculated_children(false), children() {}
186744d93782SGreg Clayton 
1868b9c1b51eSKate Stone   size_t GetDepth() const {
186944d93782SGreg Clayton     if (parent)
187044d93782SGreg Clayton       return 1 + parent->GetDepth();
187144d93782SGreg Clayton     return 0;
187244d93782SGreg Clayton   }
187344d93782SGreg Clayton 
1874171dd2e6SJonas Devlieghere   void Expand() { expanded = true; }
18758369b28dSGreg Clayton 
18768369b28dSGreg Clayton   std::vector<Row> &GetChildren() {
18778369b28dSGreg Clayton     ProcessSP process_sp = value.GetProcessSP();
18788369b28dSGreg Clayton     auto stop_id = process_sp->GetStopID();
18798369b28dSGreg Clayton     if (process_sp && stop_id != children_stop_id) {
18808369b28dSGreg Clayton       children_stop_id = stop_id;
18818369b28dSGreg Clayton       calculated_children = false;
18828369b28dSGreg Clayton     }
1883b9c1b51eSKate Stone     if (!calculated_children) {
18848369b28dSGreg Clayton       children.clear();
188544d93782SGreg Clayton       calculated_children = true;
18868369b28dSGreg Clayton       ValueObjectSP valobj = value.GetSP();
1887b9c1b51eSKate Stone       if (valobj) {
188844d93782SGreg Clayton         const size_t num_children = valobj->GetNumChildren();
1889b9c1b51eSKate Stone         for (size_t i = 0; i < num_children; ++i) {
189044d93782SGreg Clayton           children.push_back(Row(valobj->GetChildAtIndex(i, true), this));
189144d93782SGreg Clayton         }
189244d93782SGreg Clayton       }
189344d93782SGreg Clayton     }
18948369b28dSGreg Clayton     return children;
189544d93782SGreg Clayton   }
189644d93782SGreg Clayton 
18978369b28dSGreg Clayton   void Unexpand() {
18988369b28dSGreg Clayton     expanded = false;
18998369b28dSGreg Clayton     calculated_children = false;
19008369b28dSGreg Clayton     children.clear();
19018369b28dSGreg Clayton   }
190244d93782SGreg Clayton 
1903b9c1b51eSKate Stone   void DrawTree(Window &window) {
190444d93782SGreg Clayton     if (parent)
190544d93782SGreg Clayton       parent->DrawTreeForChild(window, this, 0);
190644d93782SGreg Clayton 
1907b9c1b51eSKate Stone     if (might_have_children) {
190805097246SAdrian Prantl       // It we can get UTF8 characters to work we should try to use the
190905097246SAdrian Prantl       // "symbol" UTF8 string below
191044d93782SGreg Clayton       //            const char *symbol = "";
191144d93782SGreg Clayton       //            if (row.expanded)
191244d93782SGreg Clayton       //                symbol = "\xe2\x96\xbd ";
191344d93782SGreg Clayton       //            else
191444d93782SGreg Clayton       //                symbol = "\xe2\x96\xb7 ";
191544d93782SGreg Clayton       //            window.PutCString (symbol);
191644d93782SGreg Clayton 
191705097246SAdrian Prantl       // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 'v'
191805097246SAdrian Prantl       // or '>' character...
191944d93782SGreg Clayton       //            if (expanded)
192044d93782SGreg Clayton       //                window.PutChar (ACS_DARROW);
192144d93782SGreg Clayton       //            else
192244d93782SGreg Clayton       //                window.PutChar (ACS_RARROW);
192305097246SAdrian Prantl       // Since we can't find any good looking right arrow/down arrow symbols,
192405097246SAdrian Prantl       // just use a diamond...
192544d93782SGreg Clayton       window.PutChar(ACS_DIAMOND);
192644d93782SGreg Clayton       window.PutChar(ACS_HLINE);
192744d93782SGreg Clayton     }
192844d93782SGreg Clayton   }
192944d93782SGreg Clayton 
1930b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, Row *child, uint32_t reverse_depth) {
193144d93782SGreg Clayton     if (parent)
193244d93782SGreg Clayton       parent->DrawTreeForChild(window, this, reverse_depth + 1);
193344d93782SGreg Clayton 
19348369b28dSGreg Clayton     if (&GetChildren().back() == child) {
193544d93782SGreg Clayton       // Last child
1936b9c1b51eSKate Stone       if (reverse_depth == 0) {
193744d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
193844d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1939b9c1b51eSKate Stone       } else {
194044d93782SGreg Clayton         window.PutChar(' ');
194144d93782SGreg Clayton         window.PutChar(' ');
194244d93782SGreg Clayton       }
1943b9c1b51eSKate Stone     } else {
1944b9c1b51eSKate Stone       if (reverse_depth == 0) {
194544d93782SGreg Clayton         window.PutChar(ACS_LTEE);
194644d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1947b9c1b51eSKate Stone       } else {
194844d93782SGreg Clayton         window.PutChar(ACS_VLINE);
194944d93782SGreg Clayton         window.PutChar(' ');
195044d93782SGreg Clayton       }
195144d93782SGreg Clayton     }
195244d93782SGreg Clayton   }
195344d93782SGreg Clayton };
195444d93782SGreg Clayton 
1955b9c1b51eSKate Stone struct DisplayOptions {
195644d93782SGreg Clayton   bool show_types;
195744d93782SGreg Clayton };
195844d93782SGreg Clayton 
195944d93782SGreg Clayton class TreeItem;
196044d93782SGreg Clayton 
1961b9c1b51eSKate Stone class TreeDelegate {
196244d93782SGreg Clayton public:
1963c5dac77aSEugene Zelenko   TreeDelegate() = default;
1964315b6884SEugene Zelenko   virtual ~TreeDelegate() = default;
1965315b6884SEugene Zelenko 
196644d93782SGreg Clayton   virtual void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) = 0;
196744d93782SGreg Clayton   virtual void TreeDelegateGenerateChildren(TreeItem &item) = 0;
1968b9c1b51eSKate Stone   virtual bool TreeDelegateItemSelected(
1969b9c1b51eSKate Stone       TreeItem &item) = 0; // Return true if we need to update views
197044d93782SGreg Clayton };
1971315b6884SEugene Zelenko 
197244d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
197344d93782SGreg Clayton 
1974b9c1b51eSKate Stone class TreeItem {
197544d93782SGreg Clayton public:
1976b9c1b51eSKate Stone   TreeItem(TreeItem *parent, TreeDelegate &delegate, bool might_have_children)
1977b9c1b51eSKate Stone       : m_parent(parent), m_delegate(delegate), m_user_data(nullptr),
1978b9c1b51eSKate Stone         m_identifier(0), m_row_idx(-1), m_children(),
1979b9c1b51eSKate Stone         m_might_have_children(might_have_children), m_is_expanded(false) {}
198044d93782SGreg Clayton 
1981b9c1b51eSKate Stone   TreeItem &operator=(const TreeItem &rhs) {
1982b9c1b51eSKate Stone     if (this != &rhs) {
198344d93782SGreg Clayton       m_parent = rhs.m_parent;
198444d93782SGreg Clayton       m_delegate = rhs.m_delegate;
1985ec990867SGreg Clayton       m_user_data = rhs.m_user_data;
198644d93782SGreg Clayton       m_identifier = rhs.m_identifier;
198744d93782SGreg Clayton       m_row_idx = rhs.m_row_idx;
198844d93782SGreg Clayton       m_children = rhs.m_children;
198944d93782SGreg Clayton       m_might_have_children = rhs.m_might_have_children;
199044d93782SGreg Clayton       m_is_expanded = rhs.m_is_expanded;
199144d93782SGreg Clayton     }
199244d93782SGreg Clayton     return *this;
199344d93782SGreg Clayton   }
199444d93782SGreg Clayton 
1995b9c1b51eSKate Stone   size_t GetDepth() const {
199644d93782SGreg Clayton     if (m_parent)
199744d93782SGreg Clayton       return 1 + m_parent->GetDepth();
199844d93782SGreg Clayton     return 0;
199944d93782SGreg Clayton   }
200044d93782SGreg Clayton 
2001b9c1b51eSKate Stone   int GetRowIndex() const { return m_row_idx; }
200244d93782SGreg Clayton 
2003b9c1b51eSKate Stone   void ClearChildren() { m_children.clear(); }
200444d93782SGreg Clayton 
2005b9c1b51eSKate Stone   void Resize(size_t n, const TreeItem &t) { m_children.resize(n, t); }
200644d93782SGreg Clayton 
2007b9c1b51eSKate Stone   TreeItem &operator[](size_t i) { return m_children[i]; }
200844d93782SGreg Clayton 
2009b9c1b51eSKate Stone   void SetRowIndex(int row_idx) { m_row_idx = row_idx; }
201044d93782SGreg Clayton 
2011b9c1b51eSKate Stone   size_t GetNumChildren() {
201244d93782SGreg Clayton     m_delegate.TreeDelegateGenerateChildren(*this);
201344d93782SGreg Clayton     return m_children.size();
201444d93782SGreg Clayton   }
201544d93782SGreg Clayton 
2016b9c1b51eSKate Stone   void ItemWasSelected() { m_delegate.TreeDelegateItemSelected(*this); }
2017315b6884SEugene Zelenko 
2018b9c1b51eSKate Stone   void CalculateRowIndexes(int &row_idx) {
201944d93782SGreg Clayton     SetRowIndex(row_idx);
202044d93782SGreg Clayton     ++row_idx;
202144d93782SGreg Clayton 
2022ec990867SGreg Clayton     const bool expanded = IsExpanded();
2023ec990867SGreg Clayton 
202405097246SAdrian Prantl     // The root item must calculate its children, or we must calculate the
202505097246SAdrian Prantl     // number of children if the item is expanded
2026c5dac77aSEugene Zelenko     if (m_parent == nullptr || expanded)
202744d93782SGreg Clayton       GetNumChildren();
202844d93782SGreg Clayton 
2029b9c1b51eSKate Stone     for (auto &item : m_children) {
203044d93782SGreg Clayton       if (expanded)
203144d93782SGreg Clayton         item.CalculateRowIndexes(row_idx);
203244d93782SGreg Clayton       else
203344d93782SGreg Clayton         item.SetRowIndex(-1);
203444d93782SGreg Clayton     }
203544d93782SGreg Clayton   }
203644d93782SGreg Clayton 
2037b9c1b51eSKate Stone   TreeItem *GetParent() { return m_parent; }
203844d93782SGreg Clayton 
2039b9c1b51eSKate Stone   bool IsExpanded() const { return m_is_expanded; }
204044d93782SGreg Clayton 
2041b9c1b51eSKate Stone   void Expand() { m_is_expanded = true; }
204244d93782SGreg Clayton 
2043b9c1b51eSKate Stone   void Unexpand() { m_is_expanded = false; }
204444d93782SGreg Clayton 
2045b9c1b51eSKate Stone   bool Draw(Window &window, const int first_visible_row,
2046b9c1b51eSKate Stone             const uint32_t selected_row_idx, int &row_idx, int &num_rows_left) {
204744d93782SGreg Clayton     if (num_rows_left <= 0)
204844d93782SGreg Clayton       return false;
204944d93782SGreg Clayton 
2050b9c1b51eSKate Stone     if (m_row_idx >= first_visible_row) {
205144d93782SGreg Clayton       window.MoveCursor(2, row_idx + 1);
205244d93782SGreg Clayton 
205344d93782SGreg Clayton       if (m_parent)
205444d93782SGreg Clayton         m_parent->DrawTreeForChild(window, this, 0);
205544d93782SGreg Clayton 
2056b9c1b51eSKate Stone       if (m_might_have_children) {
2057b9c1b51eSKate Stone         // It we can get UTF8 characters to work we should try to use the
205805097246SAdrian Prantl         // "symbol" UTF8 string below
205944d93782SGreg Clayton         //            const char *symbol = "";
206044d93782SGreg Clayton         //            if (row.expanded)
206144d93782SGreg Clayton         //                symbol = "\xe2\x96\xbd ";
206244d93782SGreg Clayton         //            else
206344d93782SGreg Clayton         //                symbol = "\xe2\x96\xb7 ";
206444d93782SGreg Clayton         //            window.PutCString (symbol);
206544d93782SGreg Clayton 
206644d93782SGreg Clayton         // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
206744d93782SGreg Clayton         // 'v' or '>' character...
206844d93782SGreg Clayton         //            if (expanded)
206944d93782SGreg Clayton         //                window.PutChar (ACS_DARROW);
207044d93782SGreg Clayton         //            else
207144d93782SGreg Clayton         //                window.PutChar (ACS_RARROW);
207205097246SAdrian Prantl         // Since we can't find any good looking right arrow/down arrow symbols,
207305097246SAdrian Prantl         // just use a diamond...
207444d93782SGreg Clayton         window.PutChar(ACS_DIAMOND);
207544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
207644d93782SGreg Clayton       }
2077b9c1b51eSKate Stone       bool highlight = (selected_row_idx == static_cast<size_t>(m_row_idx)) &&
2078b9c1b51eSKate Stone                        window.IsActive();
207944d93782SGreg Clayton 
208044d93782SGreg Clayton       if (highlight)
208144d93782SGreg Clayton         window.AttributeOn(A_REVERSE);
208244d93782SGreg Clayton 
208344d93782SGreg Clayton       m_delegate.TreeDelegateDrawTreeItem(*this, window);
208444d93782SGreg Clayton 
208544d93782SGreg Clayton       if (highlight)
208644d93782SGreg Clayton         window.AttributeOff(A_REVERSE);
208744d93782SGreg Clayton       ++row_idx;
208844d93782SGreg Clayton       --num_rows_left;
208944d93782SGreg Clayton     }
209044d93782SGreg Clayton 
209144d93782SGreg Clayton     if (num_rows_left <= 0)
209244d93782SGreg Clayton       return false; // We are done drawing...
209344d93782SGreg Clayton 
2094b9c1b51eSKate Stone     if (IsExpanded()) {
2095b9c1b51eSKate Stone       for (auto &item : m_children) {
209605097246SAdrian Prantl         // If we displayed all the rows and item.Draw() returns false we are
209705097246SAdrian Prantl         // done drawing and can exit this for loop
2098b9c1b51eSKate Stone         if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx,
2099b9c1b51eSKate Stone                        num_rows_left))
210044d93782SGreg Clayton           break;
210144d93782SGreg Clayton       }
210244d93782SGreg Clayton     }
210344d93782SGreg Clayton     return num_rows_left >= 0; // Return true if not done drawing yet
210444d93782SGreg Clayton   }
210544d93782SGreg Clayton 
2106b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, TreeItem *child,
2107b9c1b51eSKate Stone                         uint32_t reverse_depth) {
210844d93782SGreg Clayton     if (m_parent)
210944d93782SGreg Clayton       m_parent->DrawTreeForChild(window, this, reverse_depth + 1);
211044d93782SGreg Clayton 
2111b9c1b51eSKate Stone     if (&m_children.back() == child) {
211244d93782SGreg Clayton       // Last child
2113b9c1b51eSKate Stone       if (reverse_depth == 0) {
211444d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
211544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2116b9c1b51eSKate Stone       } else {
211744d93782SGreg Clayton         window.PutChar(' ');
211844d93782SGreg Clayton         window.PutChar(' ');
211944d93782SGreg Clayton       }
2120b9c1b51eSKate Stone     } else {
2121b9c1b51eSKate Stone       if (reverse_depth == 0) {
212244d93782SGreg Clayton         window.PutChar(ACS_LTEE);
212344d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2124b9c1b51eSKate Stone       } else {
212544d93782SGreg Clayton         window.PutChar(ACS_VLINE);
212644d93782SGreg Clayton         window.PutChar(' ');
212744d93782SGreg Clayton       }
212844d93782SGreg Clayton     }
212944d93782SGreg Clayton   }
213044d93782SGreg Clayton 
2131b9c1b51eSKate Stone   TreeItem *GetItemForRowIndex(uint32_t row_idx) {
21323985c8c6SSaleem Abdulrasool     if (static_cast<uint32_t>(m_row_idx) == row_idx)
213344d93782SGreg Clayton       return this;
213444d93782SGreg Clayton     if (m_children.empty())
2135c5dac77aSEugene Zelenko       return nullptr;
2136b9c1b51eSKate Stone     if (IsExpanded()) {
2137b9c1b51eSKate Stone       for (auto &item : m_children) {
213844d93782SGreg Clayton         TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
213944d93782SGreg Clayton         if (selected_item_ptr)
214044d93782SGreg Clayton           return selected_item_ptr;
214144d93782SGreg Clayton       }
214244d93782SGreg Clayton     }
2143c5dac77aSEugene Zelenko     return nullptr;
214444d93782SGreg Clayton   }
214544d93782SGreg Clayton 
2146b9c1b51eSKate Stone   void *GetUserData() const { return m_user_data; }
2147ec990867SGreg Clayton 
2148b9c1b51eSKate Stone   void SetUserData(void *user_data) { m_user_data = user_data; }
2149ec990867SGreg Clayton 
2150b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
215144d93782SGreg Clayton 
2152b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
215344d93782SGreg Clayton 
2154b9c1b51eSKate Stone   void SetMightHaveChildren(bool b) { m_might_have_children = b; }
2155ec990867SGreg Clayton 
215644d93782SGreg Clayton protected:
215744d93782SGreg Clayton   TreeItem *m_parent;
215844d93782SGreg Clayton   TreeDelegate &m_delegate;
2159ec990867SGreg Clayton   void *m_user_data;
216044d93782SGreg Clayton   uint64_t m_identifier;
2161b9c1b51eSKate Stone   int m_row_idx; // Zero based visible row index, -1 if not visible or for the
2162b9c1b51eSKate Stone                  // root item
216344d93782SGreg Clayton   std::vector<TreeItem> m_children;
216444d93782SGreg Clayton   bool m_might_have_children;
216544d93782SGreg Clayton   bool m_is_expanded;
216644d93782SGreg Clayton };
216744d93782SGreg Clayton 
2168b9c1b51eSKate Stone class TreeWindowDelegate : public WindowDelegate {
216944d93782SGreg Clayton public:
2170b9c1b51eSKate Stone   TreeWindowDelegate(Debugger &debugger, const TreeDelegateSP &delegate_sp)
2171b9c1b51eSKate Stone       : m_debugger(debugger), m_delegate_sp(delegate_sp),
2172b9c1b51eSKate Stone         m_root(nullptr, *delegate_sp, true), m_selected_item(nullptr),
2173b9c1b51eSKate Stone         m_num_rows(0), m_selected_row_idx(0), m_first_visible_row(0),
2174b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
217544d93782SGreg Clayton 
2176b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
217744d93782SGreg Clayton 
2178b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2179b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2180b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
218144d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
218244d93782SGreg Clayton 
218344d93782SGreg Clayton     bool display_content = false;
2184b9c1b51eSKate Stone     if (process) {
218544d93782SGreg Clayton       StateType state = process->GetState();
2186b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
218744d93782SGreg Clayton         // We are stopped, so it is ok to
218844d93782SGreg Clayton         display_content = true;
2189b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
219044d93782SGreg Clayton         return true; // Don't do any updating when we are running
219144d93782SGreg Clayton       }
219244d93782SGreg Clayton     }
219344d93782SGreg Clayton 
219444d93782SGreg Clayton     m_min_x = 2;
219544d93782SGreg Clayton     m_min_y = 1;
219644d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
219744d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
219844d93782SGreg Clayton 
219944d93782SGreg Clayton     window.Erase();
220044d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
220144d93782SGreg Clayton 
2202b9c1b51eSKate Stone     if (display_content) {
220344d93782SGreg Clayton       const int num_visible_rows = NumVisibleRows();
220444d93782SGreg Clayton       m_num_rows = 0;
220544d93782SGreg Clayton       m_root.CalculateRowIndexes(m_num_rows);
220644d93782SGreg Clayton 
220705097246SAdrian Prantl       // If we unexpanded while having something selected our total number of
220805097246SAdrian Prantl       // rows is less than the num visible rows, then make sure we show all the
220905097246SAdrian Prantl       // rows by setting the first visible row accordingly.
221044d93782SGreg Clayton       if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
221144d93782SGreg Clayton         m_first_visible_row = 0;
221244d93782SGreg Clayton 
221344d93782SGreg Clayton       // Make sure the selected row is always visible
221444d93782SGreg Clayton       if (m_selected_row_idx < m_first_visible_row)
221544d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx;
221644d93782SGreg Clayton       else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
221744d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
221844d93782SGreg Clayton 
221944d93782SGreg Clayton       int row_idx = 0;
222044d93782SGreg Clayton       int num_rows_left = num_visible_rows;
2221b9c1b51eSKate Stone       m_root.Draw(window, m_first_visible_row, m_selected_row_idx, row_idx,
2222b9c1b51eSKate Stone                   num_rows_left);
222344d93782SGreg Clayton       // Get the selected row
222444d93782SGreg Clayton       m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
2225b9c1b51eSKate Stone     } else {
2226c5dac77aSEugene Zelenko       m_selected_item = nullptr;
222744d93782SGreg Clayton     }
222844d93782SGreg Clayton 
222944d93782SGreg Clayton     return true; // Drawing handled
223044d93782SGreg Clayton   }
223144d93782SGreg Clayton 
2232b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
223344d93782SGreg Clayton     return "Thread window keyboard shortcuts:";
223444d93782SGreg Clayton   }
223544d93782SGreg Clayton 
2236b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
223744d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
223844d93782SGreg Clayton         {KEY_UP, "Select previous item"},
223944d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
224044d93782SGreg Clayton         {KEY_RIGHT, "Expand the selected item"},
2241b9c1b51eSKate Stone         {KEY_LEFT,
2242b9c1b51eSKate Stone          "Unexpand the selected item or select parent if not expanded"},
224344d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
224444d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
224544d93782SGreg Clayton         {'h', "Show help dialog"},
224644d93782SGreg Clayton         {' ', "Toggle item expansion"},
224744d93782SGreg Clayton         {',', "Page up"},
224844d93782SGreg Clayton         {'.', "Page down"},
2249b9c1b51eSKate Stone         {'\0', nullptr}};
225044d93782SGreg Clayton     return g_source_view_key_help;
225144d93782SGreg Clayton   }
225244d93782SGreg Clayton 
2253b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2254b9c1b51eSKate Stone     switch (c) {
225544d93782SGreg Clayton     case ',':
225644d93782SGreg Clayton     case KEY_PPAGE:
225744d93782SGreg Clayton       // Page up key
2258b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
225944d93782SGreg Clayton         if (m_first_visible_row > m_max_y)
226044d93782SGreg Clayton           m_first_visible_row -= m_max_y;
226144d93782SGreg Clayton         else
226244d93782SGreg Clayton           m_first_visible_row = 0;
226344d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
226444d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
226544d93782SGreg Clayton         if (m_selected_item)
226644d93782SGreg Clayton           m_selected_item->ItemWasSelected();
226744d93782SGreg Clayton       }
226844d93782SGreg Clayton       return eKeyHandled;
226944d93782SGreg Clayton 
227044d93782SGreg Clayton     case '.':
227144d93782SGreg Clayton     case KEY_NPAGE:
227244d93782SGreg Clayton       // Page down key
2273b9c1b51eSKate Stone       if (m_num_rows > m_max_y) {
2274b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
227544d93782SGreg Clayton           m_first_visible_row += m_max_y;
227644d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
227744d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
227844d93782SGreg Clayton           if (m_selected_item)
227944d93782SGreg Clayton             m_selected_item->ItemWasSelected();
228044d93782SGreg Clayton         }
228144d93782SGreg Clayton       }
228244d93782SGreg Clayton       return eKeyHandled;
228344d93782SGreg Clayton 
228444d93782SGreg Clayton     case KEY_UP:
2285b9c1b51eSKate Stone       if (m_selected_row_idx > 0) {
228644d93782SGreg Clayton         --m_selected_row_idx;
228744d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
228844d93782SGreg Clayton         if (m_selected_item)
228944d93782SGreg Clayton           m_selected_item->ItemWasSelected();
229044d93782SGreg Clayton       }
229144d93782SGreg Clayton       return eKeyHandled;
2292315b6884SEugene Zelenko 
229344d93782SGreg Clayton     case KEY_DOWN:
2294b9c1b51eSKate Stone       if (m_selected_row_idx + 1 < m_num_rows) {
229544d93782SGreg Clayton         ++m_selected_row_idx;
229644d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
229744d93782SGreg Clayton         if (m_selected_item)
229844d93782SGreg Clayton           m_selected_item->ItemWasSelected();
229944d93782SGreg Clayton       }
230044d93782SGreg Clayton       return eKeyHandled;
230144d93782SGreg Clayton 
230244d93782SGreg Clayton     case KEY_RIGHT:
2303b9c1b51eSKate Stone       if (m_selected_item) {
230444d93782SGreg Clayton         if (!m_selected_item->IsExpanded())
230544d93782SGreg Clayton           m_selected_item->Expand();
230644d93782SGreg Clayton       }
230744d93782SGreg Clayton       return eKeyHandled;
230844d93782SGreg Clayton 
230944d93782SGreg Clayton     case KEY_LEFT:
2310b9c1b51eSKate Stone       if (m_selected_item) {
231144d93782SGreg Clayton         if (m_selected_item->IsExpanded())
231244d93782SGreg Clayton           m_selected_item->Unexpand();
2313b9c1b51eSKate Stone         else if (m_selected_item->GetParent()) {
231444d93782SGreg Clayton           m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
231544d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
231644d93782SGreg Clayton           if (m_selected_item)
231744d93782SGreg Clayton             m_selected_item->ItemWasSelected();
231844d93782SGreg Clayton         }
231944d93782SGreg Clayton       }
232044d93782SGreg Clayton       return eKeyHandled;
232144d93782SGreg Clayton 
232244d93782SGreg Clayton     case ' ':
232344d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2324b9c1b51eSKate Stone       if (m_selected_item) {
232544d93782SGreg Clayton         if (m_selected_item->IsExpanded())
232644d93782SGreg Clayton           m_selected_item->Unexpand();
232744d93782SGreg Clayton         else
232844d93782SGreg Clayton           m_selected_item->Expand();
232944d93782SGreg Clayton       }
233044d93782SGreg Clayton       return eKeyHandled;
233144d93782SGreg Clayton 
233244d93782SGreg Clayton     case 'h':
233344d93782SGreg Clayton       window.CreateHelpSubwindow();
233444d93782SGreg Clayton       return eKeyHandled;
233544d93782SGreg Clayton 
233644d93782SGreg Clayton     default:
233744d93782SGreg Clayton       break;
233844d93782SGreg Clayton     }
233944d93782SGreg Clayton     return eKeyNotHandled;
234044d93782SGreg Clayton   }
234144d93782SGreg Clayton 
234244d93782SGreg Clayton protected:
234344d93782SGreg Clayton   Debugger &m_debugger;
234444d93782SGreg Clayton   TreeDelegateSP m_delegate_sp;
234544d93782SGreg Clayton   TreeItem m_root;
234644d93782SGreg Clayton   TreeItem *m_selected_item;
234744d93782SGreg Clayton   int m_num_rows;
234844d93782SGreg Clayton   int m_selected_row_idx;
234944d93782SGreg Clayton   int m_first_visible_row;
235044d93782SGreg Clayton   int m_min_x;
235144d93782SGreg Clayton   int m_min_y;
235244d93782SGreg Clayton   int m_max_x;
235344d93782SGreg Clayton   int m_max_y;
235444d93782SGreg Clayton };
235544d93782SGreg Clayton 
2356b9c1b51eSKate Stone class FrameTreeDelegate : public TreeDelegate {
235744d93782SGreg Clayton public:
2358b9c1b51eSKate Stone   FrameTreeDelegate() : TreeDelegate() {
2359b9c1b51eSKate Stone     FormatEntity::Parse(
2360b9c1b51eSKate Stone         "frame #${frame.index}: {${function.name}${function.pc-offset}}}",
2361554f68d3SGreg Clayton         m_format);
236244d93782SGreg Clayton   }
236344d93782SGreg Clayton 
2364315b6884SEugene Zelenko   ~FrameTreeDelegate() override = default;
236544d93782SGreg Clayton 
2366b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2367ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2368b9c1b51eSKate Stone     if (thread) {
236944d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2370ec990867SGreg Clayton       StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
2371b9c1b51eSKate Stone       if (frame_sp) {
237244d93782SGreg Clayton         StreamString strm;
2373b9c1b51eSKate Stone         const SymbolContext &sc =
2374b9c1b51eSKate Stone             frame_sp->GetSymbolContext(eSymbolContextEverything);
237544d93782SGreg Clayton         ExecutionContext exe_ctx(frame_sp);
2376b9c1b51eSKate Stone         if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr,
2377b9c1b51eSKate Stone                                  nullptr, false, false)) {
237844d93782SGreg Clayton           int right_pad = 1;
2379c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
238044d93782SGreg Clayton         }
238144d93782SGreg Clayton       }
238244d93782SGreg Clayton     }
238344d93782SGreg Clayton   }
2384315b6884SEugene Zelenko 
2385b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
238644d93782SGreg Clayton     // No children for frames yet...
238744d93782SGreg Clayton   }
238844d93782SGreg Clayton 
2389b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2390ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2391b9c1b51eSKate Stone     if (thread) {
2392b9c1b51eSKate Stone       thread->GetProcess()->GetThreadList().SetSelectedThreadByID(
2393b9c1b51eSKate Stone           thread->GetID());
239444d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2395ec990867SGreg Clayton       thread->SetSelectedFrameByIndex(frame_idx);
239644d93782SGreg Clayton       return true;
239744d93782SGreg Clayton     }
239844d93782SGreg Clayton     return false;
239944d93782SGreg Clayton   }
2400315b6884SEugene Zelenko 
2401554f68d3SGreg Clayton protected:
2402554f68d3SGreg Clayton   FormatEntity::Entry m_format;
240344d93782SGreg Clayton };
240444d93782SGreg Clayton 
2405b9c1b51eSKate Stone class ThreadTreeDelegate : public TreeDelegate {
240644d93782SGreg Clayton public:
2407b9c1b51eSKate Stone   ThreadTreeDelegate(Debugger &debugger)
2408b9c1b51eSKate Stone       : TreeDelegate(), m_debugger(debugger), m_tid(LLDB_INVALID_THREAD_ID),
2409b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2410b9c1b51eSKate Stone     FormatEntity::Parse("thread #${thread.index}: tid = ${thread.id}{, stop "
2411b9c1b51eSKate Stone                         "reason = ${thread.stop-reason}}",
2412554f68d3SGreg Clayton                         m_format);
241344d93782SGreg Clayton   }
241444d93782SGreg Clayton 
2415315b6884SEugene Zelenko   ~ThreadTreeDelegate() override = default;
241644d93782SGreg Clayton 
2417b9c1b51eSKate Stone   ProcessSP GetProcess() {
2418b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2419b9c1b51eSKate Stone         .GetExecutionContext()
2420b9c1b51eSKate Stone         .GetProcessSP();
2421ec990867SGreg Clayton   }
2422ec990867SGreg Clayton 
2423b9c1b51eSKate Stone   ThreadSP GetThread(const TreeItem &item) {
2424ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2425ec990867SGreg Clayton     if (process_sp)
2426ec990867SGreg Clayton       return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
2427ec990867SGreg Clayton     return ThreadSP();
2428ec990867SGreg Clayton   }
2429ec990867SGreg Clayton 
2430b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2431ec990867SGreg Clayton     ThreadSP thread_sp = GetThread(item);
2432b9c1b51eSKate Stone     if (thread_sp) {
243344d93782SGreg Clayton       StreamString strm;
243444d93782SGreg Clayton       ExecutionContext exe_ctx(thread_sp);
2435b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2436b9c1b51eSKate Stone                                nullptr, false, false)) {
243744d93782SGreg Clayton         int right_pad = 1;
2438c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
243944d93782SGreg Clayton       }
244044d93782SGreg Clayton     }
244144d93782SGreg Clayton   }
2442315b6884SEugene Zelenko 
2443b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2444ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2445b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
244644d93782SGreg Clayton       StateType state = process_sp->GetState();
2447b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2448ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2449b9c1b51eSKate Stone         if (thread_sp) {
2450b9c1b51eSKate Stone           if (m_stop_id == process_sp->GetStopID() &&
2451b9c1b51eSKate Stone               thread_sp->GetID() == m_tid)
245244d93782SGreg Clayton             return; // Children are already up to date
2453b9c1b51eSKate Stone           if (!m_frame_delegate_sp) {
245444d93782SGreg Clayton             // Always expand the thread item the first time we show it
2455796ac80bSJonas Devlieghere             m_frame_delegate_sp = std::make_shared<FrameTreeDelegate>();
245644d93782SGreg Clayton           }
245744d93782SGreg Clayton 
245844d93782SGreg Clayton           m_stop_id = process_sp->GetStopID();
245944d93782SGreg Clayton           m_tid = thread_sp->GetID();
246044d93782SGreg Clayton 
246144d93782SGreg Clayton           TreeItem t(&item, *m_frame_delegate_sp, false);
246244d93782SGreg Clayton           size_t num_frames = thread_sp->GetStackFrameCount();
246344d93782SGreg Clayton           item.Resize(num_frames, t);
2464b9c1b51eSKate Stone           for (size_t i = 0; i < num_frames; ++i) {
2465ec990867SGreg Clayton             item[i].SetUserData(thread_sp.get());
246644d93782SGreg Clayton             item[i].SetIdentifier(i);
246744d93782SGreg Clayton           }
246844d93782SGreg Clayton         }
246944d93782SGreg Clayton         return;
247044d93782SGreg Clayton       }
247144d93782SGreg Clayton     }
247244d93782SGreg Clayton     item.ClearChildren();
247344d93782SGreg Clayton   }
247444d93782SGreg Clayton 
2475b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2476ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2477b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2478ec990867SGreg Clayton       StateType state = process_sp->GetState();
2479b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2480ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2481b9c1b51eSKate Stone         if (thread_sp) {
248244d93782SGreg Clayton           ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
2483bb19a13cSSaleem Abdulrasool           std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
248444d93782SGreg Clayton           ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
2485b9c1b51eSKate Stone           if (selected_thread_sp->GetID() != thread_sp->GetID()) {
248644d93782SGreg Clayton             thread_list.SetSelectedThreadByID(thread_sp->GetID());
248744d93782SGreg Clayton             return true;
248844d93782SGreg Clayton           }
248944d93782SGreg Clayton         }
2490ec990867SGreg Clayton       }
2491ec990867SGreg Clayton     }
249244d93782SGreg Clayton     return false;
249344d93782SGreg Clayton   }
249444d93782SGreg Clayton 
249544d93782SGreg Clayton protected:
249644d93782SGreg Clayton   Debugger &m_debugger;
249744d93782SGreg Clayton   std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
249844d93782SGreg Clayton   lldb::user_id_t m_tid;
249944d93782SGreg Clayton   uint32_t m_stop_id;
2500554f68d3SGreg Clayton   FormatEntity::Entry m_format;
250144d93782SGreg Clayton };
250244d93782SGreg Clayton 
2503b9c1b51eSKate Stone class ThreadsTreeDelegate : public TreeDelegate {
2504ec990867SGreg Clayton public:
2505b9c1b51eSKate Stone   ThreadsTreeDelegate(Debugger &debugger)
2506b9c1b51eSKate Stone       : TreeDelegate(), m_thread_delegate_sp(), m_debugger(debugger),
2507b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2508554f68d3SGreg Clayton     FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
2509554f68d3SGreg Clayton                         m_format);
2510ec990867SGreg Clayton   }
2511ec990867SGreg Clayton 
2512315b6884SEugene Zelenko   ~ThreadsTreeDelegate() override = default;
2513ec990867SGreg Clayton 
2514b9c1b51eSKate Stone   ProcessSP GetProcess() {
2515b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2516b9c1b51eSKate Stone         .GetExecutionContext()
2517b9c1b51eSKate Stone         .GetProcessSP();
2518ec990867SGreg Clayton   }
2519ec990867SGreg Clayton 
2520b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2521ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2522b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2523ec990867SGreg Clayton       StreamString strm;
2524ec990867SGreg Clayton       ExecutionContext exe_ctx(process_sp);
2525b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2526b9c1b51eSKate Stone                                nullptr, false, false)) {
2527ec990867SGreg Clayton         int right_pad = 1;
2528c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
2529ec990867SGreg Clayton       }
2530ec990867SGreg Clayton     }
2531ec990867SGreg Clayton   }
2532ec990867SGreg Clayton 
2533b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2534ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2535b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2536ec990867SGreg Clayton       StateType state = process_sp->GetState();
2537b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2538ec990867SGreg Clayton         const uint32_t stop_id = process_sp->GetStopID();
2539ec990867SGreg Clayton         if (m_stop_id == stop_id)
2540ec990867SGreg Clayton           return; // Children are already up to date
2541ec990867SGreg Clayton 
2542ec990867SGreg Clayton         m_stop_id = stop_id;
2543ec990867SGreg Clayton 
2544b9c1b51eSKate Stone         if (!m_thread_delegate_sp) {
2545ec990867SGreg Clayton           // Always expand the thread item the first time we show it
2546ec990867SGreg Clayton           // item.Expand();
2547796ac80bSJonas Devlieghere           m_thread_delegate_sp =
2548796ac80bSJonas Devlieghere               std::make_shared<ThreadTreeDelegate>(m_debugger);
2549ec990867SGreg Clayton         }
2550ec990867SGreg Clayton 
2551ec990867SGreg Clayton         TreeItem t(&item, *m_thread_delegate_sp, false);
2552ec990867SGreg Clayton         ThreadList &threads = process_sp->GetThreadList();
2553bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
2554ec990867SGreg Clayton         size_t num_threads = threads.GetSize();
2555ec990867SGreg Clayton         item.Resize(num_threads, t);
2556b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
2557ec990867SGreg Clayton           item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
2558ec990867SGreg Clayton           item[i].SetMightHaveChildren(true);
2559ec990867SGreg Clayton         }
2560ec990867SGreg Clayton         return;
2561ec990867SGreg Clayton       }
2562ec990867SGreg Clayton     }
2563ec990867SGreg Clayton     item.ClearChildren();
2564ec990867SGreg Clayton   }
2565ec990867SGreg Clayton 
2566b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override { return false; }
2567ec990867SGreg Clayton 
2568ec990867SGreg Clayton protected:
2569ec990867SGreg Clayton   std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
2570ec990867SGreg Clayton   Debugger &m_debugger;
2571ec990867SGreg Clayton   uint32_t m_stop_id;
2572554f68d3SGreg Clayton   FormatEntity::Entry m_format;
2573ec990867SGreg Clayton };
2574ec990867SGreg Clayton 
2575b9c1b51eSKate Stone class ValueObjectListDelegate : public WindowDelegate {
257644d93782SGreg Clayton public:
2577b9c1b51eSKate Stone   ValueObjectListDelegate()
2578171dd2e6SJonas Devlieghere       : m_rows(), m_selected_row(nullptr), m_selected_row_idx(0),
2579171dd2e6SJonas Devlieghere         m_first_visible_row(0), m_num_rows(0), m_max_x(0), m_max_y(0) {}
258044d93782SGreg Clayton 
2581b9c1b51eSKate Stone   ValueObjectListDelegate(ValueObjectList &valobj_list)
2582171dd2e6SJonas Devlieghere       : m_rows(), m_selected_row(nullptr), m_selected_row_idx(0),
2583171dd2e6SJonas Devlieghere         m_first_visible_row(0), m_num_rows(0), m_max_x(0), m_max_y(0) {
258444d93782SGreg Clayton     SetValues(valobj_list);
258544d93782SGreg Clayton   }
258644d93782SGreg Clayton 
2587315b6884SEugene Zelenko   ~ValueObjectListDelegate() override = default;
258844d93782SGreg Clayton 
2589b9c1b51eSKate Stone   void SetValues(ValueObjectList &valobj_list) {
2590c5dac77aSEugene Zelenko     m_selected_row = nullptr;
259144d93782SGreg Clayton     m_selected_row_idx = 0;
259244d93782SGreg Clayton     m_first_visible_row = 0;
259344d93782SGreg Clayton     m_num_rows = 0;
259444d93782SGreg Clayton     m_rows.clear();
25958369b28dSGreg Clayton     for (auto &valobj_sp : valobj_list.GetObjects())
25968369b28dSGreg Clayton       m_rows.push_back(Row(valobj_sp, nullptr));
259744d93782SGreg Clayton   }
259844d93782SGreg Clayton 
2599b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
260044d93782SGreg Clayton     m_num_rows = 0;
260144d93782SGreg Clayton     m_min_x = 2;
260244d93782SGreg Clayton     m_min_y = 1;
260344d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
260444d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
260544d93782SGreg Clayton 
260644d93782SGreg Clayton     window.Erase();
260744d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
260844d93782SGreg Clayton 
260944d93782SGreg Clayton     const int num_visible_rows = NumVisibleRows();
261044d93782SGreg Clayton     const int num_rows = CalculateTotalNumberRows(m_rows);
261144d93782SGreg Clayton 
261205097246SAdrian Prantl     // If we unexpanded while having something selected our total number of
261305097246SAdrian Prantl     // rows is less than the num visible rows, then make sure we show all the
261405097246SAdrian Prantl     // rows by setting the first visible row accordingly.
261544d93782SGreg Clayton     if (m_first_visible_row > 0 && num_rows < num_visible_rows)
261644d93782SGreg Clayton       m_first_visible_row = 0;
261744d93782SGreg Clayton 
261844d93782SGreg Clayton     // Make sure the selected row is always visible
261944d93782SGreg Clayton     if (m_selected_row_idx < m_first_visible_row)
262044d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx;
262144d93782SGreg Clayton     else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
262244d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
262344d93782SGreg Clayton 
262444d93782SGreg Clayton     DisplayRows(window, m_rows, g_options);
262544d93782SGreg Clayton 
262644d93782SGreg Clayton     // Get the selected row
262744d93782SGreg Clayton     m_selected_row = GetRowForRowIndex(m_selected_row_idx);
262805097246SAdrian Prantl     // Keep the cursor on the selected row so the highlight and the cursor are
262905097246SAdrian Prantl     // always on the same line
263044d93782SGreg Clayton     if (m_selected_row)
2631b9c1b51eSKate Stone       window.MoveCursor(m_selected_row->x, m_selected_row->y);
263244d93782SGreg Clayton 
263344d93782SGreg Clayton     return true; // Drawing handled
263444d93782SGreg Clayton   }
263544d93782SGreg Clayton 
2636b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
263744d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
263844d93782SGreg Clayton         {KEY_UP, "Select previous item"},
263944d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
264044d93782SGreg Clayton         {KEY_RIGHT, "Expand selected item"},
264144d93782SGreg Clayton         {KEY_LEFT, "Unexpand selected item or select parent if not expanded"},
264244d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
264344d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
264444d93782SGreg Clayton         {'A', "Format as annotated address"},
264544d93782SGreg Clayton         {'b', "Format as binary"},
264644d93782SGreg Clayton         {'B', "Format as hex bytes with ASCII"},
264744d93782SGreg Clayton         {'c', "Format as character"},
264844d93782SGreg Clayton         {'d', "Format as a signed integer"},
264944d93782SGreg Clayton         {'D', "Format selected value using the default format for the type"},
265044d93782SGreg Clayton         {'f', "Format as float"},
265144d93782SGreg Clayton         {'h', "Show help dialog"},
265244d93782SGreg Clayton         {'i', "Format as instructions"},
265344d93782SGreg Clayton         {'o', "Format as octal"},
265444d93782SGreg Clayton         {'p', "Format as pointer"},
265544d93782SGreg Clayton         {'s', "Format as C string"},
265644d93782SGreg Clayton         {'t', "Toggle showing/hiding type names"},
265744d93782SGreg Clayton         {'u', "Format as an unsigned integer"},
265844d93782SGreg Clayton         {'x', "Format as hex"},
265944d93782SGreg Clayton         {'X', "Format as uppercase hex"},
266044d93782SGreg Clayton         {' ', "Toggle item expansion"},
266144d93782SGreg Clayton         {',', "Page up"},
266244d93782SGreg Clayton         {'.', "Page down"},
2663b9c1b51eSKate Stone         {'\0', nullptr}};
266444d93782SGreg Clayton     return g_source_view_key_help;
266544d93782SGreg Clayton   }
266644d93782SGreg Clayton 
2667b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2668b9c1b51eSKate Stone     switch (c) {
266944d93782SGreg Clayton     case 'x':
267044d93782SGreg Clayton     case 'X':
267144d93782SGreg Clayton     case 'o':
267244d93782SGreg Clayton     case 's':
267344d93782SGreg Clayton     case 'u':
267444d93782SGreg Clayton     case 'd':
267544d93782SGreg Clayton     case 'D':
267644d93782SGreg Clayton     case 'i':
267744d93782SGreg Clayton     case 'A':
267844d93782SGreg Clayton     case 'p':
267944d93782SGreg Clayton     case 'c':
268044d93782SGreg Clayton     case 'b':
268144d93782SGreg Clayton     case 'B':
268244d93782SGreg Clayton     case 'f':
268344d93782SGreg Clayton       // Change the format for the currently selected item
26848369b28dSGreg Clayton       if (m_selected_row) {
26858369b28dSGreg Clayton         auto valobj_sp = m_selected_row->value.GetSP();
26868369b28dSGreg Clayton         if (valobj_sp)
26878369b28dSGreg Clayton           valobj_sp->SetFormat(FormatForChar(c));
26888369b28dSGreg Clayton       }
268944d93782SGreg Clayton       return eKeyHandled;
269044d93782SGreg Clayton 
269144d93782SGreg Clayton     case 't':
269244d93782SGreg Clayton       // Toggle showing type names
269344d93782SGreg Clayton       g_options.show_types = !g_options.show_types;
269444d93782SGreg Clayton       return eKeyHandled;
269544d93782SGreg Clayton 
269644d93782SGreg Clayton     case ',':
269744d93782SGreg Clayton     case KEY_PPAGE:
269844d93782SGreg Clayton       // Page up key
2699b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
27003985c8c6SSaleem Abdulrasool         if (static_cast<int>(m_first_visible_row) > m_max_y)
270144d93782SGreg Clayton           m_first_visible_row -= m_max_y;
270244d93782SGreg Clayton         else
270344d93782SGreg Clayton           m_first_visible_row = 0;
270444d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
270544d93782SGreg Clayton       }
270644d93782SGreg Clayton       return eKeyHandled;
270744d93782SGreg Clayton 
270844d93782SGreg Clayton     case '.':
270944d93782SGreg Clayton     case KEY_NPAGE:
271044d93782SGreg Clayton       // Page down key
2711b9c1b51eSKate Stone       if (m_num_rows > static_cast<size_t>(m_max_y)) {
2712b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
271344d93782SGreg Clayton           m_first_visible_row += m_max_y;
271444d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
271544d93782SGreg Clayton         }
271644d93782SGreg Clayton       }
271744d93782SGreg Clayton       return eKeyHandled;
271844d93782SGreg Clayton 
271944d93782SGreg Clayton     case KEY_UP:
272044d93782SGreg Clayton       if (m_selected_row_idx > 0)
272144d93782SGreg Clayton         --m_selected_row_idx;
272244d93782SGreg Clayton       return eKeyHandled;
2723315b6884SEugene Zelenko 
272444d93782SGreg Clayton     case KEY_DOWN:
272544d93782SGreg Clayton       if (m_selected_row_idx + 1 < m_num_rows)
272644d93782SGreg Clayton         ++m_selected_row_idx;
272744d93782SGreg Clayton       return eKeyHandled;
272844d93782SGreg Clayton 
272944d93782SGreg Clayton     case KEY_RIGHT:
2730b9c1b51eSKate Stone       if (m_selected_row) {
273144d93782SGreg Clayton         if (!m_selected_row->expanded)
273244d93782SGreg Clayton           m_selected_row->Expand();
273344d93782SGreg Clayton       }
273444d93782SGreg Clayton       return eKeyHandled;
273544d93782SGreg Clayton 
273644d93782SGreg Clayton     case KEY_LEFT:
2737b9c1b51eSKate Stone       if (m_selected_row) {
273844d93782SGreg Clayton         if (m_selected_row->expanded)
273944d93782SGreg Clayton           m_selected_row->Unexpand();
274044d93782SGreg Clayton         else if (m_selected_row->parent)
274144d93782SGreg Clayton           m_selected_row_idx = m_selected_row->parent->row_idx;
274244d93782SGreg Clayton       }
274344d93782SGreg Clayton       return eKeyHandled;
274444d93782SGreg Clayton 
274544d93782SGreg Clayton     case ' ':
274644d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2747b9c1b51eSKate Stone       if (m_selected_row) {
274844d93782SGreg Clayton         if (m_selected_row->expanded)
274944d93782SGreg Clayton           m_selected_row->Unexpand();
275044d93782SGreg Clayton         else
275144d93782SGreg Clayton           m_selected_row->Expand();
275244d93782SGreg Clayton       }
275344d93782SGreg Clayton       return eKeyHandled;
275444d93782SGreg Clayton 
275544d93782SGreg Clayton     case 'h':
275644d93782SGreg Clayton       window.CreateHelpSubwindow();
275744d93782SGreg Clayton       return eKeyHandled;
275844d93782SGreg Clayton 
275944d93782SGreg Clayton     default:
276044d93782SGreg Clayton       break;
276144d93782SGreg Clayton     }
276244d93782SGreg Clayton     return eKeyNotHandled;
276344d93782SGreg Clayton   }
276444d93782SGreg Clayton 
276544d93782SGreg Clayton protected:
276644d93782SGreg Clayton   std::vector<Row> m_rows;
276744d93782SGreg Clayton   Row *m_selected_row;
276844d93782SGreg Clayton   uint32_t m_selected_row_idx;
276944d93782SGreg Clayton   uint32_t m_first_visible_row;
277044d93782SGreg Clayton   uint32_t m_num_rows;
277144d93782SGreg Clayton   int m_min_x;
277244d93782SGreg Clayton   int m_min_y;
277344d93782SGreg Clayton   int m_max_x;
277444d93782SGreg Clayton   int m_max_y;
277544d93782SGreg Clayton 
2776b9c1b51eSKate Stone   static Format FormatForChar(int c) {
2777b9c1b51eSKate Stone     switch (c) {
2778b9c1b51eSKate Stone     case 'x':
2779b9c1b51eSKate Stone       return eFormatHex;
2780b9c1b51eSKate Stone     case 'X':
2781b9c1b51eSKate Stone       return eFormatHexUppercase;
2782b9c1b51eSKate Stone     case 'o':
2783b9c1b51eSKate Stone       return eFormatOctal;
2784b9c1b51eSKate Stone     case 's':
2785b9c1b51eSKate Stone       return eFormatCString;
2786b9c1b51eSKate Stone     case 'u':
2787b9c1b51eSKate Stone       return eFormatUnsigned;
2788b9c1b51eSKate Stone     case 'd':
2789b9c1b51eSKate Stone       return eFormatDecimal;
2790b9c1b51eSKate Stone     case 'D':
2791b9c1b51eSKate Stone       return eFormatDefault;
2792b9c1b51eSKate Stone     case 'i':
2793b9c1b51eSKate Stone       return eFormatInstruction;
2794b9c1b51eSKate Stone     case 'A':
2795b9c1b51eSKate Stone       return eFormatAddressInfo;
2796b9c1b51eSKate Stone     case 'p':
2797b9c1b51eSKate Stone       return eFormatPointer;
2798b9c1b51eSKate Stone     case 'c':
2799b9c1b51eSKate Stone       return eFormatChar;
2800b9c1b51eSKate Stone     case 'b':
2801b9c1b51eSKate Stone       return eFormatBinary;
2802b9c1b51eSKate Stone     case 'B':
2803b9c1b51eSKate Stone       return eFormatBytesWithASCII;
2804b9c1b51eSKate Stone     case 'f':
2805b9c1b51eSKate Stone       return eFormatFloat;
280644d93782SGreg Clayton     }
280744d93782SGreg Clayton     return eFormatDefault;
280844d93782SGreg Clayton   }
280944d93782SGreg Clayton 
2810b9c1b51eSKate Stone   bool DisplayRowObject(Window &window, Row &row, DisplayOptions &options,
2811b9c1b51eSKate Stone                         bool highlight, bool last_child) {
28128369b28dSGreg Clayton     ValueObject *valobj = row.value.GetSP().get();
281344d93782SGreg Clayton 
2814c5dac77aSEugene Zelenko     if (valobj == nullptr)
281544d93782SGreg Clayton       return false;
281644d93782SGreg Clayton 
2817b9c1b51eSKate Stone     const char *type_name =
2818b9c1b51eSKate Stone         options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
281944d93782SGreg Clayton     const char *name = valobj->GetName().GetCString();
282044d93782SGreg Clayton     const char *value = valobj->GetValueAsCString();
282144d93782SGreg Clayton     const char *summary = valobj->GetSummaryAsCString();
282244d93782SGreg Clayton 
282344d93782SGreg Clayton     window.MoveCursor(row.x, row.y);
282444d93782SGreg Clayton 
282544d93782SGreg Clayton     row.DrawTree(window);
282644d93782SGreg Clayton 
282744d93782SGreg Clayton     if (highlight)
282844d93782SGreg Clayton       window.AttributeOn(A_REVERSE);
282944d93782SGreg Clayton 
283044d93782SGreg Clayton     if (type_name && type_name[0])
283144d93782SGreg Clayton       window.Printf("(%s) ", type_name);
283244d93782SGreg Clayton 
283344d93782SGreg Clayton     if (name && name[0])
283444d93782SGreg Clayton       window.PutCString(name);
283544d93782SGreg Clayton 
283644d93782SGreg Clayton     attr_t changd_attr = 0;
283744d93782SGreg Clayton     if (valobj->GetValueDidChange())
283844d93782SGreg Clayton       changd_attr = COLOR_PAIR(5) | A_BOLD;
283944d93782SGreg Clayton 
2840b9c1b51eSKate Stone     if (value && value[0]) {
284144d93782SGreg Clayton       window.PutCString(" = ");
284244d93782SGreg Clayton       if (changd_attr)
284344d93782SGreg Clayton         window.AttributeOn(changd_attr);
284444d93782SGreg Clayton       window.PutCString(value);
284544d93782SGreg Clayton       if (changd_attr)
284644d93782SGreg Clayton         window.AttributeOff(changd_attr);
284744d93782SGreg Clayton     }
284844d93782SGreg Clayton 
2849b9c1b51eSKate Stone     if (summary && summary[0]) {
285044d93782SGreg Clayton       window.PutChar(' ');
285144d93782SGreg Clayton       if (changd_attr)
285244d93782SGreg Clayton         window.AttributeOn(changd_attr);
285344d93782SGreg Clayton       window.PutCString(summary);
285444d93782SGreg Clayton       if (changd_attr)
285544d93782SGreg Clayton         window.AttributeOff(changd_attr);
285644d93782SGreg Clayton     }
285744d93782SGreg Clayton 
285844d93782SGreg Clayton     if (highlight)
285944d93782SGreg Clayton       window.AttributeOff(A_REVERSE);
286044d93782SGreg Clayton 
286144d93782SGreg Clayton     return true;
286244d93782SGreg Clayton   }
2863315b6884SEugene Zelenko 
2864b9c1b51eSKate Stone   void DisplayRows(Window &window, std::vector<Row> &rows,
2865b9c1b51eSKate Stone                    DisplayOptions &options) {
286644d93782SGreg Clayton     // >   0x25B7
286744d93782SGreg Clayton     // \/  0x25BD
286844d93782SGreg Clayton 
286944d93782SGreg Clayton     bool window_is_active = window.IsActive();
2870b9c1b51eSKate Stone     for (auto &row : rows) {
287144d93782SGreg Clayton       const bool last_child = row.parent && &rows[rows.size() - 1] == &row;
287244d93782SGreg Clayton       // Save the row index in each Row structure
287344d93782SGreg Clayton       row.row_idx = m_num_rows;
287444d93782SGreg Clayton       if ((m_num_rows >= m_first_visible_row) &&
2875b9c1b51eSKate Stone           ((m_num_rows - m_first_visible_row) <
2876b9c1b51eSKate Stone            static_cast<size_t>(NumVisibleRows()))) {
287744d93782SGreg Clayton         row.x = m_min_x;
287844d93782SGreg Clayton         row.y = m_num_rows - m_first_visible_row + 1;
2879b9c1b51eSKate Stone         if (DisplayRowObject(window, row, options,
2880b9c1b51eSKate Stone                              window_is_active &&
2881b9c1b51eSKate Stone                                  m_num_rows == m_selected_row_idx,
2882b9c1b51eSKate Stone                              last_child)) {
288344d93782SGreg Clayton           ++m_num_rows;
2884b9c1b51eSKate Stone         } else {
288544d93782SGreg Clayton           row.x = 0;
288644d93782SGreg Clayton           row.y = 0;
288744d93782SGreg Clayton         }
2888b9c1b51eSKate Stone       } else {
288944d93782SGreg Clayton         row.x = 0;
289044d93782SGreg Clayton         row.y = 0;
289144d93782SGreg Clayton         ++m_num_rows;
289244d93782SGreg Clayton       }
289344d93782SGreg Clayton 
28948369b28dSGreg Clayton       auto &children = row.GetChildren();
28958369b28dSGreg Clayton       if (row.expanded && !children.empty()) {
28968369b28dSGreg Clayton         DisplayRows(window, children, options);
289744d93782SGreg Clayton       }
289844d93782SGreg Clayton     }
289944d93782SGreg Clayton   }
290044d93782SGreg Clayton 
29018369b28dSGreg Clayton   int CalculateTotalNumberRows(std::vector<Row> &rows) {
290244d93782SGreg Clayton     int row_count = 0;
29038369b28dSGreg Clayton     for (auto &row : rows) {
290444d93782SGreg Clayton       ++row_count;
290544d93782SGreg Clayton       if (row.expanded)
29068369b28dSGreg Clayton         row_count += CalculateTotalNumberRows(row.GetChildren());
290744d93782SGreg Clayton     }
290844d93782SGreg Clayton     return row_count;
290944d93782SGreg Clayton   }
2910315b6884SEugene Zelenko 
2911b9c1b51eSKate Stone   static Row *GetRowForRowIndexImpl(std::vector<Row> &rows, size_t &row_index) {
2912b9c1b51eSKate Stone     for (auto &row : rows) {
291344d93782SGreg Clayton       if (row_index == 0)
291444d93782SGreg Clayton         return &row;
2915b9c1b51eSKate Stone       else {
291644d93782SGreg Clayton         --row_index;
29178369b28dSGreg Clayton         auto &children = row.GetChildren();
29188369b28dSGreg Clayton         if (row.expanded && !children.empty()) {
29198369b28dSGreg Clayton           Row *result = GetRowForRowIndexImpl(children, row_index);
292044d93782SGreg Clayton           if (result)
292144d93782SGreg Clayton             return result;
292244d93782SGreg Clayton         }
292344d93782SGreg Clayton       }
292444d93782SGreg Clayton     }
2925c5dac77aSEugene Zelenko     return nullptr;
292644d93782SGreg Clayton   }
292744d93782SGreg Clayton 
2928b9c1b51eSKate Stone   Row *GetRowForRowIndex(size_t row_index) {
292944d93782SGreg Clayton     return GetRowForRowIndexImpl(m_rows, row_index);
293044d93782SGreg Clayton   }
293144d93782SGreg Clayton 
2932b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
293344d93782SGreg Clayton 
293444d93782SGreg Clayton   static DisplayOptions g_options;
293544d93782SGreg Clayton };
293644d93782SGreg Clayton 
2937b9c1b51eSKate Stone class FrameVariablesWindowDelegate : public ValueObjectListDelegate {
293844d93782SGreg Clayton public:
2939b9c1b51eSKate Stone   FrameVariablesWindowDelegate(Debugger &debugger)
2940b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger),
2941b9c1b51eSKate Stone         m_frame_block(nullptr) {}
294244d93782SGreg Clayton 
2943315b6884SEugene Zelenko   ~FrameVariablesWindowDelegate() override = default;
294444d93782SGreg Clayton 
2945b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
294644d93782SGreg Clayton     return "Frame variable window keyboard shortcuts:";
294744d93782SGreg Clayton   }
294844d93782SGreg Clayton 
2949b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2950b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2951b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
295244d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
2953c5dac77aSEugene Zelenko     Block *frame_block = nullptr;
2954c5dac77aSEugene Zelenko     StackFrame *frame = nullptr;
295544d93782SGreg Clayton 
2956b9c1b51eSKate Stone     if (process) {
295744d93782SGreg Clayton       StateType state = process->GetState();
2958b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
295944d93782SGreg Clayton         frame = exe_ctx.GetFramePtr();
296044d93782SGreg Clayton         if (frame)
296144d93782SGreg Clayton           frame_block = frame->GetFrameBlock();
2962b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
296344d93782SGreg Clayton         return true; // Don't do any updating when we are running
296444d93782SGreg Clayton       }
296544d93782SGreg Clayton     }
296644d93782SGreg Clayton 
296744d93782SGreg Clayton     ValueObjectList local_values;
2968b9c1b51eSKate Stone     if (frame_block) {
296944d93782SGreg Clayton       // Only update the variables if they have changed
2970b9c1b51eSKate Stone       if (m_frame_block != frame_block) {
297144d93782SGreg Clayton         m_frame_block = frame_block;
297244d93782SGreg Clayton 
297344d93782SGreg Clayton         VariableList *locals = frame->GetVariableList(true);
2974b9c1b51eSKate Stone         if (locals) {
297544d93782SGreg Clayton           const DynamicValueType use_dynamic = eDynamicDontRunTarget;
297644d93782SGreg Clayton           const size_t num_locals = locals->GetSize();
2977b9c1b51eSKate Stone           for (size_t i = 0; i < num_locals; ++i) {
2978b9c1b51eSKate Stone             ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable(
2979b9c1b51eSKate Stone                 locals->GetVariableAtIndex(i), use_dynamic);
2980b9c1b51eSKate Stone             if (value_sp) {
2981eb72dc7dSGreg Clayton               ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
2982eb72dc7dSGreg Clayton               if (synthetic_value_sp)
2983eb72dc7dSGreg Clayton                 local_values.Append(synthetic_value_sp);
2984eb72dc7dSGreg Clayton               else
2985eb72dc7dSGreg Clayton                 local_values.Append(value_sp);
2986eb72dc7dSGreg Clayton             }
2987eb72dc7dSGreg Clayton           }
298844d93782SGreg Clayton           // Update the values
298944d93782SGreg Clayton           SetValues(local_values);
299044d93782SGreg Clayton         }
299144d93782SGreg Clayton       }
2992b9c1b51eSKate Stone     } else {
2993c5dac77aSEugene Zelenko       m_frame_block = nullptr;
299444d93782SGreg Clayton       // Update the values with an empty list if there is no frame
299544d93782SGreg Clayton       SetValues(local_values);
299644d93782SGreg Clayton     }
299744d93782SGreg Clayton 
299844d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
299944d93782SGreg Clayton   }
300044d93782SGreg Clayton 
300144d93782SGreg Clayton protected:
300244d93782SGreg Clayton   Debugger &m_debugger;
300344d93782SGreg Clayton   Block *m_frame_block;
300444d93782SGreg Clayton };
300544d93782SGreg Clayton 
3006b9c1b51eSKate Stone class RegistersWindowDelegate : public ValueObjectListDelegate {
300744d93782SGreg Clayton public:
3008b9c1b51eSKate Stone   RegistersWindowDelegate(Debugger &debugger)
3009b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger) {}
301044d93782SGreg Clayton 
3011315b6884SEugene Zelenko   ~RegistersWindowDelegate() override = default;
301244d93782SGreg Clayton 
3013b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
301444d93782SGreg Clayton     return "Register window keyboard shortcuts:";
301544d93782SGreg Clayton   }
301644d93782SGreg Clayton 
3017b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3018b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3019b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
302044d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
302144d93782SGreg Clayton 
302244d93782SGreg Clayton     ValueObjectList value_list;
3023b9c1b51eSKate Stone     if (frame) {
3024b9c1b51eSKate Stone       if (frame->GetStackID() != m_stack_id) {
302544d93782SGreg Clayton         m_stack_id = frame->GetStackID();
302644d93782SGreg Clayton         RegisterContextSP reg_ctx(frame->GetRegisterContext());
3027b9c1b51eSKate Stone         if (reg_ctx) {
302844d93782SGreg Clayton           const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
3029b9c1b51eSKate Stone           for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) {
3030b9c1b51eSKate Stone             value_list.Append(
3031b9c1b51eSKate Stone                 ValueObjectRegisterSet::Create(frame, reg_ctx, set_idx));
303244d93782SGreg Clayton           }
303344d93782SGreg Clayton         }
303444d93782SGreg Clayton         SetValues(value_list);
303544d93782SGreg Clayton       }
3036b9c1b51eSKate Stone     } else {
303744d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
303844d93782SGreg Clayton       if (process && process->IsAlive())
303944d93782SGreg Clayton         return true; // Don't do any updating if we are running
3040b9c1b51eSKate Stone       else {
304105097246SAdrian Prantl         // Update the values with an empty list if there is no process or the
304205097246SAdrian Prantl         // process isn't alive anymore
304344d93782SGreg Clayton         SetValues(value_list);
304444d93782SGreg Clayton       }
304544d93782SGreg Clayton     }
304644d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
304744d93782SGreg Clayton   }
304844d93782SGreg Clayton 
304944d93782SGreg Clayton protected:
305044d93782SGreg Clayton   Debugger &m_debugger;
305144d93782SGreg Clayton   StackID m_stack_id;
305244d93782SGreg Clayton };
305344d93782SGreg Clayton 
3054b9c1b51eSKate Stone static const char *CursesKeyToCString(int ch) {
305544d93782SGreg Clayton   static char g_desc[32];
3056b9c1b51eSKate Stone   if (ch >= KEY_F0 && ch < KEY_F0 + 64) {
305744d93782SGreg Clayton     snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
305844d93782SGreg Clayton     return g_desc;
305944d93782SGreg Clayton   }
3060b9c1b51eSKate Stone   switch (ch) {
3061b9c1b51eSKate Stone   case KEY_DOWN:
3062b9c1b51eSKate Stone     return "down";
3063b9c1b51eSKate Stone   case KEY_UP:
3064b9c1b51eSKate Stone     return "up";
3065b9c1b51eSKate Stone   case KEY_LEFT:
3066b9c1b51eSKate Stone     return "left";
3067b9c1b51eSKate Stone   case KEY_RIGHT:
3068b9c1b51eSKate Stone     return "right";
3069b9c1b51eSKate Stone   case KEY_HOME:
3070b9c1b51eSKate Stone     return "home";
3071b9c1b51eSKate Stone   case KEY_BACKSPACE:
3072b9c1b51eSKate Stone     return "backspace";
3073b9c1b51eSKate Stone   case KEY_DL:
3074b9c1b51eSKate Stone     return "delete-line";
3075b9c1b51eSKate Stone   case KEY_IL:
3076b9c1b51eSKate Stone     return "insert-line";
3077b9c1b51eSKate Stone   case KEY_DC:
3078b9c1b51eSKate Stone     return "delete-char";
3079b9c1b51eSKate Stone   case KEY_IC:
3080b9c1b51eSKate Stone     return "insert-char";
3081b9c1b51eSKate Stone   case KEY_CLEAR:
3082b9c1b51eSKate Stone     return "clear";
3083b9c1b51eSKate Stone   case KEY_EOS:
3084b9c1b51eSKate Stone     return "clear-to-eos";
3085b9c1b51eSKate Stone   case KEY_EOL:
3086b9c1b51eSKate Stone     return "clear-to-eol";
3087b9c1b51eSKate Stone   case KEY_SF:
3088b9c1b51eSKate Stone     return "scroll-forward";
3089b9c1b51eSKate Stone   case KEY_SR:
3090b9c1b51eSKate Stone     return "scroll-backward";
3091b9c1b51eSKate Stone   case KEY_NPAGE:
3092b9c1b51eSKate Stone     return "page-down";
3093b9c1b51eSKate Stone   case KEY_PPAGE:
3094b9c1b51eSKate Stone     return "page-up";
3095b9c1b51eSKate Stone   case KEY_STAB:
3096b9c1b51eSKate Stone     return "set-tab";
3097b9c1b51eSKate Stone   case KEY_CTAB:
3098b9c1b51eSKate Stone     return "clear-tab";
3099b9c1b51eSKate Stone   case KEY_CATAB:
3100b9c1b51eSKate Stone     return "clear-all-tabs";
3101b9c1b51eSKate Stone   case KEY_ENTER:
3102b9c1b51eSKate Stone     return "enter";
3103b9c1b51eSKate Stone   case KEY_PRINT:
3104b9c1b51eSKate Stone     return "print";
3105b9c1b51eSKate Stone   case KEY_LL:
3106b9c1b51eSKate Stone     return "lower-left key";
3107b9c1b51eSKate Stone   case KEY_A1:
3108b9c1b51eSKate Stone     return "upper left of keypad";
3109b9c1b51eSKate Stone   case KEY_A3:
3110b9c1b51eSKate Stone     return "upper right of keypad";
3111b9c1b51eSKate Stone   case KEY_B2:
3112b9c1b51eSKate Stone     return "center of keypad";
3113b9c1b51eSKate Stone   case KEY_C1:
3114b9c1b51eSKate Stone     return "lower left of keypad";
3115b9c1b51eSKate Stone   case KEY_C3:
3116b9c1b51eSKate Stone     return "lower right of keypad";
3117b9c1b51eSKate Stone   case KEY_BTAB:
3118b9c1b51eSKate Stone     return "back-tab key";
3119b9c1b51eSKate Stone   case KEY_BEG:
3120b9c1b51eSKate Stone     return "begin key";
3121b9c1b51eSKate Stone   case KEY_CANCEL:
3122b9c1b51eSKate Stone     return "cancel key";
3123b9c1b51eSKate Stone   case KEY_CLOSE:
3124b9c1b51eSKate Stone     return "close key";
3125b9c1b51eSKate Stone   case KEY_COMMAND:
3126b9c1b51eSKate Stone     return "command key";
3127b9c1b51eSKate Stone   case KEY_COPY:
3128b9c1b51eSKate Stone     return "copy key";
3129b9c1b51eSKate Stone   case KEY_CREATE:
3130b9c1b51eSKate Stone     return "create key";
3131b9c1b51eSKate Stone   case KEY_END:
3132b9c1b51eSKate Stone     return "end key";
3133b9c1b51eSKate Stone   case KEY_EXIT:
3134b9c1b51eSKate Stone     return "exit key";
3135b9c1b51eSKate Stone   case KEY_FIND:
3136b9c1b51eSKate Stone     return "find key";
3137b9c1b51eSKate Stone   case KEY_HELP:
3138b9c1b51eSKate Stone     return "help key";
3139b9c1b51eSKate Stone   case KEY_MARK:
3140b9c1b51eSKate Stone     return "mark key";
3141b9c1b51eSKate Stone   case KEY_MESSAGE:
3142b9c1b51eSKate Stone     return "message key";
3143b9c1b51eSKate Stone   case KEY_MOVE:
3144b9c1b51eSKate Stone     return "move key";
3145b9c1b51eSKate Stone   case KEY_NEXT:
3146b9c1b51eSKate Stone     return "next key";
3147b9c1b51eSKate Stone   case KEY_OPEN:
3148b9c1b51eSKate Stone     return "open key";
3149b9c1b51eSKate Stone   case KEY_OPTIONS:
3150b9c1b51eSKate Stone     return "options key";
3151b9c1b51eSKate Stone   case KEY_PREVIOUS:
3152b9c1b51eSKate Stone     return "previous key";
3153b9c1b51eSKate Stone   case KEY_REDO:
3154b9c1b51eSKate Stone     return "redo key";
3155b9c1b51eSKate Stone   case KEY_REFERENCE:
3156b9c1b51eSKate Stone     return "reference key";
3157b9c1b51eSKate Stone   case KEY_REFRESH:
3158b9c1b51eSKate Stone     return "refresh key";
3159b9c1b51eSKate Stone   case KEY_REPLACE:
3160b9c1b51eSKate Stone     return "replace key";
3161b9c1b51eSKate Stone   case KEY_RESTART:
3162b9c1b51eSKate Stone     return "restart key";
3163b9c1b51eSKate Stone   case KEY_RESUME:
3164b9c1b51eSKate Stone     return "resume key";
3165b9c1b51eSKate Stone   case KEY_SAVE:
3166b9c1b51eSKate Stone     return "save key";
3167b9c1b51eSKate Stone   case KEY_SBEG:
3168b9c1b51eSKate Stone     return "shifted begin key";
3169b9c1b51eSKate Stone   case KEY_SCANCEL:
3170b9c1b51eSKate Stone     return "shifted cancel key";
3171b9c1b51eSKate Stone   case KEY_SCOMMAND:
3172b9c1b51eSKate Stone     return "shifted command key";
3173b9c1b51eSKate Stone   case KEY_SCOPY:
3174b9c1b51eSKate Stone     return "shifted copy key";
3175b9c1b51eSKate Stone   case KEY_SCREATE:
3176b9c1b51eSKate Stone     return "shifted create key";
3177b9c1b51eSKate Stone   case KEY_SDC:
3178b9c1b51eSKate Stone     return "shifted delete-character key";
3179b9c1b51eSKate Stone   case KEY_SDL:
3180b9c1b51eSKate Stone     return "shifted delete-line key";
3181b9c1b51eSKate Stone   case KEY_SELECT:
3182b9c1b51eSKate Stone     return "select key";
3183b9c1b51eSKate Stone   case KEY_SEND:
3184b9c1b51eSKate Stone     return "shifted end key";
3185b9c1b51eSKate Stone   case KEY_SEOL:
3186b9c1b51eSKate Stone     return "shifted clear-to-end-of-line key";
3187b9c1b51eSKate Stone   case KEY_SEXIT:
3188b9c1b51eSKate Stone     return "shifted exit key";
3189b9c1b51eSKate Stone   case KEY_SFIND:
3190b9c1b51eSKate Stone     return "shifted find key";
3191b9c1b51eSKate Stone   case KEY_SHELP:
3192b9c1b51eSKate Stone     return "shifted help key";
3193b9c1b51eSKate Stone   case KEY_SHOME:
3194b9c1b51eSKate Stone     return "shifted home key";
3195b9c1b51eSKate Stone   case KEY_SIC:
3196b9c1b51eSKate Stone     return "shifted insert-character key";
3197b9c1b51eSKate Stone   case KEY_SLEFT:
3198b9c1b51eSKate Stone     return "shifted left-arrow key";
3199b9c1b51eSKate Stone   case KEY_SMESSAGE:
3200b9c1b51eSKate Stone     return "shifted message key";
3201b9c1b51eSKate Stone   case KEY_SMOVE:
3202b9c1b51eSKate Stone     return "shifted move key";
3203b9c1b51eSKate Stone   case KEY_SNEXT:
3204b9c1b51eSKate Stone     return "shifted next key";
3205b9c1b51eSKate Stone   case KEY_SOPTIONS:
3206b9c1b51eSKate Stone     return "shifted options key";
3207b9c1b51eSKate Stone   case KEY_SPREVIOUS:
3208b9c1b51eSKate Stone     return "shifted previous key";
3209b9c1b51eSKate Stone   case KEY_SPRINT:
3210b9c1b51eSKate Stone     return "shifted print key";
3211b9c1b51eSKate Stone   case KEY_SREDO:
3212b9c1b51eSKate Stone     return "shifted redo key";
3213b9c1b51eSKate Stone   case KEY_SREPLACE:
3214b9c1b51eSKate Stone     return "shifted replace key";
3215b9c1b51eSKate Stone   case KEY_SRIGHT:
3216b9c1b51eSKate Stone     return "shifted right-arrow key";
3217b9c1b51eSKate Stone   case KEY_SRSUME:
3218b9c1b51eSKate Stone     return "shifted resume key";
3219b9c1b51eSKate Stone   case KEY_SSAVE:
3220b9c1b51eSKate Stone     return "shifted save key";
3221b9c1b51eSKate Stone   case KEY_SSUSPEND:
3222b9c1b51eSKate Stone     return "shifted suspend key";
3223b9c1b51eSKate Stone   case KEY_SUNDO:
3224b9c1b51eSKate Stone     return "shifted undo key";
3225b9c1b51eSKate Stone   case KEY_SUSPEND:
3226b9c1b51eSKate Stone     return "suspend key";
3227b9c1b51eSKate Stone   case KEY_UNDO:
3228b9c1b51eSKate Stone     return "undo key";
3229b9c1b51eSKate Stone   case KEY_MOUSE:
3230b9c1b51eSKate Stone     return "Mouse event has occurred";
3231b9c1b51eSKate Stone   case KEY_RESIZE:
3232b9c1b51eSKate Stone     return "Terminal resize event";
323327801f4fSBruce Mitchener #ifdef KEY_EVENT
3234b9c1b51eSKate Stone   case KEY_EVENT:
3235b9c1b51eSKate Stone     return "We were interrupted by an event";
323627801f4fSBruce Mitchener #endif
3237b9c1b51eSKate Stone   case KEY_RETURN:
3238b9c1b51eSKate Stone     return "return";
3239b9c1b51eSKate Stone   case ' ':
3240b9c1b51eSKate Stone     return "space";
3241b9c1b51eSKate Stone   case '\t':
3242b9c1b51eSKate Stone     return "tab";
3243b9c1b51eSKate Stone   case KEY_ESCAPE:
3244b9c1b51eSKate Stone     return "escape";
324544d93782SGreg Clayton   default:
324644d93782SGreg Clayton     if (isprint(ch))
324744d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "%c", ch);
324844d93782SGreg Clayton     else
324944d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
325044d93782SGreg Clayton     return g_desc;
325144d93782SGreg Clayton   }
3252c5dac77aSEugene Zelenko   return nullptr;
325344d93782SGreg Clayton }
325444d93782SGreg Clayton 
3255b9c1b51eSKate Stone HelpDialogDelegate::HelpDialogDelegate(const char *text,
3256b9c1b51eSKate Stone                                        KeyHelp *key_help_array)
3257b9c1b51eSKate Stone     : m_text(), m_first_visible_line(0) {
3258b9c1b51eSKate Stone   if (text && text[0]) {
325944d93782SGreg Clayton     m_text.SplitIntoLines(text);
326044d93782SGreg Clayton     m_text.AppendString("");
326144d93782SGreg Clayton   }
3262b9c1b51eSKate Stone   if (key_help_array) {
3263b9c1b51eSKate Stone     for (KeyHelp *key = key_help_array; key->ch; ++key) {
326444d93782SGreg Clayton       StreamString key_description;
3265b9c1b51eSKate Stone       key_description.Printf("%10s - %s", CursesKeyToCString(key->ch),
3266b9c1b51eSKate Stone                              key->description);
3267c156427dSZachary Turner       m_text.AppendString(key_description.GetString());
326844d93782SGreg Clayton     }
326944d93782SGreg Clayton   }
327044d93782SGreg Clayton }
327144d93782SGreg Clayton 
3272315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
327344d93782SGreg Clayton 
3274b9c1b51eSKate Stone bool HelpDialogDelegate::WindowDelegateDraw(Window &window, bool force) {
327544d93782SGreg Clayton   window.Erase();
327644d93782SGreg Clayton   const int window_height = window.GetHeight();
327744d93782SGreg Clayton   int x = 2;
327844d93782SGreg Clayton   int y = 1;
327944d93782SGreg Clayton   const int min_y = y;
328044d93782SGreg Clayton   const int max_y = window_height - 1 - y;
32813985c8c6SSaleem Abdulrasool   const size_t num_visible_lines = max_y - min_y + 1;
328244d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
328344d93782SGreg Clayton   const char *bottom_message;
328444d93782SGreg Clayton   if (num_lines <= num_visible_lines)
328544d93782SGreg Clayton     bottom_message = "Press any key to exit";
328644d93782SGreg Clayton   else
328744d93782SGreg Clayton     bottom_message = "Use arrows to scroll, any other key to exit";
328844d93782SGreg Clayton   window.DrawTitleBox(window.GetName(), bottom_message);
3289b9c1b51eSKate Stone   while (y <= max_y) {
329044d93782SGreg Clayton     window.MoveCursor(x, y);
3291b9c1b51eSKate Stone     window.PutCStringTruncated(
3292b9c1b51eSKate Stone         m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
329344d93782SGreg Clayton     ++y;
329444d93782SGreg Clayton   }
329544d93782SGreg Clayton   return true;
329644d93782SGreg Clayton }
329744d93782SGreg Clayton 
3298b9c1b51eSKate Stone HandleCharResult HelpDialogDelegate::WindowDelegateHandleChar(Window &window,
3299b9c1b51eSKate Stone                                                               int key) {
330044d93782SGreg Clayton   bool done = false;
330144d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
330244d93782SGreg Clayton   const size_t num_visible_lines = window.GetHeight() - 2;
330344d93782SGreg Clayton 
3304b9c1b51eSKate Stone   if (num_lines <= num_visible_lines) {
330544d93782SGreg Clayton     done = true;
330605097246SAdrian Prantl     // If we have all lines visible and don't need scrolling, then any key
330705097246SAdrian Prantl     // press will cause us to exit
3308b9c1b51eSKate Stone   } else {
3309b9c1b51eSKate Stone     switch (key) {
331044d93782SGreg Clayton     case KEY_UP:
331144d93782SGreg Clayton       if (m_first_visible_line > 0)
331244d93782SGreg Clayton         --m_first_visible_line;
331344d93782SGreg Clayton       break;
331444d93782SGreg Clayton 
331544d93782SGreg Clayton     case KEY_DOWN:
331644d93782SGreg Clayton       if (m_first_visible_line + num_visible_lines < num_lines)
331744d93782SGreg Clayton         ++m_first_visible_line;
331844d93782SGreg Clayton       break;
331944d93782SGreg Clayton 
332044d93782SGreg Clayton     case KEY_PPAGE:
332144d93782SGreg Clayton     case ',':
3322b9c1b51eSKate Stone       if (m_first_visible_line > 0) {
33233985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
332444d93782SGreg Clayton           m_first_visible_line -= num_visible_lines;
332544d93782SGreg Clayton         else
332644d93782SGreg Clayton           m_first_visible_line = 0;
332744d93782SGreg Clayton       }
332844d93782SGreg Clayton       break;
3329315b6884SEugene Zelenko 
333044d93782SGreg Clayton     case KEY_NPAGE:
333144d93782SGreg Clayton     case '.':
3332b9c1b51eSKate Stone       if (m_first_visible_line + num_visible_lines < num_lines) {
333344d93782SGreg Clayton         m_first_visible_line += num_visible_lines;
33343985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > num_lines)
333544d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
333644d93782SGreg Clayton       }
333744d93782SGreg Clayton       break;
3338315b6884SEugene Zelenko 
333944d93782SGreg Clayton     default:
334044d93782SGreg Clayton       done = true;
334144d93782SGreg Clayton       break;
334244d93782SGreg Clayton     }
334344d93782SGreg Clayton   }
334444d93782SGreg Clayton   if (done)
334544d93782SGreg Clayton     window.GetParent()->RemoveSubWindow(&window);
334644d93782SGreg Clayton   return eKeyHandled;
334744d93782SGreg Clayton }
334844d93782SGreg Clayton 
3349b9c1b51eSKate Stone class ApplicationDelegate : public WindowDelegate, public MenuDelegate {
335044d93782SGreg Clayton public:
335144d93782SGreg Clayton   enum {
335244d93782SGreg Clayton     eMenuID_LLDB = 1,
335344d93782SGreg Clayton     eMenuID_LLDBAbout,
335444d93782SGreg Clayton     eMenuID_LLDBExit,
335544d93782SGreg Clayton 
335644d93782SGreg Clayton     eMenuID_Target,
335744d93782SGreg Clayton     eMenuID_TargetCreate,
335844d93782SGreg Clayton     eMenuID_TargetDelete,
335944d93782SGreg Clayton 
336044d93782SGreg Clayton     eMenuID_Process,
336144d93782SGreg Clayton     eMenuID_ProcessAttach,
336244d93782SGreg Clayton     eMenuID_ProcessDetach,
336344d93782SGreg Clayton     eMenuID_ProcessLaunch,
336444d93782SGreg Clayton     eMenuID_ProcessContinue,
336544d93782SGreg Clayton     eMenuID_ProcessHalt,
336644d93782SGreg Clayton     eMenuID_ProcessKill,
336744d93782SGreg Clayton 
336844d93782SGreg Clayton     eMenuID_Thread,
336944d93782SGreg Clayton     eMenuID_ThreadStepIn,
337044d93782SGreg Clayton     eMenuID_ThreadStepOver,
337144d93782SGreg Clayton     eMenuID_ThreadStepOut,
337244d93782SGreg Clayton 
337344d93782SGreg Clayton     eMenuID_View,
337444d93782SGreg Clayton     eMenuID_ViewBacktrace,
337544d93782SGreg Clayton     eMenuID_ViewRegisters,
337644d93782SGreg Clayton     eMenuID_ViewSource,
337744d93782SGreg Clayton     eMenuID_ViewVariables,
337844d93782SGreg Clayton 
337944d93782SGreg Clayton     eMenuID_Help,
338044d93782SGreg Clayton     eMenuID_HelpGUIHelp
338144d93782SGreg Clayton   };
338244d93782SGreg Clayton 
3383b9c1b51eSKate Stone   ApplicationDelegate(Application &app, Debugger &debugger)
3384b9c1b51eSKate Stone       : WindowDelegate(), MenuDelegate(), m_app(app), m_debugger(debugger) {}
338544d93782SGreg Clayton 
3386315b6884SEugene Zelenko   ~ApplicationDelegate() override = default;
3387bd5ae6b4SGreg Clayton 
3388b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
338944d93782SGreg Clayton     return false; // Drawing not handled, let standard window drawing happen
339044d93782SGreg Clayton   }
339144d93782SGreg Clayton 
3392b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override {
3393b9c1b51eSKate Stone     switch (key) {
33945fdb09bbSGreg Clayton     case '\t':
339544d93782SGreg Clayton       window.SelectNextWindowAsActive();
339644d93782SGreg Clayton       return eKeyHandled;
33975fdb09bbSGreg Clayton 
33985fdb09bbSGreg Clayton     case 'h':
33995fdb09bbSGreg Clayton       window.CreateHelpSubwindow();
34005fdb09bbSGreg Clayton       return eKeyHandled;
34015fdb09bbSGreg Clayton 
34025fdb09bbSGreg Clayton     case KEY_ESCAPE:
34035fdb09bbSGreg Clayton       return eQuitApplication;
34045fdb09bbSGreg Clayton 
34055fdb09bbSGreg Clayton     default:
34065fdb09bbSGreg Clayton       break;
340744d93782SGreg Clayton     }
340844d93782SGreg Clayton     return eKeyNotHandled;
340944d93782SGreg Clayton   }
341044d93782SGreg Clayton 
3411b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
34125fdb09bbSGreg Clayton     return "Welcome to the LLDB curses GUI.\n\n"
34135fdb09bbSGreg Clayton            "Press the TAB key to change the selected view.\n"
3414b9c1b51eSKate Stone            "Each view has its own keyboard shortcuts, press 'h' to open a "
3415b9c1b51eSKate Stone            "dialog to display them.\n\n"
34165fdb09bbSGreg Clayton            "Common key bindings for all views:";
34175fdb09bbSGreg Clayton   }
34185fdb09bbSGreg Clayton 
3419b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
34205fdb09bbSGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
34215fdb09bbSGreg Clayton         {'\t', "Select next view"},
34225fdb09bbSGreg Clayton         {'h', "Show help dialog with view specific key bindings"},
34235fdb09bbSGreg Clayton         {',', "Page up"},
34245fdb09bbSGreg Clayton         {'.', "Page down"},
34255fdb09bbSGreg Clayton         {KEY_UP, "Select previous"},
34265fdb09bbSGreg Clayton         {KEY_DOWN, "Select next"},
34275fdb09bbSGreg Clayton         {KEY_LEFT, "Unexpand or select parent"},
34285fdb09bbSGreg Clayton         {KEY_RIGHT, "Expand"},
34295fdb09bbSGreg Clayton         {KEY_PPAGE, "Page up"},
34305fdb09bbSGreg Clayton         {KEY_NPAGE, "Page down"},
3431b9c1b51eSKate Stone         {'\0', nullptr}};
34325fdb09bbSGreg Clayton     return g_source_view_key_help;
34335fdb09bbSGreg Clayton   }
34345fdb09bbSGreg Clayton 
3435b9c1b51eSKate Stone   MenuActionResult MenuDelegateAction(Menu &menu) override {
3436b9c1b51eSKate Stone     switch (menu.GetIdentifier()) {
3437b9c1b51eSKate Stone     case eMenuID_ThreadStepIn: {
3438b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3439b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3440b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
344144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3442b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3443b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
34444b4b2478SJim Ingham           exe_ctx.GetThreadRef().StepIn(true);
344544d93782SGreg Clayton       }
344644d93782SGreg Clayton     }
344744d93782SGreg Clayton       return MenuActionResult::Handled;
344844d93782SGreg Clayton 
3449b9c1b51eSKate Stone     case eMenuID_ThreadStepOut: {
3450b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3451b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3452b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
345344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3454b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3455b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
345644d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
345744d93782SGreg Clayton       }
345844d93782SGreg Clayton     }
345944d93782SGreg Clayton       return MenuActionResult::Handled;
346044d93782SGreg Clayton 
3461b9c1b51eSKate Stone     case eMenuID_ThreadStepOver: {
3462b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3463b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3464b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
346544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3466b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3467b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
346844d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOver(true);
346944d93782SGreg Clayton       }
347044d93782SGreg Clayton     }
347144d93782SGreg Clayton       return MenuActionResult::Handled;
347244d93782SGreg Clayton 
3473b9c1b51eSKate Stone     case eMenuID_ProcessContinue: {
3474b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3475b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3476b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
347744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3478b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3479b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
348044d93782SGreg Clayton           process->Resume();
348144d93782SGreg Clayton       }
348244d93782SGreg Clayton     }
348344d93782SGreg Clayton       return MenuActionResult::Handled;
348444d93782SGreg Clayton 
3485b9c1b51eSKate Stone     case eMenuID_ProcessKill: {
3486b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3487b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3488b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
348944d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
349044d93782SGreg Clayton         if (process && process->IsAlive())
3491ede3193bSJason Molenda           process->Destroy(false);
349244d93782SGreg Clayton       }
349344d93782SGreg Clayton     }
349444d93782SGreg Clayton       return MenuActionResult::Handled;
349544d93782SGreg Clayton 
3496b9c1b51eSKate Stone     case eMenuID_ProcessHalt: {
3497b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3498b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3499b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
350044d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
350144d93782SGreg Clayton         if (process && process->IsAlive())
350244d93782SGreg Clayton           process->Halt();
350344d93782SGreg Clayton       }
350444d93782SGreg Clayton     }
350544d93782SGreg Clayton       return MenuActionResult::Handled;
350644d93782SGreg Clayton 
3507b9c1b51eSKate Stone     case eMenuID_ProcessDetach: {
3508b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3509b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3510b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
351144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
351244d93782SGreg Clayton         if (process && process->IsAlive())
351344d93782SGreg Clayton           process->Detach(false);
351444d93782SGreg Clayton       }
351544d93782SGreg Clayton     }
351644d93782SGreg Clayton       return MenuActionResult::Handled;
351744d93782SGreg Clayton 
3518b9c1b51eSKate Stone     case eMenuID_Process: {
3519b9c1b51eSKate Stone       // Populate the menu with all of the threads if the process is stopped
352005097246SAdrian Prantl       // when the Process menu gets selected and is about to display its
352105097246SAdrian Prantl       // submenu.
352244d93782SGreg Clayton       Menus &submenus = menu.GetSubmenus();
3523b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3524b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
352544d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
3526b9c1b51eSKate Stone       if (process && process->IsAlive() &&
3527b9c1b51eSKate Stone           StateIsStoppedState(process->GetState(), true)) {
352844d93782SGreg Clayton         if (submenus.size() == 7)
352944d93782SGreg Clayton           menu.AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
353044d93782SGreg Clayton         else if (submenus.size() > 8)
353144d93782SGreg Clayton           submenus.erase(submenus.begin() + 8, submenus.end());
353244d93782SGreg Clayton 
353344d93782SGreg Clayton         ThreadList &threads = process->GetThreadList();
3534bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
353544d93782SGreg Clayton         size_t num_threads = threads.GetSize();
3536b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
353744d93782SGreg Clayton           ThreadSP thread_sp = threads.GetThreadAtIndex(i);
353844d93782SGreg Clayton           char menu_char = '\0';
353944d93782SGreg Clayton           if (i < 9)
354044d93782SGreg Clayton             menu_char = '1' + i;
354144d93782SGreg Clayton           StreamString thread_menu_title;
354244d93782SGreg Clayton           thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
354344d93782SGreg Clayton           const char *thread_name = thread_sp->GetName();
354444d93782SGreg Clayton           if (thread_name && thread_name[0])
354544d93782SGreg Clayton             thread_menu_title.Printf(" %s", thread_name);
3546b9c1b51eSKate Stone           else {
354744d93782SGreg Clayton             const char *queue_name = thread_sp->GetQueueName();
354844d93782SGreg Clayton             if (queue_name && queue_name[0])
354944d93782SGreg Clayton               thread_menu_title.Printf(" %s", queue_name);
355044d93782SGreg Clayton           }
3551b9c1b51eSKate Stone           menu.AddSubmenu(
3552c156427dSZachary Turner               MenuSP(new Menu(thread_menu_title.GetString().str().c_str(),
3553c156427dSZachary Turner                               nullptr, menu_char, thread_sp->GetID())));
355444d93782SGreg Clayton         }
3555b9c1b51eSKate Stone       } else if (submenus.size() > 7) {
355605097246SAdrian Prantl         // Remove the separator and any other thread submenu items that were
355705097246SAdrian Prantl         // previously added
355844d93782SGreg Clayton         submenus.erase(submenus.begin() + 7, submenus.end());
355944d93782SGreg Clayton       }
3560b9c1b51eSKate Stone       // Since we are adding and removing items we need to recalculate the name
3561b9c1b51eSKate Stone       // lengths
356244d93782SGreg Clayton       menu.RecalculateNameLengths();
356344d93782SGreg Clayton     }
356444d93782SGreg Clayton       return MenuActionResult::Handled;
356544d93782SGreg Clayton 
3566b9c1b51eSKate Stone     case eMenuID_ViewVariables: {
356744d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
356844d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
356944d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
357044d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
357144d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
357244d93782SGreg Clayton 
3573b9c1b51eSKate Stone       if (variables_window_sp) {
357444d93782SGreg Clayton         const Rect variables_bounds = variables_window_sp->GetBounds();
357544d93782SGreg Clayton 
357644d93782SGreg Clayton         main_window_sp->RemoveSubWindow(variables_window_sp.get());
357744d93782SGreg Clayton 
3578b9c1b51eSKate Stone         if (registers_window_sp) {
3579b9c1b51eSKate Stone           // We have a registers window, so give all the area back to the
3580b9c1b51eSKate Stone           // registers window
358144d93782SGreg Clayton           Rect registers_bounds = variables_bounds;
358244d93782SGreg Clayton           registers_bounds.size.width = source_bounds.size.width;
358344d93782SGreg Clayton           registers_window_sp->SetBounds(registers_bounds);
3584b9c1b51eSKate Stone         } else {
358505097246SAdrian Prantl           // We have no registers window showing so give the bottom area back
358605097246SAdrian Prantl           // to the source view
358744d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3588b9c1b51eSKate Stone                                    source_bounds.size.height +
3589b9c1b51eSKate Stone                                        variables_bounds.size.height);
359044d93782SGreg Clayton         }
3591b9c1b51eSKate Stone       } else {
359244d93782SGreg Clayton         Rect new_variables_rect;
3593b9c1b51eSKate Stone         if (registers_window_sp) {
359444d93782SGreg Clayton           // We have a registers window so split the area of the registers
359544d93782SGreg Clayton           // window into two columns where the left hand side will be the
359644d93782SGreg Clayton           // variables and the right hand side will be the registers
359744d93782SGreg Clayton           const Rect variables_bounds = registers_window_sp->GetBounds();
359844d93782SGreg Clayton           Rect new_registers_rect;
3599b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_variables_rect,
3600b9c1b51eSKate Stone                                                    new_registers_rect);
360144d93782SGreg Clayton           registers_window_sp->SetBounds(new_registers_rect);
3602b9c1b51eSKate Stone         } else {
360344d93782SGreg Clayton           // No variables window, grab the bottom part of the source window
360444d93782SGreg Clayton           Rect new_source_rect;
3605b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3606b9c1b51eSKate Stone                                                   new_variables_rect);
360744d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
360844d93782SGreg Clayton         }
3609b9c1b51eSKate Stone         WindowSP new_window_sp = main_window_sp->CreateSubWindow(
3610b9c1b51eSKate Stone             "Variables", new_variables_rect, false);
3611b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3612b9c1b51eSKate Stone             WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
361344d93782SGreg Clayton       }
361444d93782SGreg Clayton       touchwin(stdscr);
361544d93782SGreg Clayton     }
361644d93782SGreg Clayton       return MenuActionResult::Handled;
361744d93782SGreg Clayton 
3618b9c1b51eSKate Stone     case eMenuID_ViewRegisters: {
361944d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
362044d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
362144d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
362244d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
362344d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
362444d93782SGreg Clayton 
3625b9c1b51eSKate Stone       if (registers_window_sp) {
3626b9c1b51eSKate Stone         if (variables_window_sp) {
362744d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
362844d93782SGreg Clayton 
3629b9c1b51eSKate Stone           // We have a variables window, so give all the area back to the
3630b9c1b51eSKate Stone           // variables window
3631b9c1b51eSKate Stone           variables_window_sp->Resize(variables_bounds.size.width +
3632b9c1b51eSKate Stone                                           registers_window_sp->GetWidth(),
363344d93782SGreg Clayton                                       variables_bounds.size.height);
3634b9c1b51eSKate Stone         } else {
363505097246SAdrian Prantl           // We have no variables window showing so give the bottom area back
363605097246SAdrian Prantl           // to the source view
363744d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3638b9c1b51eSKate Stone                                    source_bounds.size.height +
3639b9c1b51eSKate Stone                                        registers_window_sp->GetHeight());
364044d93782SGreg Clayton         }
364144d93782SGreg Clayton         main_window_sp->RemoveSubWindow(registers_window_sp.get());
3642b9c1b51eSKate Stone       } else {
364344d93782SGreg Clayton         Rect new_regs_rect;
3644b9c1b51eSKate Stone         if (variables_window_sp) {
364505097246SAdrian Prantl           // We have a variables window, split it into two columns where the
364605097246SAdrian Prantl           // left hand side will be the variables and the right hand side will
364705097246SAdrian Prantl           // be the registers
364844d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
364944d93782SGreg Clayton           Rect new_vars_rect;
3650b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_vars_rect,
3651b9c1b51eSKate Stone                                                    new_regs_rect);
365244d93782SGreg Clayton           variables_window_sp->SetBounds(new_vars_rect);
3653b9c1b51eSKate Stone         } else {
365444d93782SGreg Clayton           // No registers window, grab the bottom part of the source window
365544d93782SGreg Clayton           Rect new_source_rect;
3656b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3657b9c1b51eSKate Stone                                                   new_regs_rect);
365844d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
365944d93782SGreg Clayton         }
3660b9c1b51eSKate Stone         WindowSP new_window_sp =
3661b9c1b51eSKate Stone             main_window_sp->CreateSubWindow("Registers", new_regs_rect, false);
3662b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3663b9c1b51eSKate Stone             WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
366444d93782SGreg Clayton       }
366544d93782SGreg Clayton       touchwin(stdscr);
366644d93782SGreg Clayton     }
366744d93782SGreg Clayton       return MenuActionResult::Handled;
366844d93782SGreg Clayton 
366944d93782SGreg Clayton     case eMenuID_HelpGUIHelp:
36705fdb09bbSGreg Clayton       m_app.GetMainWindow()->CreateHelpSubwindow();
367144d93782SGreg Clayton       return MenuActionResult::Handled;
367244d93782SGreg Clayton 
367344d93782SGreg Clayton     default:
367444d93782SGreg Clayton       break;
367544d93782SGreg Clayton     }
367644d93782SGreg Clayton 
367744d93782SGreg Clayton     return MenuActionResult::NotHandled;
367844d93782SGreg Clayton   }
3679b9c1b51eSKate Stone 
368044d93782SGreg Clayton protected:
368144d93782SGreg Clayton   Application &m_app;
368244d93782SGreg Clayton   Debugger &m_debugger;
368344d93782SGreg Clayton };
368444d93782SGreg Clayton 
3685b9c1b51eSKate Stone class StatusBarWindowDelegate : public WindowDelegate {
368644d93782SGreg Clayton public:
3687b9c1b51eSKate Stone   StatusBarWindowDelegate(Debugger &debugger) : m_debugger(debugger) {
3688b9c1b51eSKate Stone     FormatEntity::Parse("Thread: ${thread.id%tid}", m_format);
368944d93782SGreg Clayton   }
369044d93782SGreg Clayton 
3691315b6884SEugene Zelenko   ~StatusBarWindowDelegate() override = default;
3692bd5ae6b4SGreg Clayton 
3693b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3694b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3695b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
369644d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
369744d93782SGreg Clayton     Thread *thread = exe_ctx.GetThreadPtr();
369844d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
369944d93782SGreg Clayton     window.Erase();
370044d93782SGreg Clayton     window.SetBackground(2);
370144d93782SGreg Clayton     window.MoveCursor(0, 0);
3702b9c1b51eSKate Stone     if (process) {
370344d93782SGreg Clayton       const StateType state = process->GetState();
3704b9c1b51eSKate Stone       window.Printf("Process: %5" PRIu64 " %10s", process->GetID(),
3705b9c1b51eSKate Stone                     StateAsCString(state));
370644d93782SGreg Clayton 
3707b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
37085b031ebcSEd Maste         StreamString strm;
3709b9c1b51eSKate Stone         if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx,
3710b9c1b51eSKate Stone                                            nullptr, nullptr, false, false)) {
371144d93782SGreg Clayton           window.MoveCursor(40, 0);
3712c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), 1);
37135b031ebcSEd Maste         }
371444d93782SGreg Clayton 
371544d93782SGreg Clayton         window.MoveCursor(60, 0);
371644d93782SGreg Clayton         if (frame)
3717b9c1b51eSKate Stone           window.Printf("Frame: %3u  PC = 0x%16.16" PRIx64,
3718b9c1b51eSKate Stone                         frame->GetFrameIndex(),
3719b9c1b51eSKate Stone                         frame->GetFrameCodeAddress().GetOpcodeLoadAddress(
3720b9c1b51eSKate Stone                             exe_ctx.GetTargetPtr()));
3721b9c1b51eSKate Stone       } else if (state == eStateExited) {
372244d93782SGreg Clayton         const char *exit_desc = process->GetExitDescription();
372344d93782SGreg Clayton         const int exit_status = process->GetExitStatus();
372444d93782SGreg Clayton         if (exit_desc && exit_desc[0])
372544d93782SGreg Clayton           window.Printf(" with status = %i (%s)", exit_status, exit_desc);
372644d93782SGreg Clayton         else
372744d93782SGreg Clayton           window.Printf(" with status = %i", exit_status);
372844d93782SGreg Clayton       }
372944d93782SGreg Clayton     }
373044d93782SGreg Clayton     return true;
373144d93782SGreg Clayton   }
373244d93782SGreg Clayton 
373344d93782SGreg Clayton protected:
373444d93782SGreg Clayton   Debugger &m_debugger;
3735554f68d3SGreg Clayton   FormatEntity::Entry m_format;
373644d93782SGreg Clayton };
373744d93782SGreg Clayton 
3738b9c1b51eSKate Stone class SourceFileWindowDelegate : public WindowDelegate {
373944d93782SGreg Clayton public:
3740b9c1b51eSKate Stone   SourceFileWindowDelegate(Debugger &debugger)
3741b9c1b51eSKate Stone       : WindowDelegate(), m_debugger(debugger), m_sc(), m_file_sp(),
3742b9c1b51eSKate Stone         m_disassembly_scope(nullptr), m_disassembly_sp(), m_disassembly_range(),
3743b9c1b51eSKate Stone         m_title(), m_line_width(4), m_selected_line(0), m_pc_line(0),
3744b9c1b51eSKate Stone         m_stop_id(0), m_frame_idx(UINT32_MAX), m_first_visible_line(0),
3745b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
374644d93782SGreg Clayton 
3747315b6884SEugene Zelenko   ~SourceFileWindowDelegate() override = default;
374844d93782SGreg Clayton 
3749b9c1b51eSKate Stone   void Update(const SymbolContext &sc) { m_sc = sc; }
375044d93782SGreg Clayton 
3751b9c1b51eSKate Stone   uint32_t NumVisibleLines() const { return m_max_y - m_min_y; }
375244d93782SGreg Clayton 
3753b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
375444d93782SGreg Clayton     return "Source/Disassembly window keyboard shortcuts:";
375544d93782SGreg Clayton   }
375644d93782SGreg Clayton 
3757b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
375844d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
375944d93782SGreg Clayton         {KEY_RETURN, "Run to selected line with one shot breakpoint"},
376044d93782SGreg Clayton         {KEY_UP, "Select previous source line"},
376144d93782SGreg Clayton         {KEY_DOWN, "Select next source line"},
376244d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
376344d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
376444d93782SGreg Clayton         {'b', "Set breakpoint on selected source/disassembly line"},
376544d93782SGreg Clayton         {'c', "Continue process"},
376644d93782SGreg Clayton         {'d', "Detach and resume process"},
376744d93782SGreg Clayton         {'D', "Detach with process suspended"},
376844d93782SGreg Clayton         {'h', "Show help dialog"},
376944d93782SGreg Clayton         {'k', "Kill process"},
377044d93782SGreg Clayton         {'n', "Step over (source line)"},
377144d93782SGreg Clayton         {'N', "Step over (single instruction)"},
377244d93782SGreg Clayton         {'o', "Step out"},
377344d93782SGreg Clayton         {'s', "Step in (source line)"},
377444d93782SGreg Clayton         {'S', "Step in (single instruction)"},
377544d93782SGreg Clayton         {',', "Page up"},
377644d93782SGreg Clayton         {'.', "Page down"},
3777b9c1b51eSKate Stone         {'\0', nullptr}};
377844d93782SGreg Clayton     return g_source_view_key_help;
377944d93782SGreg Clayton   }
378044d93782SGreg Clayton 
3781b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3782b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3783b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
378444d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3785c5dac77aSEugene Zelenko     Thread *thread = nullptr;
378644d93782SGreg Clayton 
378744d93782SGreg Clayton     bool update_location = false;
3788b9c1b51eSKate Stone     if (process) {
378944d93782SGreg Clayton       StateType state = process->GetState();
3790b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
379144d93782SGreg Clayton         // We are stopped, so it is ok to
379244d93782SGreg Clayton         update_location = true;
379344d93782SGreg Clayton       }
379444d93782SGreg Clayton     }
379544d93782SGreg Clayton 
379644d93782SGreg Clayton     m_min_x = 1;
3797ec990867SGreg Clayton     m_min_y = 2;
379844d93782SGreg Clayton     m_max_x = window.GetMaxX() - 1;
379944d93782SGreg Clayton     m_max_y = window.GetMaxY() - 1;
380044d93782SGreg Clayton 
380144d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
380244d93782SGreg Clayton     StackFrameSP frame_sp;
380344d93782SGreg Clayton     bool set_selected_line_to_pc = false;
380444d93782SGreg Clayton 
3805b9c1b51eSKate Stone     if (update_location) {
380644d93782SGreg Clayton       const bool process_alive = process ? process->IsAlive() : false;
380744d93782SGreg Clayton       bool thread_changed = false;
3808b9c1b51eSKate Stone       if (process_alive) {
380944d93782SGreg Clayton         thread = exe_ctx.GetThreadPtr();
3810b9c1b51eSKate Stone         if (thread) {
381144d93782SGreg Clayton           frame_sp = thread->GetSelectedFrame();
381244d93782SGreg Clayton           auto tid = thread->GetID();
381344d93782SGreg Clayton           thread_changed = tid != m_tid;
381444d93782SGreg Clayton           m_tid = tid;
3815b9c1b51eSKate Stone         } else {
3816b9c1b51eSKate Stone           if (m_tid != LLDB_INVALID_THREAD_ID) {
381744d93782SGreg Clayton             thread_changed = true;
381844d93782SGreg Clayton             m_tid = LLDB_INVALID_THREAD_ID;
381944d93782SGreg Clayton           }
382044d93782SGreg Clayton         }
382144d93782SGreg Clayton       }
382244d93782SGreg Clayton       const uint32_t stop_id = process ? process->GetStopID() : 0;
382344d93782SGreg Clayton       const bool stop_id_changed = stop_id != m_stop_id;
382444d93782SGreg Clayton       bool frame_changed = false;
382544d93782SGreg Clayton       m_stop_id = stop_id;
3826ec990867SGreg Clayton       m_title.Clear();
3827b9c1b51eSKate Stone       if (frame_sp) {
382844d93782SGreg Clayton         m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
3829b9c1b51eSKate Stone         if (m_sc.module_sp) {
3830b9c1b51eSKate Stone           m_title.Printf(
3831b9c1b51eSKate Stone               "%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
3832ec990867SGreg Clayton           ConstString func_name = m_sc.GetFunctionName();
3833ec990867SGreg Clayton           if (func_name)
3834ec990867SGreg Clayton             m_title.Printf("`%s", func_name.GetCString());
3835ec990867SGreg Clayton         }
383644d93782SGreg Clayton         const uint32_t frame_idx = frame_sp->GetFrameIndex();
383744d93782SGreg Clayton         frame_changed = frame_idx != m_frame_idx;
383844d93782SGreg Clayton         m_frame_idx = frame_idx;
3839b9c1b51eSKate Stone       } else {
384044d93782SGreg Clayton         m_sc.Clear(true);
384144d93782SGreg Clayton         frame_changed = m_frame_idx != UINT32_MAX;
384244d93782SGreg Clayton         m_frame_idx = UINT32_MAX;
384344d93782SGreg Clayton       }
384444d93782SGreg Clayton 
3845b9c1b51eSKate Stone       const bool context_changed =
3846b9c1b51eSKate Stone           thread_changed || frame_changed || stop_id_changed;
384744d93782SGreg Clayton 
3848b9c1b51eSKate Stone       if (process_alive) {
3849b9c1b51eSKate Stone         if (m_sc.line_entry.IsValid()) {
385044d93782SGreg Clayton           m_pc_line = m_sc.line_entry.line;
385144d93782SGreg Clayton           if (m_pc_line != UINT32_MAX)
385244d93782SGreg Clayton             --m_pc_line; // Convert to zero based line number...
385344d93782SGreg Clayton           // Update the selected line if the stop ID changed...
385444d93782SGreg Clayton           if (context_changed)
385544d93782SGreg Clayton             m_selected_line = m_pc_line;
385644d93782SGreg Clayton 
3857b9c1b51eSKate Stone           if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) {
385805097246SAdrian Prantl             // Same file, nothing to do, we should either have the lines or not
385905097246SAdrian Prantl             // (source file missing)
3860b9c1b51eSKate Stone             if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) {
386144d93782SGreg Clayton               if (m_selected_line >= m_first_visible_line + num_visible_lines)
386244d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
3863b9c1b51eSKate Stone             } else {
386444d93782SGreg Clayton               if (m_selected_line > 10)
386544d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
386644d93782SGreg Clayton               else
386744d93782SGreg Clayton                 m_first_visible_line = 0;
386844d93782SGreg Clayton             }
3869b9c1b51eSKate Stone           } else {
387044d93782SGreg Clayton             // File changed, set selected line to the line with the PC
387144d93782SGreg Clayton             m_selected_line = m_pc_line;
3872b9c1b51eSKate Stone             m_file_sp =
3873b9c1b51eSKate Stone                 m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
3874b9c1b51eSKate Stone             if (m_file_sp) {
387544d93782SGreg Clayton               const size_t num_lines = m_file_sp->GetNumLines();
3876eebf32faSPavel Labath               m_line_width = 1;
387744d93782SGreg Clayton               for (size_t n = num_lines; n >= 10; n = n / 10)
387844d93782SGreg Clayton                 ++m_line_width;
387944d93782SGreg Clayton 
3880b9c1b51eSKate Stone               if (num_lines < num_visible_lines ||
3881b9c1b51eSKate Stone                   m_selected_line < num_visible_lines)
388244d93782SGreg Clayton                 m_first_visible_line = 0;
388344d93782SGreg Clayton               else
388444d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
388544d93782SGreg Clayton             }
388644d93782SGreg Clayton           }
3887b9c1b51eSKate Stone         } else {
388844d93782SGreg Clayton           m_file_sp.reset();
388944d93782SGreg Clayton         }
389044d93782SGreg Clayton 
3891b9c1b51eSKate Stone         if (!m_file_sp || m_file_sp->GetNumLines() == 0) {
389244d93782SGreg Clayton           // Show disassembly
389344d93782SGreg Clayton           bool prefer_file_cache = false;
3894b9c1b51eSKate Stone           if (m_sc.function) {
3895b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.function) {
389644d93782SGreg Clayton               m_disassembly_scope = m_sc.function;
3897b9c1b51eSKate Stone               m_disassembly_sp = m_sc.function->GetInstructions(
3898b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3899b9c1b51eSKate Stone               if (m_disassembly_sp) {
390044d93782SGreg Clayton                 set_selected_line_to_pc = true;
390144d93782SGreg Clayton                 m_disassembly_range = m_sc.function->GetAddressRange();
3902b9c1b51eSKate Stone               } else {
390344d93782SGreg Clayton                 m_disassembly_range.Clear();
390444d93782SGreg Clayton               }
3905b9c1b51eSKate Stone             } else {
390644d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
390744d93782SGreg Clayton             }
3908b9c1b51eSKate Stone           } else if (m_sc.symbol) {
3909b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.symbol) {
391044d93782SGreg Clayton               m_disassembly_scope = m_sc.symbol;
3911b9c1b51eSKate Stone               m_disassembly_sp = m_sc.symbol->GetInstructions(
3912b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3913b9c1b51eSKate Stone               if (m_disassembly_sp) {
391444d93782SGreg Clayton                 set_selected_line_to_pc = true;
3915b9c1b51eSKate Stone                 m_disassembly_range.GetBaseAddress() =
3916b9c1b51eSKate Stone                     m_sc.symbol->GetAddress();
391744d93782SGreg Clayton                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
3918b9c1b51eSKate Stone               } else {
391944d93782SGreg Clayton                 m_disassembly_range.Clear();
392044d93782SGreg Clayton               }
3921b9c1b51eSKate Stone             } else {
392244d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
392344d93782SGreg Clayton             }
392444d93782SGreg Clayton           }
392544d93782SGreg Clayton         }
3926b9c1b51eSKate Stone       } else {
392744d93782SGreg Clayton         m_pc_line = UINT32_MAX;
392844d93782SGreg Clayton       }
392944d93782SGreg Clayton     }
393044d93782SGreg Clayton 
3931ec990867SGreg Clayton     const int window_width = window.GetWidth();
393244d93782SGreg Clayton     window.Erase();
393344d93782SGreg Clayton     window.DrawTitleBox("Sources");
3934b9c1b51eSKate Stone     if (!m_title.GetString().empty()) {
3935ec990867SGreg Clayton       window.AttributeOn(A_REVERSE);
3936ec990867SGreg Clayton       window.MoveCursor(1, 1);
3937ec990867SGreg Clayton       window.PutChar(' ');
3938c156427dSZachary Turner       window.PutCStringTruncated(m_title.GetString().str().c_str(), 1);
3939ec990867SGreg Clayton       int x = window.GetCursorX();
3940b9c1b51eSKate Stone       if (x < window_width - 1) {
3941ec990867SGreg Clayton         window.Printf("%*s", window_width - x - 1, "");
3942ec990867SGreg Clayton       }
3943ec990867SGreg Clayton       window.AttributeOff(A_REVERSE);
3944ec990867SGreg Clayton     }
394544d93782SGreg Clayton 
394644d93782SGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
394744d93782SGreg Clayton     const size_t num_source_lines = GetNumSourceLines();
3948b9c1b51eSKate Stone     if (num_source_lines > 0) {
394944d93782SGreg Clayton       // Display source
395044d93782SGreg Clayton       BreakpointLines bp_lines;
3951b9c1b51eSKate Stone       if (target) {
395244d93782SGreg Clayton         BreakpointList &bp_list = target->GetBreakpointList();
395344d93782SGreg Clayton         const size_t num_bps = bp_list.GetSize();
3954b9c1b51eSKate Stone         for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
395544d93782SGreg Clayton           BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
395644d93782SGreg Clayton           const size_t num_bps_locs = bp_sp->GetNumLocations();
3957b9c1b51eSKate Stone           for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs; ++bp_loc_idx) {
3958b9c1b51eSKate Stone             BreakpointLocationSP bp_loc_sp =
3959b9c1b51eSKate Stone                 bp_sp->GetLocationAtIndex(bp_loc_idx);
396044d93782SGreg Clayton             LineEntry bp_loc_line_entry;
3961b9c1b51eSKate Stone             if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry(
3962b9c1b51eSKate Stone                     bp_loc_line_entry)) {
3963b9c1b51eSKate Stone               if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) {
396444d93782SGreg Clayton                 bp_lines.insert(bp_loc_line_entry.line);
396544d93782SGreg Clayton               }
396644d93782SGreg Clayton             }
396744d93782SGreg Clayton           }
396844d93782SGreg Clayton         }
396944d93782SGreg Clayton       }
397044d93782SGreg Clayton 
397144d93782SGreg Clayton       const attr_t selected_highlight_attr = A_REVERSE;
397244d93782SGreg Clayton       const attr_t pc_highlight_attr = COLOR_PAIR(1);
397344d93782SGreg Clayton 
3974b9c1b51eSKate Stone       for (size_t i = 0; i < num_visible_lines; ++i) {
397544d93782SGreg Clayton         const uint32_t curr_line = m_first_visible_line + i;
3976b9c1b51eSKate Stone         if (curr_line < num_source_lines) {
3977ec990867SGreg Clayton           const int line_y = m_min_y + i;
397844d93782SGreg Clayton           window.MoveCursor(1, line_y);
397944d93782SGreg Clayton           const bool is_pc_line = curr_line == m_pc_line;
398044d93782SGreg Clayton           const bool line_is_selected = m_selected_line == curr_line;
398144d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
398244d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
398344d93782SGreg Clayton           attr_t highlight_attr = 0;
398444d93782SGreg Clayton           attr_t bp_attr = 0;
398544d93782SGreg Clayton           if (is_pc_line)
398644d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
398744d93782SGreg Clayton           else if (line_is_selected)
398844d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
398944d93782SGreg Clayton 
399044d93782SGreg Clayton           if (bp_lines.find(curr_line + 1) != bp_lines.end())
399144d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
399244d93782SGreg Clayton 
399344d93782SGreg Clayton           if (bp_attr)
399444d93782SGreg Clayton             window.AttributeOn(bp_attr);
399544d93782SGreg Clayton 
3996eebf32faSPavel Labath           window.Printf(" %*u ", m_line_width, curr_line + 1);
399744d93782SGreg Clayton 
399844d93782SGreg Clayton           if (bp_attr)
399944d93782SGreg Clayton             window.AttributeOff(bp_attr);
400044d93782SGreg Clayton 
400144d93782SGreg Clayton           window.PutChar(ACS_VLINE);
400244d93782SGreg Clayton           // Mark the line with the PC with a diamond
400344d93782SGreg Clayton           if (is_pc_line)
400444d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
400544d93782SGreg Clayton           else
400644d93782SGreg Clayton             window.PutChar(' ');
400744d93782SGreg Clayton 
400844d93782SGreg Clayton           if (highlight_attr)
400944d93782SGreg Clayton             window.AttributeOn(highlight_attr);
4010b9c1b51eSKate Stone           const uint32_t line_len =
4011b9c1b51eSKate Stone               m_file_sp->GetLineLength(curr_line + 1, false);
401244d93782SGreg Clayton           if (line_len > 0)
401344d93782SGreg Clayton             window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
401444d93782SGreg Clayton 
4015b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4016b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
401744d93782SGreg Clayton             StopInfoSP stop_info_sp;
401844d93782SGreg Clayton             if (thread)
401944d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4020b9c1b51eSKate Stone             if (stop_info_sp) {
402144d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4022b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
402344d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4024ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
402544d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4026b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4027b9c1b51eSKate Stone                 // line_y);
4028b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4029b9c1b51eSKate Stone                               stop_description);
403044d93782SGreg Clayton               }
4031b9c1b51eSKate Stone             } else {
4032ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
403344d93782SGreg Clayton             }
403444d93782SGreg Clayton           }
403544d93782SGreg Clayton           if (highlight_attr)
403644d93782SGreg Clayton             window.AttributeOff(highlight_attr);
4037b9c1b51eSKate Stone         } else {
403844d93782SGreg Clayton           break;
403944d93782SGreg Clayton         }
404044d93782SGreg Clayton       }
4041b9c1b51eSKate Stone     } else {
404244d93782SGreg Clayton       size_t num_disassembly_lines = GetNumDisassemblyLines();
4043b9c1b51eSKate Stone       if (num_disassembly_lines > 0) {
404444d93782SGreg Clayton         // Display disassembly
404544d93782SGreg Clayton         BreakpointAddrs bp_file_addrs;
404644d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
4047b9c1b51eSKate Stone         if (target) {
404844d93782SGreg Clayton           BreakpointList &bp_list = target->GetBreakpointList();
404944d93782SGreg Clayton           const size_t num_bps = bp_list.GetSize();
4050b9c1b51eSKate Stone           for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
405144d93782SGreg Clayton             BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
405244d93782SGreg Clayton             const size_t num_bps_locs = bp_sp->GetNumLocations();
4053b9c1b51eSKate Stone             for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs;
4054b9c1b51eSKate Stone                  ++bp_loc_idx) {
4055b9c1b51eSKate Stone               BreakpointLocationSP bp_loc_sp =
4056b9c1b51eSKate Stone                   bp_sp->GetLocationAtIndex(bp_loc_idx);
405744d93782SGreg Clayton               LineEntry bp_loc_line_entry;
4058b9c1b51eSKate Stone               const lldb::addr_t file_addr =
4059b9c1b51eSKate Stone                   bp_loc_sp->GetAddress().GetFileAddress();
4060b9c1b51eSKate Stone               if (file_addr != LLDB_INVALID_ADDRESS) {
406144d93782SGreg Clayton                 if (m_disassembly_range.ContainsFileAddress(file_addr))
406244d93782SGreg Clayton                   bp_file_addrs.insert(file_addr);
406344d93782SGreg Clayton               }
406444d93782SGreg Clayton             }
406544d93782SGreg Clayton           }
406644d93782SGreg Clayton         }
406744d93782SGreg Clayton 
406844d93782SGreg Clayton         const attr_t selected_highlight_attr = A_REVERSE;
406944d93782SGreg Clayton         const attr_t pc_highlight_attr = COLOR_PAIR(1);
407044d93782SGreg Clayton 
407144d93782SGreg Clayton         StreamString strm;
407244d93782SGreg Clayton 
407344d93782SGreg Clayton         InstructionList &insts = m_disassembly_sp->GetInstructionList();
407444d93782SGreg Clayton         Address pc_address;
407544d93782SGreg Clayton 
407644d93782SGreg Clayton         if (frame_sp)
407744d93782SGreg Clayton           pc_address = frame_sp->GetFrameCodeAddress();
4078b9c1b51eSKate Stone         const uint32_t pc_idx =
4079b9c1b51eSKate Stone             pc_address.IsValid()
4080b9c1b51eSKate Stone                 ? insts.GetIndexOfInstructionAtAddress(pc_address)
4081b9c1b51eSKate Stone                 : UINT32_MAX;
4082b9c1b51eSKate Stone         if (set_selected_line_to_pc) {
408344d93782SGreg Clayton           m_selected_line = pc_idx;
408444d93782SGreg Clayton         }
408544d93782SGreg Clayton 
408644d93782SGreg Clayton         const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
40873985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
408844d93782SGreg Clayton           m_first_visible_line = 0;
408944d93782SGreg Clayton 
4090b9c1b51eSKate Stone         if (pc_idx < num_disassembly_lines) {
40913985c8c6SSaleem Abdulrasool           if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
409244d93782SGreg Clayton               pc_idx >= m_first_visible_line + num_visible_lines)
409344d93782SGreg Clayton             m_first_visible_line = pc_idx - non_visible_pc_offset;
409444d93782SGreg Clayton         }
409544d93782SGreg Clayton 
4096b9c1b51eSKate Stone         for (size_t i = 0; i < num_visible_lines; ++i) {
409744d93782SGreg Clayton           const uint32_t inst_idx = m_first_visible_line + i;
409844d93782SGreg Clayton           Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
409944d93782SGreg Clayton           if (!inst)
410044d93782SGreg Clayton             break;
410144d93782SGreg Clayton 
4102ec990867SGreg Clayton           const int line_y = m_min_y + i;
4103ec990867SGreg Clayton           window.MoveCursor(1, line_y);
410444d93782SGreg Clayton           const bool is_pc_line = frame_sp && inst_idx == pc_idx;
410544d93782SGreg Clayton           const bool line_is_selected = m_selected_line == inst_idx;
410644d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
410744d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
410844d93782SGreg Clayton           attr_t highlight_attr = 0;
410944d93782SGreg Clayton           attr_t bp_attr = 0;
411044d93782SGreg Clayton           if (is_pc_line)
411144d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
411244d93782SGreg Clayton           else if (line_is_selected)
411344d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
411444d93782SGreg Clayton 
4115b9c1b51eSKate Stone           if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) !=
4116b9c1b51eSKate Stone               bp_file_addrs.end())
411744d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
411844d93782SGreg Clayton 
411944d93782SGreg Clayton           if (bp_attr)
412044d93782SGreg Clayton             window.AttributeOn(bp_attr);
412144d93782SGreg Clayton 
4122324a1036SSaleem Abdulrasool           window.Printf(" 0x%16.16llx ",
4123b9c1b51eSKate Stone                         static_cast<unsigned long long>(
4124b9c1b51eSKate Stone                             inst->GetAddress().GetLoadAddress(target)));
412544d93782SGreg Clayton 
412644d93782SGreg Clayton           if (bp_attr)
412744d93782SGreg Clayton             window.AttributeOff(bp_attr);
412844d93782SGreg Clayton 
412944d93782SGreg Clayton           window.PutChar(ACS_VLINE);
413044d93782SGreg Clayton           // Mark the line with the PC with a diamond
413144d93782SGreg Clayton           if (is_pc_line)
413244d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
413344d93782SGreg Clayton           else
413444d93782SGreg Clayton             window.PutChar(' ');
413544d93782SGreg Clayton 
413644d93782SGreg Clayton           if (highlight_attr)
413744d93782SGreg Clayton             window.AttributeOn(highlight_attr);
413844d93782SGreg Clayton 
413944d93782SGreg Clayton           const char *mnemonic = inst->GetMnemonic(&exe_ctx);
414044d93782SGreg Clayton           const char *operands = inst->GetOperands(&exe_ctx);
414144d93782SGreg Clayton           const char *comment = inst->GetComment(&exe_ctx);
414244d93782SGreg Clayton 
4143c5dac77aSEugene Zelenko           if (mnemonic != nullptr && mnemonic[0] == '\0')
4144c5dac77aSEugene Zelenko             mnemonic = nullptr;
4145c5dac77aSEugene Zelenko           if (operands != nullptr && operands[0] == '\0')
4146c5dac77aSEugene Zelenko             operands = nullptr;
4147c5dac77aSEugene Zelenko           if (comment != nullptr && comment[0] == '\0')
4148c5dac77aSEugene Zelenko             comment = nullptr;
414944d93782SGreg Clayton 
415044d93782SGreg Clayton           strm.Clear();
415144d93782SGreg Clayton 
4152c5dac77aSEugene Zelenko           if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
415344d93782SGreg Clayton             strm.Printf("%-8s %-25s ; %s", mnemonic, operands, comment);
4154c5dac77aSEugene Zelenko           else if (mnemonic != nullptr && operands != nullptr)
415544d93782SGreg Clayton             strm.Printf("%-8s %s", mnemonic, operands);
4156c5dac77aSEugene Zelenko           else if (mnemonic != nullptr)
415744d93782SGreg Clayton             strm.Printf("%s", mnemonic);
415844d93782SGreg Clayton 
415944d93782SGreg Clayton           int right_pad = 1;
4160c156427dSZachary Turner           window.PutCStringTruncated(strm.GetData(), right_pad);
416144d93782SGreg Clayton 
4162b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4163b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
416444d93782SGreg Clayton             StopInfoSP stop_info_sp;
416544d93782SGreg Clayton             if (thread)
416644d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4167b9c1b51eSKate Stone             if (stop_info_sp) {
416844d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4169b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
417044d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4171ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
417244d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4173b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4174b9c1b51eSKate Stone                 // line_y);
4175b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4176b9c1b51eSKate Stone                               stop_description);
417744d93782SGreg Clayton               }
4178b9c1b51eSKate Stone             } else {
4179ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
418044d93782SGreg Clayton             }
418144d93782SGreg Clayton           }
418244d93782SGreg Clayton           if (highlight_attr)
418344d93782SGreg Clayton             window.AttributeOff(highlight_attr);
418444d93782SGreg Clayton         }
418544d93782SGreg Clayton       }
418644d93782SGreg Clayton     }
418744d93782SGreg Clayton     return true; // Drawing handled
418844d93782SGreg Clayton   }
418944d93782SGreg Clayton 
4190b9c1b51eSKate Stone   size_t GetNumLines() {
419144d93782SGreg Clayton     size_t num_lines = GetNumSourceLines();
419244d93782SGreg Clayton     if (num_lines == 0)
419344d93782SGreg Clayton       num_lines = GetNumDisassemblyLines();
419444d93782SGreg Clayton     return num_lines;
419544d93782SGreg Clayton   }
419644d93782SGreg Clayton 
4197b9c1b51eSKate Stone   size_t GetNumSourceLines() const {
419844d93782SGreg Clayton     if (m_file_sp)
419944d93782SGreg Clayton       return m_file_sp->GetNumLines();
420044d93782SGreg Clayton     return 0;
420144d93782SGreg Clayton   }
4202315b6884SEugene Zelenko 
4203b9c1b51eSKate Stone   size_t GetNumDisassemblyLines() const {
420444d93782SGreg Clayton     if (m_disassembly_sp)
420544d93782SGreg Clayton       return m_disassembly_sp->GetInstructionList().GetSize();
420644d93782SGreg Clayton     return 0;
420744d93782SGreg Clayton   }
420844d93782SGreg Clayton 
4209b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
421044d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
421144d93782SGreg Clayton     const size_t num_lines = GetNumLines();
421244d93782SGreg Clayton 
4213b9c1b51eSKate Stone     switch (c) {
421444d93782SGreg Clayton     case ',':
421544d93782SGreg Clayton     case KEY_PPAGE:
421644d93782SGreg Clayton       // Page up key
42173985c8c6SSaleem Abdulrasool       if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
421844d93782SGreg Clayton         m_first_visible_line -= num_visible_lines;
421944d93782SGreg Clayton       else
422044d93782SGreg Clayton         m_first_visible_line = 0;
422144d93782SGreg Clayton       m_selected_line = m_first_visible_line;
422244d93782SGreg Clayton       return eKeyHandled;
422344d93782SGreg Clayton 
422444d93782SGreg Clayton     case '.':
422544d93782SGreg Clayton     case KEY_NPAGE:
422644d93782SGreg Clayton       // Page down key
422744d93782SGreg Clayton       {
422844d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < num_lines)
422944d93782SGreg Clayton           m_first_visible_line += num_visible_lines;
423044d93782SGreg Clayton         else if (num_lines < num_visible_lines)
423144d93782SGreg Clayton           m_first_visible_line = 0;
423244d93782SGreg Clayton         else
423344d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
423444d93782SGreg Clayton         m_selected_line = m_first_visible_line;
423544d93782SGreg Clayton       }
423644d93782SGreg Clayton       return eKeyHandled;
423744d93782SGreg Clayton 
423844d93782SGreg Clayton     case KEY_UP:
4239b9c1b51eSKate Stone       if (m_selected_line > 0) {
424044d93782SGreg Clayton         m_selected_line--;
42413985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
424244d93782SGreg Clayton           m_first_visible_line = m_selected_line;
424344d93782SGreg Clayton       }
424444d93782SGreg Clayton       return eKeyHandled;
424544d93782SGreg Clayton 
424644d93782SGreg Clayton     case KEY_DOWN:
4247b9c1b51eSKate Stone       if (m_selected_line + 1 < num_lines) {
424844d93782SGreg Clayton         m_selected_line++;
424944d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < m_selected_line)
425044d93782SGreg Clayton           m_first_visible_line++;
425144d93782SGreg Clayton       }
425244d93782SGreg Clayton       return eKeyHandled;
425344d93782SGreg Clayton 
425444d93782SGreg Clayton     case '\r':
425544d93782SGreg Clayton     case '\n':
425644d93782SGreg Clayton     case KEY_ENTER:
425744d93782SGreg Clayton       // Set a breakpoint and run to the line using a one shot breakpoint
4258b9c1b51eSKate Stone       if (GetNumSourceLines() > 0) {
4259b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4260b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4261b9c1b51eSKate Stone         if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) {
4262b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4263b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
426444d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4265b9c1b51eSKate Stone               m_selected_line +
4266b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4267431b1584SAdrian Prantl               0,     // Unspecified column.
42682411167fSJim Ingham               0,     // No offset
426944d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
427044d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
427144d93782SGreg Clayton               false,               // internal
4272055ad9beSIlia K               false,               // request_hardware
4273055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
427444d93782SGreg Clayton           // Make breakpoint one shot
427544d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
427644d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
427744d93782SGreg Clayton         }
4278b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4279b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4280b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4281b9c1b51eSKate Stone                                       .get();
4282b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4283b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4284b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
428544d93782SGreg Clayton           Address addr = inst->GetAddress();
4286b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4287b9c1b51eSKate Stone               addr,   // lldb_private::Address
428844d93782SGreg Clayton               false,  // internal
428944d93782SGreg Clayton               false); // request_hardware
429044d93782SGreg Clayton           // Make breakpoint one shot
429144d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
429244d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
429344d93782SGreg Clayton         }
429444d93782SGreg Clayton       }
429544d93782SGreg Clayton       return eKeyHandled;
429644d93782SGreg Clayton 
429744d93782SGreg Clayton     case 'b': // 'b' == toggle breakpoint on currently selected line
4298b9c1b51eSKate Stone       if (m_selected_line < GetNumSourceLines()) {
4299b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4300b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4301b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
4302b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4303b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
430444d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4305b9c1b51eSKate Stone               m_selected_line +
4306b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4307431b1584SAdrian Prantl               0,     // No column specified.
43082411167fSJim Ingham               0,     // No offset
430944d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
431044d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
431144d93782SGreg Clayton               false,               // internal
4312055ad9beSIlia K               false,               // request_hardware
4313055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
431444d93782SGreg Clayton         }
4315b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4316b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4317b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4318b9c1b51eSKate Stone                                       .get();
4319b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4320b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4321b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
432244d93782SGreg Clayton           Address addr = inst->GetAddress();
4323b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4324b9c1b51eSKate Stone               addr,   // lldb_private::Address
432544d93782SGreg Clayton               false,  // internal
432644d93782SGreg Clayton               false); // request_hardware
432744d93782SGreg Clayton         }
432844d93782SGreg Clayton       }
432944d93782SGreg Clayton       return eKeyHandled;
433044d93782SGreg Clayton 
433144d93782SGreg Clayton     case 'd': // 'd' == detach and let run
433244d93782SGreg Clayton     case 'D': // 'D' == detach and keep stopped
433344d93782SGreg Clayton     {
4334b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4335b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
433644d93782SGreg Clayton       if (exe_ctx.HasProcessScope())
433744d93782SGreg Clayton         exe_ctx.GetProcessRef().Detach(c == 'D');
433844d93782SGreg Clayton     }
433944d93782SGreg Clayton       return eKeyHandled;
434044d93782SGreg Clayton 
434144d93782SGreg Clayton     case 'k':
434244d93782SGreg Clayton       // 'k' == kill
434344d93782SGreg Clayton       {
4344b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4345b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
434644d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
4347ede3193bSJason Molenda           exe_ctx.GetProcessRef().Destroy(false);
434844d93782SGreg Clayton       }
434944d93782SGreg Clayton       return eKeyHandled;
435044d93782SGreg Clayton 
435144d93782SGreg Clayton     case 'c':
435244d93782SGreg Clayton       // 'c' == continue
435344d93782SGreg Clayton       {
4354b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4355b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
435644d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
435744d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
435844d93782SGreg Clayton       }
435944d93782SGreg Clayton       return eKeyHandled;
436044d93782SGreg Clayton 
436144d93782SGreg Clayton     case 'o':
436244d93782SGreg Clayton       // 'o' == step out
436344d93782SGreg Clayton       {
4364b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4365b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4366b9c1b51eSKate Stone         if (exe_ctx.HasThreadScope() &&
4367b9c1b51eSKate Stone             StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
436844d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
436944d93782SGreg Clayton         }
437044d93782SGreg Clayton       }
437144d93782SGreg Clayton       return eKeyHandled;
4372315b6884SEugene Zelenko 
437344d93782SGreg Clayton     case 'n': // 'n' == step over
437444d93782SGreg Clayton     case 'N': // 'N' == step over instruction
437544d93782SGreg Clayton     {
4376b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4377b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4378b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4379b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
438044d93782SGreg Clayton         bool source_step = (c == 'n');
438144d93782SGreg Clayton         exe_ctx.GetThreadRef().StepOver(source_step);
438244d93782SGreg Clayton       }
438344d93782SGreg Clayton     }
438444d93782SGreg Clayton       return eKeyHandled;
4385315b6884SEugene Zelenko 
438644d93782SGreg Clayton     case 's': // 's' == step into
438744d93782SGreg Clayton     case 'S': // 'S' == step into instruction
438844d93782SGreg Clayton     {
4389b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4390b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4391b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4392b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
439344d93782SGreg Clayton         bool source_step = (c == 's');
43944b4b2478SJim Ingham         exe_ctx.GetThreadRef().StepIn(source_step);
439544d93782SGreg Clayton       }
439644d93782SGreg Clayton     }
439744d93782SGreg Clayton       return eKeyHandled;
439844d93782SGreg Clayton 
439944d93782SGreg Clayton     case 'h':
440044d93782SGreg Clayton       window.CreateHelpSubwindow();
440144d93782SGreg Clayton       return eKeyHandled;
440244d93782SGreg Clayton 
440344d93782SGreg Clayton     default:
440444d93782SGreg Clayton       break;
440544d93782SGreg Clayton     }
440644d93782SGreg Clayton     return eKeyNotHandled;
440744d93782SGreg Clayton   }
440844d93782SGreg Clayton 
440944d93782SGreg Clayton protected:
441044d93782SGreg Clayton   typedef std::set<uint32_t> BreakpointLines;
441144d93782SGreg Clayton   typedef std::set<lldb::addr_t> BreakpointAddrs;
441244d93782SGreg Clayton 
441344d93782SGreg Clayton   Debugger &m_debugger;
441444d93782SGreg Clayton   SymbolContext m_sc;
441544d93782SGreg Clayton   SourceManager::FileSP m_file_sp;
441644d93782SGreg Clayton   SymbolContextScope *m_disassembly_scope;
441744d93782SGreg Clayton   lldb::DisassemblerSP m_disassembly_sp;
441844d93782SGreg Clayton   AddressRange m_disassembly_range;
4419ec990867SGreg Clayton   StreamString m_title;
442044d93782SGreg Clayton   lldb::user_id_t m_tid;
442144d93782SGreg Clayton   int m_line_width;
442244d93782SGreg Clayton   uint32_t m_selected_line; // The selected line
442344d93782SGreg Clayton   uint32_t m_pc_line;       // The line with the PC
442444d93782SGreg Clayton   uint32_t m_stop_id;
442544d93782SGreg Clayton   uint32_t m_frame_idx;
442644d93782SGreg Clayton   int m_first_visible_line;
442744d93782SGreg Clayton   int m_min_x;
442844d93782SGreg Clayton   int m_min_y;
442944d93782SGreg Clayton   int m_max_x;
443044d93782SGreg Clayton   int m_max_y;
443144d93782SGreg Clayton };
443244d93782SGreg Clayton 
443344d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = {true};
443444d93782SGreg Clayton 
4435b9c1b51eSKate Stone IOHandlerCursesGUI::IOHandlerCursesGUI(Debugger &debugger)
4436b9c1b51eSKate Stone     : IOHandler(debugger, IOHandler::Type::Curses) {}
443744d93782SGreg Clayton 
4438b9c1b51eSKate Stone void IOHandlerCursesGUI::Activate() {
443944d93782SGreg Clayton   IOHandler::Activate();
4440b9c1b51eSKate Stone   if (!m_app_ap) {
444144d93782SGreg Clayton     m_app_ap.reset(new Application(GetInputFILE(), GetOutputFILE()));
444244d93782SGreg Clayton 
444344d93782SGreg Clayton     // This is both a window and a menu delegate
4444b9c1b51eSKate Stone     std::shared_ptr<ApplicationDelegate> app_delegate_sp(
4445b9c1b51eSKate Stone         new ApplicationDelegate(*m_app_ap, m_debugger));
444644d93782SGreg Clayton 
4447b9c1b51eSKate Stone     MenuDelegateSP app_menu_delegate_sp =
4448b9c1b51eSKate Stone         std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
4449b9c1b51eSKate Stone     MenuSP lldb_menu_sp(
4450b9c1b51eSKate Stone         new Menu("LLDB", "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
4451b9c1b51eSKate Stone     MenuSP exit_menuitem_sp(
4452b9c1b51eSKate Stone         new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
445344d93782SGreg Clayton     exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
4454b9c1b51eSKate Stone     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(
4455b9c1b51eSKate Stone         "About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
445644d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
445744d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(exit_menuitem_sp);
445844d93782SGreg Clayton 
4459b9c1b51eSKate Stone     MenuSP target_menu_sp(new Menu("Target", "F2", KEY_F(2),
4460b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Target));
4461b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4462b9c1b51eSKate Stone         "Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
4463b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4464b9c1b51eSKate Stone         "Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
446544d93782SGreg Clayton 
4466b9c1b51eSKate Stone     MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3),
4467b9c1b51eSKate Stone                                     ApplicationDelegate::eMenuID_Process));
4468b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4469b9c1b51eSKate Stone         "Attach", nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
4470b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4471b9c1b51eSKate Stone         "Detach", nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
4472b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4473b9c1b51eSKate Stone         "Launch", nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
447444d93782SGreg Clayton     process_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
4475b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(
4476b9c1b51eSKate Stone         MenuSP(new Menu("Continue", nullptr, 'c',
4477b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ProcessContinue)));
4478b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4479b9c1b51eSKate Stone         "Halt", nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
4480b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4481b9c1b51eSKate Stone         "Kill", nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
448244d93782SGreg Clayton 
4483b9c1b51eSKate Stone     MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4),
4484b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Thread));
4485b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4486b9c1b51eSKate Stone         "Step In", nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
4487b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(
4488b9c1b51eSKate Stone         MenuSP(new Menu("Step Over", nullptr, 'v',
4489b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ThreadStepOver)));
4490b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4491b9c1b51eSKate Stone         "Step Out", nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
449244d93782SGreg Clayton 
4493b9c1b51eSKate Stone     MenuSP view_menu_sp(
4494b9c1b51eSKate Stone         new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
4495b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4496b9c1b51eSKate Stone         MenuSP(new Menu("Backtrace", nullptr, 'b',
4497b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewBacktrace)));
4498b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4499b9c1b51eSKate Stone         MenuSP(new Menu("Registers", nullptr, 'r',
4500b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewRegisters)));
4501b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(MenuSP(new Menu(
4502b9c1b51eSKate Stone         "Source", nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
4503b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4504b9c1b51eSKate Stone         MenuSP(new Menu("Variables", nullptr, 'v',
4505b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewVariables)));
450644d93782SGreg Clayton 
4507b9c1b51eSKate Stone     MenuSP help_menu_sp(
4508b9c1b51eSKate Stone         new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
4509b9c1b51eSKate Stone     help_menu_sp->AddSubmenu(MenuSP(new Menu(
4510b9c1b51eSKate Stone         "GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
451144d93782SGreg Clayton 
451244d93782SGreg Clayton     m_app_ap->Initialize();
451344d93782SGreg Clayton     WindowSP &main_window_sp = m_app_ap->GetMainWindow();
451444d93782SGreg Clayton 
451544d93782SGreg Clayton     MenuSP menubar_sp(new Menu(Menu::Type::Bar));
451644d93782SGreg Clayton     menubar_sp->AddSubmenu(lldb_menu_sp);
451744d93782SGreg Clayton     menubar_sp->AddSubmenu(target_menu_sp);
451844d93782SGreg Clayton     menubar_sp->AddSubmenu(process_menu_sp);
451944d93782SGreg Clayton     menubar_sp->AddSubmenu(thread_menu_sp);
452044d93782SGreg Clayton     menubar_sp->AddSubmenu(view_menu_sp);
452144d93782SGreg Clayton     menubar_sp->AddSubmenu(help_menu_sp);
452244d93782SGreg Clayton     menubar_sp->SetDelegate(app_menu_delegate_sp);
452344d93782SGreg Clayton 
452444d93782SGreg Clayton     Rect content_bounds = main_window_sp->GetFrame();
452544d93782SGreg Clayton     Rect menubar_bounds = content_bounds.MakeMenuBar();
452644d93782SGreg Clayton     Rect status_bounds = content_bounds.MakeStatusBar();
452744d93782SGreg Clayton     Rect source_bounds;
452844d93782SGreg Clayton     Rect variables_bounds;
452944d93782SGreg Clayton     Rect threads_bounds;
453044d93782SGreg Clayton     Rect source_variables_bounds;
4531b9c1b51eSKate Stone     content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds,
4532b9c1b51eSKate Stone                                            threads_bounds);
4533b9c1b51eSKate Stone     source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds,
4534b9c1b51eSKate Stone                                                       variables_bounds);
453544d93782SGreg Clayton 
4536b9c1b51eSKate Stone     WindowSP menubar_window_sp =
4537b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
453805097246SAdrian Prantl     // Let the menubar get keys if the active window doesn't handle the keys
453905097246SAdrian Prantl     // that are typed so it can respond to menubar key presses.
4540b9c1b51eSKate Stone     menubar_window_sp->SetCanBeActive(
4541b9c1b51eSKate Stone         false); // Don't let the menubar become the active window
454244d93782SGreg Clayton     menubar_window_sp->SetDelegate(menubar_sp);
454344d93782SGreg Clayton 
4544b9c1b51eSKate Stone     WindowSP source_window_sp(
4545b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Source", source_bounds, true));
4546b9c1b51eSKate Stone     WindowSP variables_window_sp(
4547b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Variables", variables_bounds, false));
4548b9c1b51eSKate Stone     WindowSP threads_window_sp(
4549b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Threads", threads_bounds, false));
4550b9c1b51eSKate Stone     WindowSP status_window_sp(
45516bb7e21fSPavel Labath         main_window_sp->CreateSubWindow("Status", status_bounds, false));
4552b9c1b51eSKate Stone     status_window_sp->SetCanBeActive(
4553b9c1b51eSKate Stone         false); // Don't let the status bar become the active window
4554b9c1b51eSKate Stone     main_window_sp->SetDelegate(
4555b9c1b51eSKate Stone         std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
4556b9c1b51eSKate Stone     source_window_sp->SetDelegate(
4557b9c1b51eSKate Stone         WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
4558b9c1b51eSKate Stone     variables_window_sp->SetDelegate(
4559b9c1b51eSKate Stone         WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
4560ec990867SGreg Clayton     TreeDelegateSP thread_delegate_sp(new ThreadsTreeDelegate(m_debugger));
4561b9c1b51eSKate Stone     threads_window_sp->SetDelegate(WindowDelegateSP(
4562b9c1b51eSKate Stone         new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
4563b9c1b51eSKate Stone     status_window_sp->SetDelegate(
4564b9c1b51eSKate Stone         WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
456544d93782SGreg Clayton 
45665fdb09bbSGreg Clayton     // Show the main help window once the first time the curses GUI is launched
45675fdb09bbSGreg Clayton     static bool g_showed_help = false;
4568b9c1b51eSKate Stone     if (!g_showed_help) {
45695fdb09bbSGreg Clayton       g_showed_help = true;
45705fdb09bbSGreg Clayton       main_window_sp->CreateHelpSubwindow();
45715fdb09bbSGreg Clayton     }
45725fdb09bbSGreg Clayton 
457344d93782SGreg Clayton     init_pair(1, COLOR_WHITE, COLOR_BLUE);
457444d93782SGreg Clayton     init_pair(2, COLOR_BLACK, COLOR_WHITE);
457544d93782SGreg Clayton     init_pair(3, COLOR_MAGENTA, COLOR_WHITE);
457644d93782SGreg Clayton     init_pair(4, COLOR_MAGENTA, COLOR_BLACK);
457744d93782SGreg Clayton     init_pair(5, COLOR_RED, COLOR_BLACK);
457844d93782SGreg Clayton   }
457944d93782SGreg Clayton }
458044d93782SGreg Clayton 
4581b9c1b51eSKate Stone void IOHandlerCursesGUI::Deactivate() { m_app_ap->Terminate(); }
458244d93782SGreg Clayton 
4583b9c1b51eSKate Stone void IOHandlerCursesGUI::Run() {
458444d93782SGreg Clayton   m_app_ap->Run(m_debugger);
458544d93782SGreg Clayton   SetIsDone(true);
458644d93782SGreg Clayton }
458744d93782SGreg Clayton 
4588315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
458944d93782SGreg Clayton 
4590b9c1b51eSKate Stone void IOHandlerCursesGUI::Cancel() {}
459144d93782SGreg Clayton 
4592b9c1b51eSKate Stone bool IOHandlerCursesGUI::Interrupt() { return false; }
459344d93782SGreg Clayton 
4594b9c1b51eSKate Stone void IOHandlerCursesGUI::GotEOF() {}
459544d93782SGreg Clayton 
4596315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
4597