144d93782SGreg Clayton //===-- IOHandler.cpp -------------------------------------------*- C++ -*-===//
244d93782SGreg Clayton //
344d93782SGreg Clayton //                     The LLVM Compiler Infrastructure
444d93782SGreg Clayton //
544d93782SGreg Clayton // This file is distributed under the University of Illinois Open Source
644d93782SGreg Clayton // License. See LICENSE.TXT for details.
744d93782SGreg Clayton //
844d93782SGreg Clayton //===----------------------------------------------------------------------===//
944d93782SGreg Clayton 
102f3df613SZachary Turner #include "lldb/Core/IOHandler.h"
112f3df613SZachary Turner 
12315b6884SEugene Zelenko // C Includes
13315b6884SEugene Zelenko #ifndef LLDB_DISABLE_CURSES
1427801f4fSBruce Mitchener #include <curses.h>
15315b6884SEugene Zelenko #include <panel.h>
16315b6884SEugene Zelenko #endif
1744d93782SGreg Clayton 
18315b6884SEugene Zelenko // C++ Includes
197c9aa073STodd Fiala #if defined(__APPLE__)
207c9aa073STodd Fiala #include <deque>
217c9aa073STodd Fiala #endif
2244d93782SGreg Clayton #include <string>
2344d93782SGreg Clayton 
24315b6884SEugene Zelenko // Other libraries and framework includes
25315b6884SEugene Zelenko // Project includes
2644d93782SGreg Clayton #include "lldb/Core/Debugger.h"
2744d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
282f3df613SZachary Turner #include "lldb/Host/File.h"            // for File
292f3df613SZachary Turner #include "lldb/Host/Predicate.h"       // for Predicate, ::eBroad...
3097206d57SZachary Turner #include "lldb/Utility/Status.h"       // for Status
312f3df613SZachary Turner #include "lldb/Utility/StreamString.h" // for StreamString
322f3df613SZachary Turner #include "lldb/Utility/StringList.h"   // for StringList
332f3df613SZachary Turner #include "lldb/lldb-forward.h"         // for StreamFileSP
342f3df613SZachary Turner 
35cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
3644d93782SGreg Clayton #include "lldb/Host/Editline.h"
37cacde7dfSTodd Fiala #endif
3844d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
3944d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
402f3df613SZachary Turner #ifndef LLDB_DISABLE_CURSES
412f3df613SZachary Turner #include "lldb/Breakpoint/BreakpointLocation.h"
422f3df613SZachary Turner #include "lldb/Core/Module.h"
432f3df613SZachary Turner #include "lldb/Core/State.h"
442f3df613SZachary Turner #include "lldb/Core/ValueObject.h"
452f3df613SZachary Turner #include "lldb/Core/ValueObjectRegister.h"
4644d93782SGreg Clayton #include "lldb/Symbol/Block.h"
4744d93782SGreg Clayton #include "lldb/Symbol/Function.h"
4844d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
49c5dac77aSEugene Zelenko #include "lldb/Symbol/VariableList.h"
50c5dac77aSEugene Zelenko #include "lldb/Target/Process.h"
512f3df613SZachary Turner #include "lldb/Target/RegisterContext.h"
52c5dac77aSEugene Zelenko #include "lldb/Target/StackFrame.h"
532f3df613SZachary Turner #include "lldb/Target/StopInfo.h"
54b9c1b51eSKate Stone #include "lldb/Target/Target.h"
55b9c1b51eSKate Stone #include "lldb/Target/Thread.h"
56c5dac77aSEugene Zelenko #endif
577c9aa073STodd Fiala 
582f3df613SZachary Turner #include "llvm/ADT/StringRef.h" // for StringRef
592f3df613SZachary Turner 
60fab31220STed Woodward #ifdef _MSC_VER
61aaea8ee6SZachary Turner #include "lldb/Host/windows/windows.h"
62fab31220STed Woodward #endif
63fab31220STed Woodward 
642f3df613SZachary Turner #include <memory> // for shared_ptr
652f3df613SZachary Turner #include <mutex>  // for recursive_mutex
662f3df613SZachary Turner 
672f3df613SZachary Turner #include <assert.h>    // for assert
682f3df613SZachary Turner #include <ctype.h>     // for isspace
692f3df613SZachary Turner #include <errno.h>     // for EINTR, errno
70f9841cecSPavel Labath #include <locale.h>    // for setlocale
712f3df613SZachary Turner #include <stdint.h>    // for uint32_t, UINT32_MAX
722f3df613SZachary Turner #include <stdio.h>     // for size_t, fprintf, feof
732f3df613SZachary Turner #include <string.h>    // for strlen
742f3df613SZachary Turner #include <type_traits> // for move
752f3df613SZachary Turner 
7644d93782SGreg Clayton using namespace lldb;
7744d93782SGreg Clayton using namespace lldb_private;
7844d93782SGreg Clayton 
79b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type)
80b9c1b51eSKate Stone     : IOHandler(debugger, type,
8144d93782SGreg Clayton                 StreamFileSP(), // Adopt STDIN from top input reader
8244d93782SGreg Clayton                 StreamFileSP(), // Adopt STDOUT from top input reader
83340b0309SGreg Clayton                 StreamFileSP(), // Adopt STDERR from top input reader
84340b0309SGreg Clayton                 0)              // Flags
85b9c1b51eSKate Stone {}
8644d93782SGreg Clayton 
87b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type,
8844d93782SGreg Clayton                      const lldb::StreamFileSP &input_sp,
8944d93782SGreg Clayton                      const lldb::StreamFileSP &output_sp,
90b9c1b51eSKate Stone                      const lldb::StreamFileSP &error_sp, uint32_t flags)
91b9c1b51eSKate Stone     : m_debugger(debugger), m_input_sp(input_sp), m_output_sp(output_sp),
92b9c1b51eSKate Stone       m_error_sp(error_sp), m_popped(false), m_flags(flags), m_type(type),
93b9c1b51eSKate Stone       m_user_data(nullptr), m_done(false), m_active(false) {
9444d93782SGreg Clayton   // If any files are not specified, then adopt them from the top input reader.
9544d93782SGreg Clayton   if (!m_input_sp || !m_output_sp || !m_error_sp)
96b9c1b51eSKate Stone     debugger.AdoptTopIOHandlerFilesIfInvalid(m_input_sp, m_output_sp,
9744d93782SGreg Clayton                                              m_error_sp);
9844d93782SGreg Clayton }
9944d93782SGreg Clayton 
100315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
10144d93782SGreg Clayton 
102b9c1b51eSKate Stone int IOHandler::GetInputFD() {
103c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetDescriptor() : -1);
10444d93782SGreg Clayton }
10544d93782SGreg Clayton 
106b9c1b51eSKate Stone int IOHandler::GetOutputFD() {
107c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetDescriptor() : -1);
10844d93782SGreg Clayton }
10944d93782SGreg Clayton 
110b9c1b51eSKate Stone int IOHandler::GetErrorFD() {
111c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetDescriptor() : -1);
11244d93782SGreg Clayton }
11344d93782SGreg Clayton 
114b9c1b51eSKate Stone FILE *IOHandler::GetInputFILE() {
115c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetStream() : nullptr);
11644d93782SGreg Clayton }
11744d93782SGreg Clayton 
118b9c1b51eSKate Stone FILE *IOHandler::GetOutputFILE() {
119c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetStream() : nullptr);
12044d93782SGreg Clayton }
12144d93782SGreg Clayton 
122b9c1b51eSKate Stone FILE *IOHandler::GetErrorFILE() {
123c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetStream() : nullptr);
12444d93782SGreg Clayton }
12544d93782SGreg Clayton 
126b9c1b51eSKate Stone StreamFileSP &IOHandler::GetInputStreamFile() { return m_input_sp; }
12744d93782SGreg Clayton 
128b9c1b51eSKate Stone StreamFileSP &IOHandler::GetOutputStreamFile() { return m_output_sp; }
12944d93782SGreg Clayton 
130b9c1b51eSKate Stone StreamFileSP &IOHandler::GetErrorStreamFile() { return m_error_sp; }
13144d93782SGreg Clayton 
132b9c1b51eSKate Stone bool IOHandler::GetIsInteractive() {
133340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsInteractive();
134340b0309SGreg Clayton }
135340b0309SGreg Clayton 
136b9c1b51eSKate Stone bool IOHandler::GetIsRealTerminal() {
137340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsRealTerminal();
138340b0309SGreg Clayton }
13944d93782SGreg Clayton 
140b9c1b51eSKate Stone void IOHandler::SetPopped(bool b) { m_popped.SetValue(b, eBroadcastOnChange); }
141e30f11d9SKate Stone 
142b9c1b51eSKate Stone void IOHandler::WaitForPop() { m_popped.WaitForValueEqualTo(true); }
143e30f11d9SKate Stone 
144b9c1b51eSKate Stone void IOHandlerStack::PrintAsync(Stream *stream, const char *s, size_t len) {
145b9c1b51eSKate Stone   if (stream) {
14616ff8604SSaleem Abdulrasool     std::lock_guard<std::recursive_mutex> guard(m_mutex);
1474446487dSPavel Labath     if (m_top)
1484446487dSPavel Labath       m_top->PrintAsync(stream, s, len);
1494446487dSPavel Labath   }
1504446487dSPavel Labath }
1514446487dSPavel Labath 
1527a120c8bSZachary Turner IOHandlerConfirm::IOHandlerConfirm(Debugger &debugger, llvm::StringRef prompt,
153b9c1b51eSKate Stone                                    bool default_response)
154b9c1b51eSKate Stone     : IOHandlerEditline(
155b9c1b51eSKate Stone           debugger, IOHandler::Type::Confirm,
156c5dac77aSEugene Zelenko           nullptr, // nullptr editline_name means no history loaded/saved
157514d8cd8SZachary Turner           llvm::StringRef(), // No prompt
158514d8cd8SZachary Turner           llvm::StringRef(), // No continuation prompt
15944d93782SGreg Clayton           false,             // Multi-line
160e30f11d9SKate Stone           false, // Don't colorize the prompt (i.e. the confirm message.)
161b9c1b51eSKate Stone           0, *this),
162b9c1b51eSKate Stone       m_default_response(default_response), m_user_response(default_response) {
16344d93782SGreg Clayton   StreamString prompt_stream;
16444d93782SGreg Clayton   prompt_stream.PutCString(prompt);
16544d93782SGreg Clayton   if (m_default_response)
16644d93782SGreg Clayton     prompt_stream.Printf(": [Y/n] ");
16744d93782SGreg Clayton   else
16844d93782SGreg Clayton     prompt_stream.Printf(": [y/N] ");
16944d93782SGreg Clayton 
170514d8cd8SZachary Turner   SetPrompt(prompt_stream.GetString());
17144d93782SGreg Clayton }
17244d93782SGreg Clayton 
173315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
17444d93782SGreg Clayton 
175b9c1b51eSKate Stone int IOHandlerConfirm::IOHandlerComplete(IOHandler &io_handler,
17644d93782SGreg Clayton                                         const char *current_line,
17744d93782SGreg Clayton                                         const char *cursor,
17844d93782SGreg Clayton                                         const char *last_char,
17944d93782SGreg Clayton                                         int skip_first_n_matches,
180b9c1b51eSKate Stone                                         int max_matches, StringList &matches) {
181b9c1b51eSKate Stone   if (current_line == cursor) {
182b9c1b51eSKate Stone     if (m_default_response) {
18344d93782SGreg Clayton       matches.AppendString("y");
184b9c1b51eSKate Stone     } else {
18544d93782SGreg Clayton       matches.AppendString("n");
18644d93782SGreg Clayton     }
18744d93782SGreg Clayton   }
18844d93782SGreg Clayton   return matches.GetSize();
18944d93782SGreg Clayton }
19044d93782SGreg Clayton 
191b9c1b51eSKate Stone void IOHandlerConfirm::IOHandlerInputComplete(IOHandler &io_handler,
192b9c1b51eSKate Stone                                               std::string &line) {
193b9c1b51eSKate Stone   if (line.empty()) {
19444d93782SGreg Clayton     // User just hit enter, set the response to the default
19544d93782SGreg Clayton     m_user_response = m_default_response;
19644d93782SGreg Clayton     io_handler.SetIsDone(true);
19744d93782SGreg Clayton     return;
19844d93782SGreg Clayton   }
19944d93782SGreg Clayton 
200b9c1b51eSKate Stone   if (line.size() == 1) {
201b9c1b51eSKate Stone     switch (line[0]) {
20244d93782SGreg Clayton     case 'y':
20344d93782SGreg Clayton     case 'Y':
20444d93782SGreg Clayton       m_user_response = true;
20544d93782SGreg Clayton       io_handler.SetIsDone(true);
20644d93782SGreg Clayton       return;
20744d93782SGreg Clayton     case 'n':
20844d93782SGreg Clayton     case 'N':
20944d93782SGreg Clayton       m_user_response = false;
21044d93782SGreg Clayton       io_handler.SetIsDone(true);
21144d93782SGreg Clayton       return;
21244d93782SGreg Clayton     default:
21344d93782SGreg Clayton       break;
21444d93782SGreg Clayton     }
21544d93782SGreg Clayton   }
21644d93782SGreg Clayton 
217b9c1b51eSKate Stone   if (line == "yes" || line == "YES" || line == "Yes") {
21844d93782SGreg Clayton     m_user_response = true;
21944d93782SGreg Clayton     io_handler.SetIsDone(true);
220b9c1b51eSKate Stone   } else if (line == "no" || line == "NO" || line == "No") {
22144d93782SGreg Clayton     m_user_response = false;
22244d93782SGreg Clayton     io_handler.SetIsDone(true);
22344d93782SGreg Clayton   }
22444d93782SGreg Clayton }
22544d93782SGreg Clayton 
226b9c1b51eSKate Stone int IOHandlerDelegate::IOHandlerComplete(IOHandler &io_handler,
22744d93782SGreg Clayton                                          const char *current_line,
22844d93782SGreg Clayton                                          const char *cursor,
22944d93782SGreg Clayton                                          const char *last_char,
23044d93782SGreg Clayton                                          int skip_first_n_matches,
231b9c1b51eSKate Stone                                          int max_matches, StringList &matches) {
232b9c1b51eSKate Stone   switch (m_completion) {
23344d93782SGreg Clayton   case Completion::None:
23444d93782SGreg Clayton     break;
23544d93782SGreg Clayton 
23644d93782SGreg Clayton   case Completion::LLDBCommand:
237b9c1b51eSKate Stone     return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion(
238b9c1b51eSKate Stone         current_line, cursor, last_char, skip_first_n_matches, max_matches,
23944d93782SGreg Clayton         matches);
24044d93782SGreg Clayton 
241b9c1b51eSKate Stone   case Completion::Expression: {
242*a2e76c0bSRaphael Isemann     CompletionRequest request(current_line, current_line - cursor,
243*a2e76c0bSRaphael Isemann                               skip_first_n_matches, max_matches, matches);
244b9c1b51eSKate Stone     CommandCompletions::InvokeCommonCompletionCallbacks(
245b9c1b51eSKate Stone         io_handler.GetDebugger().GetCommandInterpreter(),
246*a2e76c0bSRaphael Isemann         CommandCompletions::eVariablePathCompletion, request, nullptr);
24744d93782SGreg Clayton 
248*a2e76c0bSRaphael Isemann     size_t num_matches = request.GetMatches().GetSize();
249b9c1b51eSKate Stone     if (num_matches > 0) {
25044d93782SGreg Clayton       std::string common_prefix;
251*a2e76c0bSRaphael Isemann       request.GetMatches().LongestCommonPrefix(common_prefix);
252*a2e76c0bSRaphael Isemann       const size_t partial_name_len = request.GetCursorArgumentPrefix().size();
25344d93782SGreg Clayton 
25405097246SAdrian Prantl       // If we matched a unique single command, add a space... Only do this if
25505097246SAdrian Prantl       // the completer told us this was a complete word, however...
256*a2e76c0bSRaphael Isemann       if (num_matches == 1 && request.GetWordComplete()) {
25744d93782SGreg Clayton         common_prefix.push_back(' ');
25844d93782SGreg Clayton       }
25944d93782SGreg Clayton       common_prefix.erase(0, partial_name_len);
26044d93782SGreg Clayton       matches.InsertStringAtIndex(0, std::move(common_prefix));
26144d93782SGreg Clayton     }
26244d93782SGreg Clayton     return num_matches;
263b9c1b51eSKate Stone   } break;
26444d93782SGreg Clayton   }
26544d93782SGreg Clayton 
26644d93782SGreg Clayton   return 0;
26744d93782SGreg Clayton }
26844d93782SGreg Clayton 
269b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
270b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
27144d93782SGreg Clayton     const char *editline_name, // Used for saving history files
272514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
273514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
274514d8cd8SZachary Turner     IOHandlerDelegate &delegate)
275b9c1b51eSKate Stone     : IOHandlerEditline(debugger, type,
27644d93782SGreg Clayton                         StreamFileSP(), // Inherit input from top input reader
27744d93782SGreg Clayton                         StreamFileSP(), // Inherit output from top input reader
27844d93782SGreg Clayton                         StreamFileSP(), // Inherit error from top input reader
279340b0309SGreg Clayton                         0,              // Flags
28044d93782SGreg Clayton                         editline_name,  // Used for saving history files
281b9c1b51eSKate Stone                         prompt, continuation_prompt, multi_line, color_prompts,
282b9c1b51eSKate Stone                         line_number_start, delegate) {}
28344d93782SGreg Clayton 
284b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
285b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
286b9c1b51eSKate Stone     const lldb::StreamFileSP &input_sp, const lldb::StreamFileSP &output_sp,
287b9c1b51eSKate Stone     const lldb::StreamFileSP &error_sp, uint32_t flags,
28844d93782SGreg Clayton     const char *editline_name, // Used for saving history files
289514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
290514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
291514d8cd8SZachary Turner     IOHandlerDelegate &delegate)
292b9c1b51eSKate Stone     : IOHandler(debugger, type, input_sp, output_sp, error_sp, flags),
293cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
29444d93782SGreg Clayton       m_editline_ap(),
295cacde7dfSTodd Fiala #endif
296b9c1b51eSKate Stone       m_delegate(delegate), m_prompt(), m_continuation_prompt(),
297b9c1b51eSKate Stone       m_current_lines_ptr(nullptr), m_base_line_number(line_number_start),
298b9c1b51eSKate Stone       m_curr_line_idx(UINT32_MAX), m_multi_line(multi_line),
299b9c1b51eSKate Stone       m_color_prompts(color_prompts), m_interrupt_exits(true),
300b9c1b51eSKate Stone       m_editing(false) {
30144d93782SGreg Clayton   SetPrompt(prompt);
30244d93782SGreg Clayton 
303cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
304914b8d98SDeepak Panickal   bool use_editline = false;
305340b0309SGreg Clayton 
306340b0309SGreg Clayton   use_editline = m_input_sp->GetFile().GetIsRealTerminal();
30744d93782SGreg Clayton 
308b9c1b51eSKate Stone   if (use_editline) {
309b9c1b51eSKate Stone     m_editline_ap.reset(new Editline(editline_name, GetInputFILE(),
310b9c1b51eSKate Stone                                      GetOutputFILE(), GetErrorFILE(),
311e30f11d9SKate Stone                                      m_color_prompts));
312e30f11d9SKate Stone     m_editline_ap->SetIsInputCompleteCallback(IsInputCompleteCallback, this);
31344d93782SGreg Clayton     m_editline_ap->SetAutoCompleteCallback(AutoCompleteCallback, this);
314e30f11d9SKate Stone     // See if the delegate supports fixing indentation
315e30f11d9SKate Stone     const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
316b9c1b51eSKate Stone     if (indent_chars) {
317b9c1b51eSKate Stone       // The delegate does support indentation, hook it up so when any
31805097246SAdrian Prantl       // indentation character is typed, the delegate gets a chance to fix it
319b9c1b51eSKate Stone       m_editline_ap->SetFixIndentationCallback(FixIndentationCallback, this,
320b9c1b51eSKate Stone                                                indent_chars);
321e30f11d9SKate Stone     }
32244d93782SGreg Clayton   }
323cacde7dfSTodd Fiala #endif
324e30f11d9SKate Stone   SetBaseLineNumber(m_base_line_number);
325514d8cd8SZachary Turner   SetPrompt(prompt);
326e30f11d9SKate Stone   SetContinuationPrompt(continuation_prompt);
32744d93782SGreg Clayton }
32844d93782SGreg Clayton 
329b9c1b51eSKate Stone IOHandlerEditline::~IOHandlerEditline() {
330cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
33144d93782SGreg Clayton   m_editline_ap.reset();
332cacde7dfSTodd Fiala #endif
33344d93782SGreg Clayton }
33444d93782SGreg Clayton 
335b9c1b51eSKate Stone void IOHandlerEditline::Activate() {
336e30f11d9SKate Stone   IOHandler::Activate();
337e30f11d9SKate Stone   m_delegate.IOHandlerActivated(*this);
338e30f11d9SKate Stone }
339e30f11d9SKate Stone 
340b9c1b51eSKate Stone void IOHandlerEditline::Deactivate() {
341e30f11d9SKate Stone   IOHandler::Deactivate();
342e30f11d9SKate Stone   m_delegate.IOHandlerDeactivated(*this);
343e30f11d9SKate Stone }
344e30f11d9SKate Stone 
345b9c1b51eSKate Stone bool IOHandlerEditline::GetLine(std::string &line, bool &interrupted) {
346cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
347b9c1b51eSKate Stone   if (m_editline_ap) {
348e30f11d9SKate Stone     return m_editline_ap->GetLine(line, interrupted);
349b9c1b51eSKate Stone   } else {
350cacde7dfSTodd Fiala #endif
35144d93782SGreg Clayton     line.clear();
35244d93782SGreg Clayton 
35344d93782SGreg Clayton     FILE *in = GetInputFILE();
354b9c1b51eSKate Stone     if (in) {
355b9c1b51eSKate Stone       if (GetIsInteractive()) {
356c5dac77aSEugene Zelenko         const char *prompt = nullptr;
357e30f11d9SKate Stone 
358e30f11d9SKate Stone         if (m_multi_line && m_curr_line_idx > 0)
359e30f11d9SKate Stone           prompt = GetContinuationPrompt();
360e30f11d9SKate Stone 
361c5dac77aSEugene Zelenko         if (prompt == nullptr)
362e30f11d9SKate Stone           prompt = GetPrompt();
363e30f11d9SKate Stone 
364b9c1b51eSKate Stone         if (prompt && prompt[0]) {
36544d93782SGreg Clayton           FILE *out = GetOutputFILE();
366b9c1b51eSKate Stone           if (out) {
36744d93782SGreg Clayton             ::fprintf(out, "%s", prompt);
36844d93782SGreg Clayton             ::fflush(out);
36944d93782SGreg Clayton           }
37044d93782SGreg Clayton         }
37144d93782SGreg Clayton       }
37244d93782SGreg Clayton       char buffer[256];
37344d93782SGreg Clayton       bool done = false;
3740f86e6e7SGreg Clayton       bool got_line = false;
375e034a04eSGreg Clayton       m_editing = true;
376b9c1b51eSKate Stone       while (!done) {
377b9c1b51eSKate Stone         if (fgets(buffer, sizeof(buffer), in) == nullptr) {
378c7797accSGreg Clayton           const int saved_errno = errno;
379c9cf5798SGreg Clayton           if (feof(in))
38044d93782SGreg Clayton             done = true;
381b9c1b51eSKate Stone           else if (ferror(in)) {
382c7797accSGreg Clayton             if (saved_errno != EINTR)
383c7797accSGreg Clayton               done = true;
384c7797accSGreg Clayton           }
385b9c1b51eSKate Stone         } else {
3860f86e6e7SGreg Clayton           got_line = true;
38744d93782SGreg Clayton           size_t buffer_len = strlen(buffer);
38844d93782SGreg Clayton           assert(buffer[buffer_len] == '\0');
38944d93782SGreg Clayton           char last_char = buffer[buffer_len - 1];
390b9c1b51eSKate Stone           if (last_char == '\r' || last_char == '\n') {
39144d93782SGreg Clayton             done = true;
39244d93782SGreg Clayton             // Strip trailing newlines
393b9c1b51eSKate Stone             while (last_char == '\r' || last_char == '\n') {
39444d93782SGreg Clayton               --buffer_len;
39544d93782SGreg Clayton               if (buffer_len == 0)
39644d93782SGreg Clayton                 break;
39744d93782SGreg Clayton               last_char = buffer[buffer_len - 1];
39844d93782SGreg Clayton             }
39944d93782SGreg Clayton           }
40044d93782SGreg Clayton           line.append(buffer, buffer_len);
40144d93782SGreg Clayton         }
40244d93782SGreg Clayton       }
403e034a04eSGreg Clayton       m_editing = false;
40405097246SAdrian Prantl       // We might have gotten a newline on a line by itself make sure to return
40505097246SAdrian Prantl       // true in this case.
4060f86e6e7SGreg Clayton       return got_line;
407b9c1b51eSKate Stone     } else {
40844d93782SGreg Clayton       // No more input file, we are done...
40944d93782SGreg Clayton       SetIsDone(true);
41044d93782SGreg Clayton     }
411340b0309SGreg Clayton     return false;
412cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
41344d93782SGreg Clayton   }
414cacde7dfSTodd Fiala #endif
41544d93782SGreg Clayton }
41644d93782SGreg Clayton 
417cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
418b9c1b51eSKate Stone bool IOHandlerEditline::IsInputCompleteCallback(Editline *editline,
41944d93782SGreg Clayton                                                 StringList &lines,
420b9c1b51eSKate Stone                                                 void *baton) {
42144d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
422b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader,
423b9c1b51eSKate Stone                                                               lines);
424e30f11d9SKate Stone }
425e30f11d9SKate Stone 
426b9c1b51eSKate Stone int IOHandlerEditline::FixIndentationCallback(Editline *editline,
427e30f11d9SKate Stone                                               const StringList &lines,
428e30f11d9SKate Stone                                               int cursor_position,
429b9c1b51eSKate Stone                                               void *baton) {
430e30f11d9SKate Stone   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
431b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerFixIndentation(
432b9c1b51eSKate Stone       *editline_reader, lines, cursor_position);
43344d93782SGreg Clayton }
43444d93782SGreg Clayton 
435b9c1b51eSKate Stone int IOHandlerEditline::AutoCompleteCallback(const char *current_line,
43644d93782SGreg Clayton                                             const char *cursor,
43744d93782SGreg Clayton                                             const char *last_char,
43844d93782SGreg Clayton                                             int skip_first_n_matches,
43944d93782SGreg Clayton                                             int max_matches,
440b9c1b51eSKate Stone                                             StringList &matches, void *baton) {
44144d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
44244d93782SGreg Clayton   if (editline_reader)
443b9c1b51eSKate Stone     return editline_reader->m_delegate.IOHandlerComplete(
444b9c1b51eSKate Stone         *editline_reader, current_line, cursor, last_char, skip_first_n_matches,
445b9c1b51eSKate Stone         max_matches, matches);
44644d93782SGreg Clayton   return 0;
44744d93782SGreg Clayton }
448cacde7dfSTodd Fiala #endif
44944d93782SGreg Clayton 
450b9c1b51eSKate Stone const char *IOHandlerEditline::GetPrompt() {
451cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
452b9c1b51eSKate Stone   if (m_editline_ap) {
45344d93782SGreg Clayton     return m_editline_ap->GetPrompt();
454b9c1b51eSKate Stone   } else {
455cacde7dfSTodd Fiala #endif
456cacde7dfSTodd Fiala     if (m_prompt.empty())
457c5dac77aSEugene Zelenko       return nullptr;
458cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
459cacde7dfSTodd Fiala   }
460cacde7dfSTodd Fiala #endif
46144d93782SGreg Clayton   return m_prompt.c_str();
46244d93782SGreg Clayton }
46344d93782SGreg Clayton 
464514d8cd8SZachary Turner bool IOHandlerEditline::SetPrompt(llvm::StringRef prompt) {
465514d8cd8SZachary Turner   m_prompt = prompt;
466514d8cd8SZachary Turner 
467cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
46844d93782SGreg Clayton   if (m_editline_ap)
469c5dac77aSEugene Zelenko     m_editline_ap->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
470cacde7dfSTodd Fiala #endif
47144d93782SGreg Clayton   return true;
47244d93782SGreg Clayton }
47344d93782SGreg Clayton 
474b9c1b51eSKate Stone const char *IOHandlerEditline::GetContinuationPrompt() {
475b9c1b51eSKate Stone   return (m_continuation_prompt.empty() ? nullptr
476b9c1b51eSKate Stone                                         : m_continuation_prompt.c_str());
477e30f11d9SKate Stone }
478e30f11d9SKate Stone 
479514d8cd8SZachary Turner void IOHandlerEditline::SetContinuationPrompt(llvm::StringRef prompt) {
480514d8cd8SZachary Turner   m_continuation_prompt = prompt;
481e30f11d9SKate Stone 
482d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
483e30f11d9SKate Stone   if (m_editline_ap)
484b9c1b51eSKate Stone     m_editline_ap->SetContinuationPrompt(m_continuation_prompt.empty()
485b9c1b51eSKate Stone                                              ? nullptr
486b9c1b51eSKate Stone                                              : m_continuation_prompt.c_str());
487d553d00cSZachary Turner #endif
488e30f11d9SKate Stone }
489e30f11d9SKate Stone 
490b9c1b51eSKate Stone void IOHandlerEditline::SetBaseLineNumber(uint32_t line) {
491f6913cd7SGreg Clayton   m_base_line_number = line;
492f6913cd7SGreg Clayton }
493e30f11d9SKate Stone 
494b9c1b51eSKate Stone uint32_t IOHandlerEditline::GetCurrentLineIndex() const {
495d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
496e30f11d9SKate Stone   if (m_editline_ap)
497e30f11d9SKate Stone     return m_editline_ap->GetCurrentLine();
498e30f11d9SKate Stone #endif
499e30f11d9SKate Stone   return m_curr_line_idx;
500e30f11d9SKate Stone }
501e30f11d9SKate Stone 
502b9c1b51eSKate Stone bool IOHandlerEditline::GetLines(StringList &lines, bool &interrupted) {
503e30f11d9SKate Stone   m_current_lines_ptr = &lines;
504e30f11d9SKate Stone 
50544d93782SGreg Clayton   bool success = false;
506cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
507b9c1b51eSKate Stone   if (m_editline_ap) {
508e30f11d9SKate Stone     return m_editline_ap->GetLines(m_base_line_number, lines, interrupted);
509b9c1b51eSKate Stone   } else {
510cacde7dfSTodd Fiala #endif
511e30f11d9SKate Stone     bool done = false;
51297206d57SZachary Turner     Status error;
51344d93782SGreg Clayton 
514b9c1b51eSKate Stone     while (!done) {
515f6913cd7SGreg Clayton       // Show line numbers if we are asked to
51644d93782SGreg Clayton       std::string line;
517b9c1b51eSKate Stone       if (m_base_line_number > 0 && GetIsInteractive()) {
518f6913cd7SGreg Clayton         FILE *out = GetOutputFILE();
519f6913cd7SGreg Clayton         if (out)
520b9c1b51eSKate Stone           ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(),
521b9c1b51eSKate Stone                     GetPrompt() == nullptr ? " " : "");
522f6913cd7SGreg Clayton       }
523f6913cd7SGreg Clayton 
524e30f11d9SKate Stone       m_curr_line_idx = lines.GetSize();
525e30f11d9SKate Stone 
526f0066ad0SGreg Clayton       bool interrupted = false;
527b9c1b51eSKate Stone       if (GetLine(line, interrupted) && !interrupted) {
52844d93782SGreg Clayton         lines.AppendString(line);
529e30f11d9SKate Stone         done = m_delegate.IOHandlerIsInputComplete(*this, lines);
530b9c1b51eSKate Stone       } else {
531e30f11d9SKate Stone         done = true;
53244d93782SGreg Clayton       }
53344d93782SGreg Clayton     }
53444d93782SGreg Clayton     success = lines.GetSize() > 0;
535cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
53644d93782SGreg Clayton   }
537cacde7dfSTodd Fiala #endif
53844d93782SGreg Clayton   return success;
53944d93782SGreg Clayton }
54044d93782SGreg Clayton 
54105097246SAdrian Prantl // Each IOHandler gets to run until it is done. It should read data from the
54205097246SAdrian Prantl // "in" and place output into "out" and "err and return when done.
543b9c1b51eSKate Stone void IOHandlerEditline::Run() {
54444d93782SGreg Clayton   std::string line;
545b9c1b51eSKate Stone   while (IsActive()) {
546f0066ad0SGreg Clayton     bool interrupted = false;
547b9c1b51eSKate Stone     if (m_multi_line) {
54844d93782SGreg Clayton       StringList lines;
549b9c1b51eSKate Stone       if (GetLines(lines, interrupted)) {
550b9c1b51eSKate Stone         if (interrupted) {
551e30f11d9SKate Stone           m_done = m_interrupt_exits;
552e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
553e30f11d9SKate Stone 
554b9c1b51eSKate Stone         } else {
55544d93782SGreg Clayton           line = lines.CopyList();
55644d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
55744d93782SGreg Clayton         }
558b9c1b51eSKate Stone       } else {
55944d93782SGreg Clayton         m_done = true;
56044d93782SGreg Clayton       }
561b9c1b51eSKate Stone     } else {
562b9c1b51eSKate Stone       if (GetLine(line, interrupted)) {
563e30f11d9SKate Stone         if (interrupted)
564e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
565e30f11d9SKate Stone         else
56644d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
567b9c1b51eSKate Stone       } else {
56844d93782SGreg Clayton         m_done = true;
56944d93782SGreg Clayton       }
57044d93782SGreg Clayton     }
57144d93782SGreg Clayton   }
57244d93782SGreg Clayton }
57344d93782SGreg Clayton 
574b9c1b51eSKate Stone void IOHandlerEditline::Cancel() {
575cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
576e68f5d6bSGreg Clayton   if (m_editline_ap)
5774446487dSPavel Labath     m_editline_ap->Cancel();
578cacde7dfSTodd Fiala #endif
579e68f5d6bSGreg Clayton }
580e68f5d6bSGreg Clayton 
581b9c1b51eSKate Stone bool IOHandlerEditline::Interrupt() {
582f0066ad0SGreg Clayton   // Let the delgate handle it first
583f0066ad0SGreg Clayton   if (m_delegate.IOHandlerInterrupt(*this))
584f0066ad0SGreg Clayton     return true;
585f0066ad0SGreg Clayton 
586cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
58744d93782SGreg Clayton   if (m_editline_ap)
588f0066ad0SGreg Clayton     return m_editline_ap->Interrupt();
589cacde7dfSTodd Fiala #endif
590f0066ad0SGreg Clayton   return false;
59144d93782SGreg Clayton }
59244d93782SGreg Clayton 
593b9c1b51eSKate Stone void IOHandlerEditline::GotEOF() {
594cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
59544d93782SGreg Clayton   if (m_editline_ap)
59644d93782SGreg Clayton     m_editline_ap->Interrupt();
597cacde7dfSTodd Fiala #endif
59844d93782SGreg Clayton }
59944d93782SGreg Clayton 
600b9c1b51eSKate Stone void IOHandlerEditline::PrintAsync(Stream *stream, const char *s, size_t len) {
6014446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
6024446487dSPavel Labath   if (m_editline_ap)
6034446487dSPavel Labath     m_editline_ap->PrintAsync(stream, s, len);
6044446487dSPavel Labath   else
6054446487dSPavel Labath #endif
606fab31220STed Woodward   {
607fab31220STed Woodward #ifdef _MSC_VER
608341e4789SDawn Perchik     const char *prompt = GetPrompt();
609b9c1b51eSKate Stone     if (prompt) {
610fab31220STed Woodward       // Back up over previous prompt using Windows API
611fab31220STed Woodward       CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
612fab31220STed Woodward       HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
613fab31220STed Woodward       GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
614fab31220STed Woodward       COORD coord = screen_buffer_info.dwCursorPosition;
615fab31220STed Woodward       coord.X -= strlen(prompt);
616fab31220STed Woodward       if (coord.X < 0)
617fab31220STed Woodward         coord.X = 0;
618fab31220STed Woodward       SetConsoleCursorPosition(console_handle, coord);
619fab31220STed Woodward     }
620fab31220STed Woodward #endif
6214446487dSPavel Labath     IOHandler::PrintAsync(stream, s, len);
622341e4789SDawn Perchik #ifdef _MSC_VER
623fab31220STed Woodward     if (prompt)
624b9c1b51eSKate Stone       IOHandler::PrintAsync(GetOutputStreamFile().get(), prompt,
625b9c1b51eSKate Stone                             strlen(prompt));
626341e4789SDawn Perchik #endif
627fab31220STed Woodward   }
6284446487dSPavel Labath }
6294446487dSPavel Labath 
63005097246SAdrian Prantl // we may want curses to be disabled for some builds for instance, windows
631914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
632914b8d98SDeepak Panickal 
63344d93782SGreg Clayton #define KEY_RETURN 10
63444d93782SGreg Clayton #define KEY_ESCAPE 27
63544d93782SGreg Clayton 
636b9c1b51eSKate Stone namespace curses {
63744d93782SGreg Clayton class Menu;
63844d93782SGreg Clayton class MenuDelegate;
63944d93782SGreg Clayton class Window;
64044d93782SGreg Clayton class WindowDelegate;
64144d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP;
64244d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
64344d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP;
64444d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
64544d93782SGreg Clayton typedef std::vector<MenuSP> Menus;
64644d93782SGreg Clayton typedef std::vector<WindowSP> Windows;
64744d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates;
64844d93782SGreg Clayton 
64944d93782SGreg Clayton #if 0
65044d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
65144d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
65244d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
65344d93782SGreg Clayton #endif
654315b6884SEugene Zelenko 
655b9c1b51eSKate Stone struct Point {
65644d93782SGreg Clayton   int x;
65744d93782SGreg Clayton   int y;
65844d93782SGreg Clayton 
659b9c1b51eSKate Stone   Point(int _x = 0, int _y = 0) : x(_x), y(_y) {}
66044d93782SGreg Clayton 
661b9c1b51eSKate Stone   void Clear() {
66244d93782SGreg Clayton     x = 0;
66344d93782SGreg Clayton     y = 0;
66444d93782SGreg Clayton   }
66544d93782SGreg Clayton 
666b9c1b51eSKate Stone   Point &operator+=(const Point &rhs) {
66744d93782SGreg Clayton     x += rhs.x;
66844d93782SGreg Clayton     y += rhs.y;
66944d93782SGreg Clayton     return *this;
67044d93782SGreg Clayton   }
67144d93782SGreg Clayton 
672b9c1b51eSKate Stone   void Dump() { printf("(x=%i, y=%i)\n", x, y); }
67344d93782SGreg Clayton };
67444d93782SGreg Clayton 
675b9c1b51eSKate Stone bool operator==(const Point &lhs, const Point &rhs) {
67644d93782SGreg Clayton   return lhs.x == rhs.x && lhs.y == rhs.y;
67744d93782SGreg Clayton }
678315b6884SEugene Zelenko 
679b9c1b51eSKate Stone bool operator!=(const Point &lhs, const Point &rhs) {
68044d93782SGreg Clayton   return lhs.x != rhs.x || lhs.y != rhs.y;
68144d93782SGreg Clayton }
68244d93782SGreg Clayton 
683b9c1b51eSKate Stone struct Size {
68444d93782SGreg Clayton   int width;
68544d93782SGreg Clayton   int height;
686b9c1b51eSKate Stone   Size(int w = 0, int h = 0) : width(w), height(h) {}
68744d93782SGreg Clayton 
688b9c1b51eSKate Stone   void Clear() {
68944d93782SGreg Clayton     width = 0;
69044d93782SGreg Clayton     height = 0;
69144d93782SGreg Clayton   }
69244d93782SGreg Clayton 
693b9c1b51eSKate Stone   void Dump() { printf("(w=%i, h=%i)\n", width, height); }
69444d93782SGreg Clayton };
69544d93782SGreg Clayton 
696b9c1b51eSKate Stone bool operator==(const Size &lhs, const Size &rhs) {
69744d93782SGreg Clayton   return lhs.width == rhs.width && lhs.height == rhs.height;
69844d93782SGreg Clayton }
699315b6884SEugene Zelenko 
700b9c1b51eSKate Stone bool operator!=(const Size &lhs, const Size &rhs) {
70144d93782SGreg Clayton   return lhs.width != rhs.width || lhs.height != rhs.height;
70244d93782SGreg Clayton }
70344d93782SGreg Clayton 
704b9c1b51eSKate Stone struct Rect {
70544d93782SGreg Clayton   Point origin;
70644d93782SGreg Clayton   Size size;
70744d93782SGreg Clayton 
708b9c1b51eSKate Stone   Rect() : origin(), size() {}
70944d93782SGreg Clayton 
710b9c1b51eSKate Stone   Rect(const Point &p, const Size &s) : origin(p), size(s) {}
71144d93782SGreg Clayton 
712b9c1b51eSKate Stone   void Clear() {
71344d93782SGreg Clayton     origin.Clear();
71444d93782SGreg Clayton     size.Clear();
71544d93782SGreg Clayton   }
71644d93782SGreg Clayton 
717b9c1b51eSKate Stone   void Dump() {
718b9c1b51eSKate Stone     printf("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width,
719b9c1b51eSKate Stone            size.height);
72044d93782SGreg Clayton   }
72144d93782SGreg Clayton 
722b9c1b51eSKate Stone   void Inset(int w, int h) {
72344d93782SGreg Clayton     if (size.width > w * 2)
72444d93782SGreg Clayton       size.width -= w * 2;
72544d93782SGreg Clayton     origin.x += w;
72644d93782SGreg Clayton 
72744d93782SGreg Clayton     if (size.height > h * 2)
72844d93782SGreg Clayton       size.height -= h * 2;
72944d93782SGreg Clayton     origin.y += h;
73044d93782SGreg Clayton   }
731315b6884SEugene Zelenko 
73205097246SAdrian Prantl   // Return a status bar rectangle which is the last line of this rectangle.
73305097246SAdrian Prantl   // This rectangle will be modified to not include the status bar area.
734b9c1b51eSKate Stone   Rect MakeStatusBar() {
73544d93782SGreg Clayton     Rect status_bar;
736b9c1b51eSKate Stone     if (size.height > 1) {
73744d93782SGreg Clayton       status_bar.origin.x = origin.x;
73844d93782SGreg Clayton       status_bar.origin.y = size.height;
73944d93782SGreg Clayton       status_bar.size.width = size.width;
74044d93782SGreg Clayton       status_bar.size.height = 1;
74144d93782SGreg Clayton       --size.height;
74244d93782SGreg Clayton     }
74344d93782SGreg Clayton     return status_bar;
74444d93782SGreg Clayton   }
74544d93782SGreg Clayton 
74605097246SAdrian Prantl   // Return a menubar rectangle which is the first line of this rectangle. This
74705097246SAdrian Prantl   // rectangle will be modified to not include the menubar area.
748b9c1b51eSKate Stone   Rect MakeMenuBar() {
74944d93782SGreg Clayton     Rect menubar;
750b9c1b51eSKate Stone     if (size.height > 1) {
75144d93782SGreg Clayton       menubar.origin.x = origin.x;
75244d93782SGreg Clayton       menubar.origin.y = origin.y;
75344d93782SGreg Clayton       menubar.size.width = size.width;
75444d93782SGreg Clayton       menubar.size.height = 1;
75544d93782SGreg Clayton       ++origin.y;
75644d93782SGreg Clayton       --size.height;
75744d93782SGreg Clayton     }
75844d93782SGreg Clayton     return menubar;
75944d93782SGreg Clayton   }
76044d93782SGreg Clayton 
761b9c1b51eSKate Stone   void HorizontalSplitPercentage(float top_percentage, Rect &top,
762b9c1b51eSKate Stone                                  Rect &bottom) const {
76344d93782SGreg Clayton     float top_height = top_percentage * size.height;
76444d93782SGreg Clayton     HorizontalSplit(top_height, top, bottom);
76544d93782SGreg Clayton   }
76644d93782SGreg Clayton 
767b9c1b51eSKate Stone   void HorizontalSplit(int top_height, Rect &top, Rect &bottom) const {
76844d93782SGreg Clayton     top = *this;
769b9c1b51eSKate Stone     if (top_height < size.height) {
77044d93782SGreg Clayton       top.size.height = top_height;
77144d93782SGreg Clayton       bottom.origin.x = origin.x;
77244d93782SGreg Clayton       bottom.origin.y = origin.y + top.size.height;
77344d93782SGreg Clayton       bottom.size.width = size.width;
77444d93782SGreg Clayton       bottom.size.height = size.height - top.size.height;
775b9c1b51eSKate Stone     } else {
77644d93782SGreg Clayton       bottom.Clear();
77744d93782SGreg Clayton     }
77844d93782SGreg Clayton   }
77944d93782SGreg Clayton 
780b9c1b51eSKate Stone   void VerticalSplitPercentage(float left_percentage, Rect &left,
781b9c1b51eSKate Stone                                Rect &right) const {
78244d93782SGreg Clayton     float left_width = left_percentage * size.width;
78344d93782SGreg Clayton     VerticalSplit(left_width, left, right);
78444d93782SGreg Clayton   }
78544d93782SGreg Clayton 
786b9c1b51eSKate Stone   void VerticalSplit(int left_width, Rect &left, Rect &right) const {
78744d93782SGreg Clayton     left = *this;
788b9c1b51eSKate Stone     if (left_width < size.width) {
78944d93782SGreg Clayton       left.size.width = left_width;
79044d93782SGreg Clayton       right.origin.x = origin.x + left.size.width;
79144d93782SGreg Clayton       right.origin.y = origin.y;
79244d93782SGreg Clayton       right.size.width = size.width - left.size.width;
79344d93782SGreg Clayton       right.size.height = size.height;
794b9c1b51eSKate Stone     } else {
79544d93782SGreg Clayton       right.Clear();
79644d93782SGreg Clayton     }
79744d93782SGreg Clayton   }
79844d93782SGreg Clayton };
79944d93782SGreg Clayton 
800b9c1b51eSKate Stone bool operator==(const Rect &lhs, const Rect &rhs) {
80144d93782SGreg Clayton   return lhs.origin == rhs.origin && lhs.size == rhs.size;
80244d93782SGreg Clayton }
803315b6884SEugene Zelenko 
804b9c1b51eSKate Stone bool operator!=(const Rect &lhs, const Rect &rhs) {
80544d93782SGreg Clayton   return lhs.origin != rhs.origin || lhs.size != rhs.size;
80644d93782SGreg Clayton }
80744d93782SGreg Clayton 
808b9c1b51eSKate Stone enum HandleCharResult {
80944d93782SGreg Clayton   eKeyNotHandled = 0,
81044d93782SGreg Clayton   eKeyHandled = 1,
81144d93782SGreg Clayton   eQuitApplication = 2
81244d93782SGreg Clayton };
81344d93782SGreg Clayton 
814b9c1b51eSKate Stone enum class MenuActionResult {
81544d93782SGreg Clayton   Handled,
81644d93782SGreg Clayton   NotHandled,
81744d93782SGreg Clayton   Quit // Exit all menus and quit
81844d93782SGreg Clayton };
81944d93782SGreg Clayton 
820b9c1b51eSKate Stone struct KeyHelp {
82144d93782SGreg Clayton   int ch;
82244d93782SGreg Clayton   const char *description;
82344d93782SGreg Clayton };
82444d93782SGreg Clayton 
825b9c1b51eSKate Stone class WindowDelegate {
82644d93782SGreg Clayton public:
827b9c1b51eSKate Stone   virtual ~WindowDelegate() = default;
82844d93782SGreg Clayton 
829b9c1b51eSKate Stone   virtual bool WindowDelegateDraw(Window &window, bool force) {
83044d93782SGreg Clayton     return false; // Drawing not handled
83144d93782SGreg Clayton   }
83244d93782SGreg Clayton 
833b9c1b51eSKate Stone   virtual HandleCharResult WindowDelegateHandleChar(Window &window, int key) {
83444d93782SGreg Clayton     return eKeyNotHandled;
83544d93782SGreg Clayton   }
83644d93782SGreg Clayton 
837b9c1b51eSKate Stone   virtual const char *WindowDelegateGetHelpText() { return nullptr; }
83844d93782SGreg Clayton 
839b9c1b51eSKate Stone   virtual KeyHelp *WindowDelegateGetKeyHelp() { return nullptr; }
84044d93782SGreg Clayton };
84144d93782SGreg Clayton 
842b9c1b51eSKate Stone class HelpDialogDelegate : public WindowDelegate {
84344d93782SGreg Clayton public:
84444d93782SGreg Clayton   HelpDialogDelegate(const char *text, KeyHelp *key_help_array);
84544d93782SGreg Clayton 
846bd5ae6b4SGreg Clayton   ~HelpDialogDelegate() override;
84744d93782SGreg Clayton 
848b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
84944d93782SGreg Clayton 
850b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
85144d93782SGreg Clayton 
852b9c1b51eSKate Stone   size_t GetNumLines() const { return m_text.GetSize(); }
85344d93782SGreg Clayton 
854b9c1b51eSKate Stone   size_t GetMaxLineLength() const { return m_text.GetMaxStringLength(); }
85544d93782SGreg Clayton 
85644d93782SGreg Clayton protected:
85744d93782SGreg Clayton   StringList m_text;
85844d93782SGreg Clayton   int m_first_visible_line;
85944d93782SGreg Clayton };
86044d93782SGreg Clayton 
861b9c1b51eSKate Stone class Window {
86244d93782SGreg Clayton public:
863b9c1b51eSKate Stone   Window(const char *name)
864b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
865b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
866b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(false),
867b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {}
86844d93782SGreg Clayton 
869b9c1b51eSKate Stone   Window(const char *name, WINDOW *w, bool del = true)
870b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
871b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
872b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(del),
873b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
87444d93782SGreg Clayton     if (w)
87544d93782SGreg Clayton       Reset(w);
87644d93782SGreg Clayton   }
87744d93782SGreg Clayton 
878b9c1b51eSKate Stone   Window(const char *name, const Rect &bounds)
879b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_parent(nullptr), m_subwindows(),
880b9c1b51eSKate Stone         m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
881b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(true),
882b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
883b9c1b51eSKate Stone     Reset(::newwin(bounds.size.height, bounds.size.width, bounds.origin.y,
884b9c1b51eSKate Stone                    bounds.origin.y));
88544d93782SGreg Clayton   }
88644d93782SGreg Clayton 
887b9c1b51eSKate Stone   virtual ~Window() {
88844d93782SGreg Clayton     RemoveSubWindows();
88944d93782SGreg Clayton     Reset();
89044d93782SGreg Clayton   }
89144d93782SGreg Clayton 
892b9c1b51eSKate Stone   void Reset(WINDOW *w = nullptr, bool del = true) {
89344d93782SGreg Clayton     if (m_window == w)
89444d93782SGreg Clayton       return;
89544d93782SGreg Clayton 
896b9c1b51eSKate Stone     if (m_panel) {
89744d93782SGreg Clayton       ::del_panel(m_panel);
898c5dac77aSEugene Zelenko       m_panel = nullptr;
89944d93782SGreg Clayton     }
900b9c1b51eSKate Stone     if (m_window && m_delete) {
90144d93782SGreg Clayton       ::delwin(m_window);
902c5dac77aSEugene Zelenko       m_window = nullptr;
90344d93782SGreg Clayton       m_delete = false;
90444d93782SGreg Clayton     }
905b9c1b51eSKate Stone     if (w) {
90644d93782SGreg Clayton       m_window = w;
90744d93782SGreg Clayton       m_panel = ::new_panel(m_window);
90844d93782SGreg Clayton       m_delete = del;
90944d93782SGreg Clayton     }
91044d93782SGreg Clayton   }
91144d93782SGreg Clayton 
91244d93782SGreg Clayton   void AttributeOn(attr_t attr) { ::wattron(m_window, attr); }
91344d93782SGreg Clayton   void AttributeOff(attr_t attr) { ::wattroff(m_window, attr); }
914b9c1b51eSKate Stone   void Box(chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) {
915b9c1b51eSKate Stone     ::box(m_window, v_char, h_char);
916b9c1b51eSKate Stone   }
91744d93782SGreg Clayton   void Clear() { ::wclear(m_window); }
91844d93782SGreg Clayton   void Erase() { ::werase(m_window); }
919b9c1b51eSKate Stone   Rect GetBounds() {
920b9c1b51eSKate Stone     return Rect(GetParentOrigin(), GetSize());
921b9c1b51eSKate Stone   } // Get the rectangle in our parent window
92244d93782SGreg Clayton   int GetChar() { return ::wgetch(m_window); }
92344d93782SGreg Clayton   int GetCursorX() { return getcurx(m_window); }
92444d93782SGreg Clayton   int GetCursorY() { return getcury(m_window); }
925b9c1b51eSKate Stone   Rect GetFrame() {
926b9c1b51eSKate Stone     return Rect(Point(), GetSize());
927b9c1b51eSKate Stone   } // Get our rectangle in our own coordinate system
92844d93782SGreg Clayton   Point GetParentOrigin() { return Point(GetParentX(), GetParentY()); }
92944d93782SGreg Clayton   Size GetSize() { return Size(GetWidth(), GetHeight()); }
93044d93782SGreg Clayton   int GetParentX() { return getparx(m_window); }
93144d93782SGreg Clayton   int GetParentY() { return getpary(m_window); }
93244d93782SGreg Clayton   int GetMaxX() { return getmaxx(m_window); }
93344d93782SGreg Clayton   int GetMaxY() { return getmaxy(m_window); }
93444d93782SGreg Clayton   int GetWidth() { return GetMaxX(); }
93544d93782SGreg Clayton   int GetHeight() { return GetMaxY(); }
93644d93782SGreg Clayton   void MoveCursor(int x, int y) { ::wmove(m_window, y, x); }
93744d93782SGreg Clayton   void MoveWindow(int x, int y) { MoveWindow(Point(x, y)); }
93844d93782SGreg Clayton   void Resize(int w, int h) { ::wresize(m_window, h, w); }
939b9c1b51eSKate Stone   void Resize(const Size &size) {
940b9c1b51eSKate Stone     ::wresize(m_window, size.height, size.width);
941b9c1b51eSKate Stone   }
94244d93782SGreg Clayton   void PutChar(int ch) { ::waddch(m_window, ch); }
94344d93782SGreg Clayton   void PutCString(const char *s, int len = -1) { ::waddnstr(m_window, s, len); }
94444d93782SGreg Clayton   void Refresh() { ::wrefresh(m_window); }
945b9c1b51eSKate Stone   void DeferredRefresh() {
94644d93782SGreg Clayton     // We are using panels, so we don't need to call this...
94744d93782SGreg Clayton     //::wnoutrefresh(m_window);
94844d93782SGreg Clayton   }
949b9c1b51eSKate Stone   void SetBackground(int color_pair_idx) {
950b9c1b51eSKate Stone     ::wbkgd(m_window, COLOR_PAIR(color_pair_idx));
951b9c1b51eSKate Stone   }
95244d93782SGreg Clayton   void UnderlineOn() { AttributeOn(A_UNDERLINE); }
95344d93782SGreg Clayton   void UnderlineOff() { AttributeOff(A_UNDERLINE); }
95444d93782SGreg Clayton 
955b9c1b51eSKate Stone   void PutCStringTruncated(const char *s, int right_pad) {
95644d93782SGreg Clayton     int bytes_left = GetWidth() - GetCursorX();
957b9c1b51eSKate Stone     if (bytes_left > right_pad) {
95844d93782SGreg Clayton       bytes_left -= right_pad;
95944d93782SGreg Clayton       ::waddnstr(m_window, s, bytes_left);
96044d93782SGreg Clayton     }
96144d93782SGreg Clayton   }
96244d93782SGreg Clayton 
963b9c1b51eSKate Stone   void MoveWindow(const Point &origin) {
96444d93782SGreg Clayton     const bool moving_window = origin != GetParentOrigin();
965b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
96644d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
96744d93782SGreg Clayton       Size size = GetSize();
968b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, size.height, size.width, origin.y,
969b9c1b51eSKate Stone                      origin.x),
970b9c1b51eSKate Stone             true);
971b9c1b51eSKate Stone     } else {
97244d93782SGreg Clayton       ::mvwin(m_window, origin.y, origin.x);
97344d93782SGreg Clayton     }
97444d93782SGreg Clayton   }
97544d93782SGreg Clayton 
976b9c1b51eSKate Stone   void SetBounds(const Rect &bounds) {
97744d93782SGreg Clayton     const bool moving_window = bounds.origin != GetParentOrigin();
978b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
97944d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
980b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, bounds.size.height, bounds.size.width,
981b9c1b51eSKate Stone                      bounds.origin.y, bounds.origin.x),
982b9c1b51eSKate Stone             true);
983b9c1b51eSKate Stone     } else {
98444d93782SGreg Clayton       if (moving_window)
98544d93782SGreg Clayton         MoveWindow(bounds.origin);
98644d93782SGreg Clayton       Resize(bounds.size);
98744d93782SGreg Clayton     }
98844d93782SGreg Clayton   }
98944d93782SGreg Clayton 
990b9c1b51eSKate Stone   void Printf(const char *format, ...) __attribute__((format(printf, 2, 3))) {
99144d93782SGreg Clayton     va_list args;
99244d93782SGreg Clayton     va_start(args, format);
99344d93782SGreg Clayton     vwprintw(m_window, format, args);
99444d93782SGreg Clayton     va_end(args);
99544d93782SGreg Clayton   }
99644d93782SGreg Clayton 
997b9c1b51eSKate Stone   void Touch() {
99844d93782SGreg Clayton     ::touchwin(m_window);
99944d93782SGreg Clayton     if (m_parent)
100044d93782SGreg Clayton       m_parent->Touch();
100144d93782SGreg Clayton   }
100244d93782SGreg Clayton 
1003b9c1b51eSKate Stone   WindowSP CreateSubWindow(const char *name, const Rect &bounds,
1004b9c1b51eSKate Stone                            bool make_active) {
100544d93782SGreg Clayton     WindowSP subwindow_sp;
1006b9c1b51eSKate Stone     if (m_window) {
1007b9c1b51eSKate Stone       subwindow_sp.reset(new Window(
1008b9c1b51eSKate Stone           name, ::subwin(m_window, bounds.size.height, bounds.size.width,
1009b9c1b51eSKate Stone                          bounds.origin.y, bounds.origin.x),
1010b9c1b51eSKate Stone           true));
101144d93782SGreg Clayton       subwindow_sp->m_is_subwin = true;
1012b9c1b51eSKate Stone     } else {
1013b9c1b51eSKate Stone       subwindow_sp.reset(
1014b9c1b51eSKate Stone           new Window(name, ::newwin(bounds.size.height, bounds.size.width,
1015b9c1b51eSKate Stone                                     bounds.origin.y, bounds.origin.x),
1016b9c1b51eSKate Stone                      true));
101744d93782SGreg Clayton       subwindow_sp->m_is_subwin = false;
101844d93782SGreg Clayton     }
101944d93782SGreg Clayton     subwindow_sp->m_parent = this;
1020b9c1b51eSKate Stone     if (make_active) {
102144d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
102244d93782SGreg Clayton       m_curr_active_window_idx = m_subwindows.size();
102344d93782SGreg Clayton     }
102444d93782SGreg Clayton     m_subwindows.push_back(subwindow_sp);
102544d93782SGreg Clayton     ::top_panel(subwindow_sp->m_panel);
102644d93782SGreg Clayton     m_needs_update = true;
102744d93782SGreg Clayton     return subwindow_sp;
102844d93782SGreg Clayton   }
102944d93782SGreg Clayton 
1030b9c1b51eSKate Stone   bool RemoveSubWindow(Window *window) {
103144d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
103244d93782SGreg Clayton     size_t i = 0;
1033b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
1034b9c1b51eSKate Stone       if ((*pos).get() == window) {
103544d93782SGreg Clayton         if (m_prev_active_window_idx == i)
103644d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1037b9c1b51eSKate Stone         else if (m_prev_active_window_idx != UINT32_MAX &&
1038b9c1b51eSKate Stone                  m_prev_active_window_idx > i)
103944d93782SGreg Clayton           --m_prev_active_window_idx;
104044d93782SGreg Clayton 
104144d93782SGreg Clayton         if (m_curr_active_window_idx == i)
104244d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
1043b9c1b51eSKate Stone         else if (m_curr_active_window_idx != UINT32_MAX &&
1044b9c1b51eSKate Stone                  m_curr_active_window_idx > i)
104544d93782SGreg Clayton           --m_curr_active_window_idx;
104644d93782SGreg Clayton         window->Erase();
104744d93782SGreg Clayton         m_subwindows.erase(pos);
104844d93782SGreg Clayton         m_needs_update = true;
104944d93782SGreg Clayton         if (m_parent)
105044d93782SGreg Clayton           m_parent->Touch();
105144d93782SGreg Clayton         else
105244d93782SGreg Clayton           ::touchwin(stdscr);
105344d93782SGreg Clayton         return true;
105444d93782SGreg Clayton       }
105544d93782SGreg Clayton     }
105644d93782SGreg Clayton     return false;
105744d93782SGreg Clayton   }
105844d93782SGreg Clayton 
1059b9c1b51eSKate Stone   WindowSP FindSubWindow(const char *name) {
106044d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
106144d93782SGreg Clayton     size_t i = 0;
1062b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
106344d93782SGreg Clayton       if ((*pos)->m_name.compare(name) == 0)
106444d93782SGreg Clayton         return *pos;
106544d93782SGreg Clayton     }
106644d93782SGreg Clayton     return WindowSP();
106744d93782SGreg Clayton   }
106844d93782SGreg Clayton 
1069b9c1b51eSKate Stone   void RemoveSubWindows() {
107044d93782SGreg Clayton     m_curr_active_window_idx = UINT32_MAX;
107144d93782SGreg Clayton     m_prev_active_window_idx = UINT32_MAX;
107244d93782SGreg Clayton     for (Windows::iterator pos = m_subwindows.begin();
1073b9c1b51eSKate Stone          pos != m_subwindows.end(); pos = m_subwindows.erase(pos)) {
107444d93782SGreg Clayton       (*pos)->Erase();
107544d93782SGreg Clayton     }
107644d93782SGreg Clayton     if (m_parent)
107744d93782SGreg Clayton       m_parent->Touch();
107844d93782SGreg Clayton     else
107944d93782SGreg Clayton       ::touchwin(stdscr);
108044d93782SGreg Clayton   }
108144d93782SGreg Clayton 
1082b9c1b51eSKate Stone   WINDOW *get() { return m_window; }
108344d93782SGreg Clayton 
1084b9c1b51eSKate Stone   operator WINDOW *() { return m_window; }
108544d93782SGreg Clayton 
108644d93782SGreg Clayton   //----------------------------------------------------------------------
108744d93782SGreg Clayton   // Window drawing utilities
108844d93782SGreg Clayton   //----------------------------------------------------------------------
1089b9c1b51eSKate Stone   void DrawTitleBox(const char *title, const char *bottom_message = nullptr) {
109044d93782SGreg Clayton     attr_t attr = 0;
109144d93782SGreg Clayton     if (IsActive())
109244d93782SGreg Clayton       attr = A_BOLD | COLOR_PAIR(2);
109344d93782SGreg Clayton     else
109444d93782SGreg Clayton       attr = 0;
109544d93782SGreg Clayton     if (attr)
109644d93782SGreg Clayton       AttributeOn(attr);
109744d93782SGreg Clayton 
109844d93782SGreg Clayton     Box();
109944d93782SGreg Clayton     MoveCursor(3, 0);
110044d93782SGreg Clayton 
1101b9c1b51eSKate Stone     if (title && title[0]) {
110244d93782SGreg Clayton       PutChar('<');
110344d93782SGreg Clayton       PutCString(title);
110444d93782SGreg Clayton       PutChar('>');
110544d93782SGreg Clayton     }
110644d93782SGreg Clayton 
1107b9c1b51eSKate Stone     if (bottom_message && bottom_message[0]) {
110844d93782SGreg Clayton       int bottom_message_length = strlen(bottom_message);
110944d93782SGreg Clayton       int x = GetWidth() - 3 - (bottom_message_length + 2);
111044d93782SGreg Clayton 
1111b9c1b51eSKate Stone       if (x > 0) {
111244d93782SGreg Clayton         MoveCursor(x, GetHeight() - 1);
111344d93782SGreg Clayton         PutChar('[');
111444d93782SGreg Clayton         PutCString(bottom_message);
111544d93782SGreg Clayton         PutChar(']');
1116b9c1b51eSKate Stone       } else {
111744d93782SGreg Clayton         MoveCursor(1, GetHeight() - 1);
111844d93782SGreg Clayton         PutChar('[');
111944d93782SGreg Clayton         PutCStringTruncated(bottom_message, 1);
112044d93782SGreg Clayton       }
112144d93782SGreg Clayton     }
112244d93782SGreg Clayton     if (attr)
112344d93782SGreg Clayton       AttributeOff(attr);
112444d93782SGreg Clayton   }
112544d93782SGreg Clayton 
1126b9c1b51eSKate Stone   virtual void Draw(bool force) {
112744d93782SGreg Clayton     if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw(*this, force))
112844d93782SGreg Clayton       return;
112944d93782SGreg Clayton 
113044d93782SGreg Clayton     for (auto &subwindow_sp : m_subwindows)
113144d93782SGreg Clayton       subwindow_sp->Draw(force);
113244d93782SGreg Clayton   }
113344d93782SGreg Clayton 
1134b9c1b51eSKate Stone   bool CreateHelpSubwindow() {
1135b9c1b51eSKate Stone     if (m_delegate_sp) {
113644d93782SGreg Clayton       const char *text = m_delegate_sp->WindowDelegateGetHelpText();
113744d93782SGreg Clayton       KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp();
1138b9c1b51eSKate Stone       if ((text && text[0]) || key_help) {
1139e1a6074bSBenjamin Kramer         std::unique_ptr<HelpDialogDelegate> help_delegate_ap(
1140b9c1b51eSKate Stone             new HelpDialogDelegate(text, key_help));
114144d93782SGreg Clayton         const size_t num_lines = help_delegate_ap->GetNumLines();
114244d93782SGreg Clayton         const size_t max_length = help_delegate_ap->GetMaxLineLength();
114344d93782SGreg Clayton         Rect bounds = GetBounds();
114444d93782SGreg Clayton         bounds.Inset(1, 1);
1145b9c1b51eSKate Stone         if (max_length + 4 < static_cast<size_t>(bounds.size.width)) {
114644d93782SGreg Clayton           bounds.origin.x += (bounds.size.width - max_length + 4) / 2;
114744d93782SGreg Clayton           bounds.size.width = max_length + 4;
1148b9c1b51eSKate Stone         } else {
1149b9c1b51eSKate Stone           if (bounds.size.width > 100) {
115044d93782SGreg Clayton             const int inset_w = bounds.size.width / 4;
115144d93782SGreg Clayton             bounds.origin.x += inset_w;
115244d93782SGreg Clayton             bounds.size.width -= 2 * inset_w;
115344d93782SGreg Clayton           }
115444d93782SGreg Clayton         }
115544d93782SGreg Clayton 
1156b9c1b51eSKate Stone         if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) {
115744d93782SGreg Clayton           bounds.origin.y += (bounds.size.height - num_lines + 2) / 2;
115844d93782SGreg Clayton           bounds.size.height = num_lines + 2;
1159b9c1b51eSKate Stone         } else {
1160b9c1b51eSKate Stone           if (bounds.size.height > 100) {
116144d93782SGreg Clayton             const int inset_h = bounds.size.height / 4;
116244d93782SGreg Clayton             bounds.origin.y += inset_h;
116344d93782SGreg Clayton             bounds.size.height -= 2 * inset_h;
116444d93782SGreg Clayton           }
116544d93782SGreg Clayton         }
11665fdb09bbSGreg Clayton         WindowSP help_window_sp;
11675fdb09bbSGreg Clayton         Window *parent_window = GetParent();
11685fdb09bbSGreg Clayton         if (parent_window)
11695fdb09bbSGreg Clayton           help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
11705fdb09bbSGreg Clayton         else
11715fdb09bbSGreg Clayton           help_window_sp = CreateSubWindow("Help", bounds, true);
1172b9c1b51eSKate Stone         help_window_sp->SetDelegate(
1173b9c1b51eSKate Stone             WindowDelegateSP(help_delegate_ap.release()));
117444d93782SGreg Clayton         return true;
117544d93782SGreg Clayton       }
117644d93782SGreg Clayton     }
117744d93782SGreg Clayton     return false;
117844d93782SGreg Clayton   }
117944d93782SGreg Clayton 
1180b9c1b51eSKate Stone   virtual HandleCharResult HandleChar(int key) {
118144d93782SGreg Clayton     // Always check the active window first
118244d93782SGreg Clayton     HandleCharResult result = eKeyNotHandled;
118344d93782SGreg Clayton     WindowSP active_window_sp = GetActiveWindow();
1184b9c1b51eSKate Stone     if (active_window_sp) {
118544d93782SGreg Clayton       result = active_window_sp->HandleChar(key);
118644d93782SGreg Clayton       if (result != eKeyNotHandled)
118744d93782SGreg Clayton         return result;
118844d93782SGreg Clayton     }
118944d93782SGreg Clayton 
1190b9c1b51eSKate Stone     if (m_delegate_sp) {
119144d93782SGreg Clayton       result = m_delegate_sp->WindowDelegateHandleChar(*this, key);
119244d93782SGreg Clayton       if (result != eKeyNotHandled)
119344d93782SGreg Clayton         return result;
119444d93782SGreg Clayton     }
119544d93782SGreg Clayton 
119605097246SAdrian Prantl     // Then check for any windows that want any keys that weren't handled. This
119705097246SAdrian Prantl     // is typically only for a menubar. Make a copy of the subwindows in case
119805097246SAdrian Prantl     // any HandleChar() functions muck with the subwindows. If we don't do
119905097246SAdrian Prantl     // this, we can crash when iterating over the subwindows.
120044d93782SGreg Clayton     Windows subwindows(m_subwindows);
1201b9c1b51eSKate Stone     for (auto subwindow_sp : subwindows) {
1202b9c1b51eSKate Stone       if (!subwindow_sp->m_can_activate) {
120344d93782SGreg Clayton         HandleCharResult result = subwindow_sp->HandleChar(key);
120444d93782SGreg Clayton         if (result != eKeyNotHandled)
120544d93782SGreg Clayton           return result;
120644d93782SGreg Clayton       }
120744d93782SGreg Clayton     }
120844d93782SGreg Clayton 
120944d93782SGreg Clayton     return eKeyNotHandled;
121044d93782SGreg Clayton   }
121144d93782SGreg Clayton 
1212b9c1b51eSKate Stone   bool SetActiveWindow(Window *window) {
121344d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1214b9c1b51eSKate Stone     for (size_t i = 0; i < num_subwindows; ++i) {
1215b9c1b51eSKate Stone       if (m_subwindows[i].get() == window) {
121644d93782SGreg Clayton         m_prev_active_window_idx = m_curr_active_window_idx;
121744d93782SGreg Clayton         ::top_panel(window->m_panel);
121844d93782SGreg Clayton         m_curr_active_window_idx = i;
121944d93782SGreg Clayton         return true;
122044d93782SGreg Clayton       }
122144d93782SGreg Clayton     }
122244d93782SGreg Clayton     return false;
122344d93782SGreg Clayton   }
122444d93782SGreg Clayton 
1225b9c1b51eSKate Stone   WindowSP GetActiveWindow() {
1226b9c1b51eSKate Stone     if (!m_subwindows.empty()) {
1227b9c1b51eSKate Stone       if (m_curr_active_window_idx >= m_subwindows.size()) {
1228b9c1b51eSKate Stone         if (m_prev_active_window_idx < m_subwindows.size()) {
122944d93782SGreg Clayton           m_curr_active_window_idx = m_prev_active_window_idx;
123044d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1231b9c1b51eSKate Stone         } else if (IsActive()) {
123244d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
123344d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
123444d93782SGreg Clayton 
123544d93782SGreg Clayton           // Find first window that wants to be active if this window is active
123644d93782SGreg Clayton           const size_t num_subwindows = m_subwindows.size();
1237b9c1b51eSKate Stone           for (size_t i = 0; i < num_subwindows; ++i) {
1238b9c1b51eSKate Stone             if (m_subwindows[i]->GetCanBeActive()) {
123944d93782SGreg Clayton               m_curr_active_window_idx = i;
124044d93782SGreg Clayton               break;
124144d93782SGreg Clayton             }
124244d93782SGreg Clayton           }
124344d93782SGreg Clayton         }
124444d93782SGreg Clayton       }
124544d93782SGreg Clayton 
124644d93782SGreg Clayton       if (m_curr_active_window_idx < m_subwindows.size())
124744d93782SGreg Clayton         return m_subwindows[m_curr_active_window_idx];
124844d93782SGreg Clayton     }
124944d93782SGreg Clayton     return WindowSP();
125044d93782SGreg Clayton   }
125144d93782SGreg Clayton 
1252b9c1b51eSKate Stone   bool GetCanBeActive() const { return m_can_activate; }
125344d93782SGreg Clayton 
1254b9c1b51eSKate Stone   void SetCanBeActive(bool b) { m_can_activate = b; }
125544d93782SGreg Clayton 
1256b9c1b51eSKate Stone   const WindowDelegateSP &GetDelegate() const { return m_delegate_sp; }
125744d93782SGreg Clayton 
1258b9c1b51eSKate Stone   void SetDelegate(const WindowDelegateSP &delegate_sp) {
125944d93782SGreg Clayton     m_delegate_sp = delegate_sp;
126044d93782SGreg Clayton   }
126144d93782SGreg Clayton 
1262b9c1b51eSKate Stone   Window *GetParent() const { return m_parent; }
126344d93782SGreg Clayton 
1264b9c1b51eSKate Stone   bool IsActive() const {
126544d93782SGreg Clayton     if (m_parent)
126644d93782SGreg Clayton       return m_parent->GetActiveWindow().get() == this;
126744d93782SGreg Clayton     else
126844d93782SGreg Clayton       return true; // Top level window is always active
126944d93782SGreg Clayton   }
127044d93782SGreg Clayton 
1271b9c1b51eSKate Stone   void SelectNextWindowAsActive() {
127244d93782SGreg Clayton     // Move active focus to next window
127344d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1274b9c1b51eSKate Stone     if (m_curr_active_window_idx == UINT32_MAX) {
127544d93782SGreg Clayton       uint32_t idx = 0;
1276b9c1b51eSKate Stone       for (auto subwindow_sp : m_subwindows) {
1277b9c1b51eSKate Stone         if (subwindow_sp->GetCanBeActive()) {
127844d93782SGreg Clayton           m_curr_active_window_idx = idx;
127944d93782SGreg Clayton           break;
128044d93782SGreg Clayton         }
128144d93782SGreg Clayton         ++idx;
128244d93782SGreg Clayton       }
1283b9c1b51eSKate Stone     } else if (m_curr_active_window_idx + 1 < num_subwindows) {
128444d93782SGreg Clayton       bool handled = false;
128544d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1286b9c1b51eSKate Stone       for (size_t idx = m_curr_active_window_idx + 1; idx < num_subwindows;
1287b9c1b51eSKate Stone            ++idx) {
1288b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
128944d93782SGreg Clayton           m_curr_active_window_idx = idx;
129044d93782SGreg Clayton           handled = true;
129144d93782SGreg Clayton           break;
129244d93782SGreg Clayton         }
129344d93782SGreg Clayton       }
1294b9c1b51eSKate Stone       if (!handled) {
1295b9c1b51eSKate Stone         for (size_t idx = 0; idx <= m_prev_active_window_idx; ++idx) {
1296b9c1b51eSKate Stone           if (m_subwindows[idx]->GetCanBeActive()) {
129744d93782SGreg Clayton             m_curr_active_window_idx = idx;
129844d93782SGreg Clayton             break;
129944d93782SGreg Clayton           }
130044d93782SGreg Clayton         }
130144d93782SGreg Clayton       }
1302b9c1b51eSKate Stone     } else {
130344d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1304b9c1b51eSKate Stone       for (size_t idx = 0; idx < num_subwindows; ++idx) {
1305b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
130644d93782SGreg Clayton           m_curr_active_window_idx = idx;
130744d93782SGreg Clayton           break;
130844d93782SGreg Clayton         }
130944d93782SGreg Clayton       }
131044d93782SGreg Clayton     }
131144d93782SGreg Clayton   }
131244d93782SGreg Clayton 
1313b9c1b51eSKate Stone   const char *GetName() const { return m_name.c_str(); }
1314315b6884SEugene Zelenko 
131544d93782SGreg Clayton protected:
131644d93782SGreg Clayton   std::string m_name;
131744d93782SGreg Clayton   WINDOW *m_window;
131844d93782SGreg Clayton   PANEL *m_panel;
131944d93782SGreg Clayton   Window *m_parent;
132044d93782SGreg Clayton   Windows m_subwindows;
132144d93782SGreg Clayton   WindowDelegateSP m_delegate_sp;
132244d93782SGreg Clayton   uint32_t m_curr_active_window_idx;
132344d93782SGreg Clayton   uint32_t m_prev_active_window_idx;
132444d93782SGreg Clayton   bool m_delete;
132544d93782SGreg Clayton   bool m_needs_update;
132644d93782SGreg Clayton   bool m_can_activate;
132744d93782SGreg Clayton   bool m_is_subwin;
132844d93782SGreg Clayton 
132944d93782SGreg Clayton private:
133044d93782SGreg Clayton   DISALLOW_COPY_AND_ASSIGN(Window);
133144d93782SGreg Clayton };
133244d93782SGreg Clayton 
1333b9c1b51eSKate Stone class MenuDelegate {
133444d93782SGreg Clayton public:
1335315b6884SEugene Zelenko   virtual ~MenuDelegate() = default;
133644d93782SGreg Clayton 
1337b9c1b51eSKate Stone   virtual MenuActionResult MenuDelegateAction(Menu &menu) = 0;
133844d93782SGreg Clayton };
133944d93782SGreg Clayton 
1340b9c1b51eSKate Stone class Menu : public WindowDelegate {
134144d93782SGreg Clayton public:
1342b9c1b51eSKate Stone   enum class Type { Invalid, Bar, Item, Separator };
134344d93782SGreg Clayton 
134444d93782SGreg Clayton   // Menubar or separator constructor
134544d93782SGreg Clayton   Menu(Type type);
134644d93782SGreg Clayton 
134744d93782SGreg Clayton   // Menuitem constructor
1348b9c1b51eSKate Stone   Menu(const char *name, const char *key_name, int key_value,
134944d93782SGreg Clayton        uint64_t identifier);
135044d93782SGreg Clayton 
1351315b6884SEugene Zelenko   ~Menu() override = default;
135244d93782SGreg Clayton 
1353b9c1b51eSKate Stone   const MenuDelegateSP &GetDelegate() const { return m_delegate_sp; }
135444d93782SGreg Clayton 
1355b9c1b51eSKate Stone   void SetDelegate(const MenuDelegateSP &delegate_sp) {
135644d93782SGreg Clayton     m_delegate_sp = delegate_sp;
135744d93782SGreg Clayton   }
135844d93782SGreg Clayton 
1359b9c1b51eSKate Stone   void RecalculateNameLengths();
136044d93782SGreg Clayton 
1361b9c1b51eSKate Stone   void AddSubmenu(const MenuSP &menu_sp);
136244d93782SGreg Clayton 
1363b9c1b51eSKate Stone   int DrawAndRunMenu(Window &window);
136444d93782SGreg Clayton 
1365b9c1b51eSKate Stone   void DrawMenuTitle(Window &window, bool highlight);
136644d93782SGreg Clayton 
1367b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
136844d93782SGreg Clayton 
1369b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
137044d93782SGreg Clayton 
1371b9c1b51eSKate Stone   MenuActionResult ActionPrivate(Menu &menu) {
137244d93782SGreg Clayton     MenuActionResult result = MenuActionResult::NotHandled;
1373b9c1b51eSKate Stone     if (m_delegate_sp) {
137444d93782SGreg Clayton       result = m_delegate_sp->MenuDelegateAction(menu);
137544d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
137644d93782SGreg Clayton         return result;
1377b9c1b51eSKate Stone     } else if (m_parent) {
137844d93782SGreg Clayton       result = m_parent->ActionPrivate(menu);
137944d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
138044d93782SGreg Clayton         return result;
138144d93782SGreg Clayton     }
138244d93782SGreg Clayton     return m_canned_result;
138344d93782SGreg Clayton   }
138444d93782SGreg Clayton 
1385b9c1b51eSKate Stone   MenuActionResult Action() {
138605097246SAdrian Prantl     // Call the recursive action so it can try to handle it with the menu
138705097246SAdrian Prantl     // delegate, and if not, try our parent menu
138844d93782SGreg Clayton     return ActionPrivate(*this);
138944d93782SGreg Clayton   }
139044d93782SGreg Clayton 
1391b9c1b51eSKate Stone   void SetCannedResult(MenuActionResult result) { m_canned_result = result; }
139244d93782SGreg Clayton 
1393b9c1b51eSKate Stone   Menus &GetSubmenus() { return m_submenus; }
139444d93782SGreg Clayton 
1395b9c1b51eSKate Stone   const Menus &GetSubmenus() const { return m_submenus; }
139644d93782SGreg Clayton 
1397b9c1b51eSKate Stone   int GetSelectedSubmenuIndex() const { return m_selected; }
139844d93782SGreg Clayton 
1399b9c1b51eSKate Stone   void SetSelectedSubmenuIndex(int idx) { m_selected = idx; }
140044d93782SGreg Clayton 
1401b9c1b51eSKate Stone   Type GetType() const { return m_type; }
140244d93782SGreg Clayton 
1403b9c1b51eSKate Stone   int GetStartingColumn() const { return m_start_col; }
140444d93782SGreg Clayton 
1405b9c1b51eSKate Stone   void SetStartingColumn(int col) { m_start_col = col; }
140644d93782SGreg Clayton 
1407b9c1b51eSKate Stone   int GetKeyValue() const { return m_key_value; }
140844d93782SGreg Clayton 
1409b9c1b51eSKate Stone   void SetKeyValue(int key_value) { m_key_value = key_value; }
141044d93782SGreg Clayton 
1411b9c1b51eSKate Stone   std::string &GetName() { return m_name; }
141244d93782SGreg Clayton 
1413b9c1b51eSKate Stone   std::string &GetKeyName() { return m_key_name; }
141444d93782SGreg Clayton 
1415b9c1b51eSKate Stone   int GetDrawWidth() const {
141644d93782SGreg Clayton     return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
141744d93782SGreg Clayton   }
141844d93782SGreg Clayton 
1419b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
142044d93782SGreg Clayton 
1421b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
142244d93782SGreg Clayton 
142344d93782SGreg Clayton protected:
142444d93782SGreg Clayton   std::string m_name;
142544d93782SGreg Clayton   std::string m_key_name;
142644d93782SGreg Clayton   uint64_t m_identifier;
142744d93782SGreg Clayton   Type m_type;
142844d93782SGreg Clayton   int m_key_value;
142944d93782SGreg Clayton   int m_start_col;
143044d93782SGreg Clayton   int m_max_submenu_name_length;
143144d93782SGreg Clayton   int m_max_submenu_key_name_length;
143244d93782SGreg Clayton   int m_selected;
143344d93782SGreg Clayton   Menu *m_parent;
143444d93782SGreg Clayton   Menus m_submenus;
143544d93782SGreg Clayton   WindowSP m_menu_window_sp;
143644d93782SGreg Clayton   MenuActionResult m_canned_result;
143744d93782SGreg Clayton   MenuDelegateSP m_delegate_sp;
143844d93782SGreg Clayton };
143944d93782SGreg Clayton 
144044d93782SGreg Clayton // Menubar or separator constructor
1441b9c1b51eSKate Stone Menu::Menu(Type type)
1442b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(0), m_type(type), m_key_value(0),
1443b9c1b51eSKate Stone       m_start_col(0), m_max_submenu_name_length(0),
1444b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1445b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1446b9c1b51eSKate Stone       m_delegate_sp() {}
144744d93782SGreg Clayton 
144844d93782SGreg Clayton // Menuitem constructor
1449b9c1b51eSKate Stone Menu::Menu(const char *name, const char *key_name, int key_value,
1450b9c1b51eSKate Stone            uint64_t identifier)
1451b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(identifier), m_type(Type::Invalid),
1452b9c1b51eSKate Stone       m_key_value(key_value), m_start_col(0), m_max_submenu_name_length(0),
1453b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1454b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1455b9c1b51eSKate Stone       m_delegate_sp() {
1456b9c1b51eSKate Stone   if (name && name[0]) {
145744d93782SGreg Clayton     m_name = name;
145844d93782SGreg Clayton     m_type = Type::Item;
145944d93782SGreg Clayton     if (key_name && key_name[0])
146044d93782SGreg Clayton       m_key_name = key_name;
1461b9c1b51eSKate Stone   } else {
146244d93782SGreg Clayton     m_type = Type::Separator;
146344d93782SGreg Clayton   }
146444d93782SGreg Clayton }
146544d93782SGreg Clayton 
1466b9c1b51eSKate Stone void Menu::RecalculateNameLengths() {
146744d93782SGreg Clayton   m_max_submenu_name_length = 0;
146844d93782SGreg Clayton   m_max_submenu_key_name_length = 0;
146944d93782SGreg Clayton   Menus &submenus = GetSubmenus();
147044d93782SGreg Clayton   const size_t num_submenus = submenus.size();
1471b9c1b51eSKate Stone   for (size_t i = 0; i < num_submenus; ++i) {
147244d93782SGreg Clayton     Menu *submenu = submenus[i].get();
14733985c8c6SSaleem Abdulrasool     if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
147444d93782SGreg Clayton       m_max_submenu_name_length = submenu->m_name.size();
1475b9c1b51eSKate Stone     if (static_cast<size_t>(m_max_submenu_key_name_length) <
1476b9c1b51eSKate Stone         submenu->m_key_name.size())
147744d93782SGreg Clayton       m_max_submenu_key_name_length = submenu->m_key_name.size();
147844d93782SGreg Clayton   }
147944d93782SGreg Clayton }
148044d93782SGreg Clayton 
1481b9c1b51eSKate Stone void Menu::AddSubmenu(const MenuSP &menu_sp) {
148244d93782SGreg Clayton   menu_sp->m_parent = this;
14833985c8c6SSaleem Abdulrasool   if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
148444d93782SGreg Clayton     m_max_submenu_name_length = menu_sp->m_name.size();
1485b9c1b51eSKate Stone   if (static_cast<size_t>(m_max_submenu_key_name_length) <
1486b9c1b51eSKate Stone       menu_sp->m_key_name.size())
148744d93782SGreg Clayton     m_max_submenu_key_name_length = menu_sp->m_key_name.size();
148844d93782SGreg Clayton   m_submenus.push_back(menu_sp);
148944d93782SGreg Clayton }
149044d93782SGreg Clayton 
1491b9c1b51eSKate Stone void Menu::DrawMenuTitle(Window &window, bool highlight) {
1492b9c1b51eSKate Stone   if (m_type == Type::Separator) {
149344d93782SGreg Clayton     window.MoveCursor(0, window.GetCursorY());
149444d93782SGreg Clayton     window.PutChar(ACS_LTEE);
149544d93782SGreg Clayton     int width = window.GetWidth();
1496b9c1b51eSKate Stone     if (width > 2) {
149744d93782SGreg Clayton       width -= 2;
14983985c8c6SSaleem Abdulrasool       for (int i = 0; i < width; ++i)
149944d93782SGreg Clayton         window.PutChar(ACS_HLINE);
150044d93782SGreg Clayton     }
150144d93782SGreg Clayton     window.PutChar(ACS_RTEE);
1502b9c1b51eSKate Stone   } else {
150344d93782SGreg Clayton     const int shortcut_key = m_key_value;
150444d93782SGreg Clayton     bool underlined_shortcut = false;
150544d93782SGreg Clayton     const attr_t hilgight_attr = A_REVERSE;
150644d93782SGreg Clayton     if (highlight)
150744d93782SGreg Clayton       window.AttributeOn(hilgight_attr);
1508b9c1b51eSKate Stone     if (isprint(shortcut_key)) {
150944d93782SGreg Clayton       size_t lower_pos = m_name.find(tolower(shortcut_key));
151044d93782SGreg Clayton       size_t upper_pos = m_name.find(toupper(shortcut_key));
151144d93782SGreg Clayton       const char *name = m_name.c_str();
151244d93782SGreg Clayton       size_t pos = std::min<size_t>(lower_pos, upper_pos);
1513b9c1b51eSKate Stone       if (pos != std::string::npos) {
151444d93782SGreg Clayton         underlined_shortcut = true;
1515b9c1b51eSKate Stone         if (pos > 0) {
151644d93782SGreg Clayton           window.PutCString(name, pos);
151744d93782SGreg Clayton           name += pos;
151844d93782SGreg Clayton         }
151944d93782SGreg Clayton         const attr_t shortcut_attr = A_UNDERLINE | A_BOLD;
152044d93782SGreg Clayton         window.AttributeOn(shortcut_attr);
152144d93782SGreg Clayton         window.PutChar(name[0]);
152244d93782SGreg Clayton         window.AttributeOff(shortcut_attr);
152344d93782SGreg Clayton         name++;
152444d93782SGreg Clayton         if (name[0])
152544d93782SGreg Clayton           window.PutCString(name);
152644d93782SGreg Clayton       }
152744d93782SGreg Clayton     }
152844d93782SGreg Clayton 
1529b9c1b51eSKate Stone     if (!underlined_shortcut) {
153044d93782SGreg Clayton       window.PutCString(m_name.c_str());
153144d93782SGreg Clayton     }
153244d93782SGreg Clayton 
153344d93782SGreg Clayton     if (highlight)
153444d93782SGreg Clayton       window.AttributeOff(hilgight_attr);
153544d93782SGreg Clayton 
1536b9c1b51eSKate Stone     if (m_key_name.empty()) {
1537b9c1b51eSKate Stone       if (!underlined_shortcut && isprint(m_key_value)) {
153844d93782SGreg Clayton         window.AttributeOn(COLOR_PAIR(3));
153944d93782SGreg Clayton         window.Printf(" (%c)", m_key_value);
154044d93782SGreg Clayton         window.AttributeOff(COLOR_PAIR(3));
154144d93782SGreg Clayton       }
1542b9c1b51eSKate Stone     } else {
154344d93782SGreg Clayton       window.AttributeOn(COLOR_PAIR(3));
154444d93782SGreg Clayton       window.Printf(" (%s)", m_key_name.c_str());
154544d93782SGreg Clayton       window.AttributeOff(COLOR_PAIR(3));
154644d93782SGreg Clayton     }
154744d93782SGreg Clayton   }
154844d93782SGreg Clayton }
154944d93782SGreg Clayton 
1550b9c1b51eSKate Stone bool Menu::WindowDelegateDraw(Window &window, bool force) {
155144d93782SGreg Clayton   Menus &submenus = GetSubmenus();
155244d93782SGreg Clayton   const size_t num_submenus = submenus.size();
155344d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
155444d93782SGreg Clayton   Menu::Type menu_type = GetType();
1555b9c1b51eSKate Stone   switch (menu_type) {
1556b9c1b51eSKate Stone   case Menu::Type::Bar: {
155744d93782SGreg Clayton     window.SetBackground(2);
155844d93782SGreg Clayton     window.MoveCursor(0, 0);
1559b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
156044d93782SGreg Clayton       Menu *menu = submenus[i].get();
156144d93782SGreg Clayton       if (i > 0)
156244d93782SGreg Clayton         window.PutChar(' ');
156344d93782SGreg Clayton       menu->SetStartingColumn(window.GetCursorX());
156444d93782SGreg Clayton       window.PutCString("| ");
156544d93782SGreg Clayton       menu->DrawMenuTitle(window, false);
156644d93782SGreg Clayton     }
156744d93782SGreg Clayton     window.PutCString(" |");
156844d93782SGreg Clayton     window.DeferredRefresh();
1569b9c1b51eSKate Stone   } break;
157044d93782SGreg Clayton 
1571b9c1b51eSKate Stone   case Menu::Type::Item: {
157244d93782SGreg Clayton     int y = 1;
157344d93782SGreg Clayton     int x = 3;
157444d93782SGreg Clayton     // Draw the menu
157544d93782SGreg Clayton     int cursor_x = 0;
157644d93782SGreg Clayton     int cursor_y = 0;
157744d93782SGreg Clayton     window.Erase();
157844d93782SGreg Clayton     window.SetBackground(2);
157944d93782SGreg Clayton     window.Box();
1580b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
1581b9c1b51eSKate Stone       const bool is_selected = (i == static_cast<size_t>(selected_idx));
158244d93782SGreg Clayton       window.MoveCursor(x, y + i);
1583b9c1b51eSKate Stone       if (is_selected) {
158444d93782SGreg Clayton         // Remember where we want the cursor to be
158544d93782SGreg Clayton         cursor_x = x - 1;
158644d93782SGreg Clayton         cursor_y = y + i;
158744d93782SGreg Clayton       }
158844d93782SGreg Clayton       submenus[i]->DrawMenuTitle(window, is_selected);
158944d93782SGreg Clayton     }
159044d93782SGreg Clayton     window.MoveCursor(cursor_x, cursor_y);
159144d93782SGreg Clayton     window.DeferredRefresh();
1592b9c1b51eSKate Stone   } break;
159344d93782SGreg Clayton 
159444d93782SGreg Clayton   default:
159544d93782SGreg Clayton   case Menu::Type::Separator:
159644d93782SGreg Clayton     break;
159744d93782SGreg Clayton   }
159844d93782SGreg Clayton   return true; // Drawing handled...
159944d93782SGreg Clayton }
160044d93782SGreg Clayton 
1601b9c1b51eSKate Stone HandleCharResult Menu::WindowDelegateHandleChar(Window &window, int key) {
160244d93782SGreg Clayton   HandleCharResult result = eKeyNotHandled;
160344d93782SGreg Clayton 
160444d93782SGreg Clayton   Menus &submenus = GetSubmenus();
160544d93782SGreg Clayton   const size_t num_submenus = submenus.size();
160644d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
160744d93782SGreg Clayton   Menu::Type menu_type = GetType();
1608b9c1b51eSKate Stone   if (menu_type == Menu::Type::Bar) {
160944d93782SGreg Clayton     MenuSP run_menu_sp;
1610b9c1b51eSKate Stone     switch (key) {
161144d93782SGreg Clayton     case KEY_DOWN:
161244d93782SGreg Clayton     case KEY_UP:
161344d93782SGreg Clayton       // Show last menu or first menu
16143985c8c6SSaleem Abdulrasool       if (selected_idx < static_cast<int>(num_submenus))
161544d93782SGreg Clayton         run_menu_sp = submenus[selected_idx];
161644d93782SGreg Clayton       else if (!submenus.empty())
161744d93782SGreg Clayton         run_menu_sp = submenus.front();
161844d93782SGreg Clayton       result = eKeyHandled;
161944d93782SGreg Clayton       break;
162044d93782SGreg Clayton 
162144d93782SGreg Clayton     case KEY_RIGHT:
162244d93782SGreg Clayton       ++m_selected;
16233985c8c6SSaleem Abdulrasool       if (m_selected >= static_cast<int>(num_submenus))
162444d93782SGreg Clayton         m_selected = 0;
16253985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
162644d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
162744d93782SGreg Clayton       else if (!submenus.empty())
162844d93782SGreg Clayton         run_menu_sp = submenus.front();
162944d93782SGreg Clayton       result = eKeyHandled;
163044d93782SGreg Clayton       break;
163144d93782SGreg Clayton 
163244d93782SGreg Clayton     case KEY_LEFT:
163344d93782SGreg Clayton       --m_selected;
163444d93782SGreg Clayton       if (m_selected < 0)
163544d93782SGreg Clayton         m_selected = num_submenus - 1;
16363985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
163744d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
163844d93782SGreg Clayton       else if (!submenus.empty())
163944d93782SGreg Clayton         run_menu_sp = submenus.front();
164044d93782SGreg Clayton       result = eKeyHandled;
164144d93782SGreg Clayton       break;
164244d93782SGreg Clayton 
164344d93782SGreg Clayton     default:
1644b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
1645b9c1b51eSKate Stone         if (submenus[i]->GetKeyValue() == key) {
164644d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
164744d93782SGreg Clayton           run_menu_sp = submenus[i];
164844d93782SGreg Clayton           result = eKeyHandled;
164944d93782SGreg Clayton           break;
165044d93782SGreg Clayton         }
165144d93782SGreg Clayton       }
165244d93782SGreg Clayton       break;
165344d93782SGreg Clayton     }
165444d93782SGreg Clayton 
1655b9c1b51eSKate Stone     if (run_menu_sp) {
165605097246SAdrian Prantl       // Run the action on this menu in case we need to populate the menu with
165705097246SAdrian Prantl       // dynamic content and also in case check marks, and any other menu
165805097246SAdrian Prantl       // decorations need to be calculated
165944d93782SGreg Clayton       if (run_menu_sp->Action() == MenuActionResult::Quit)
166044d93782SGreg Clayton         return eQuitApplication;
166144d93782SGreg Clayton 
166244d93782SGreg Clayton       Rect menu_bounds;
166344d93782SGreg Clayton       menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
166444d93782SGreg Clayton       menu_bounds.origin.y = 1;
166544d93782SGreg Clayton       menu_bounds.size.width = run_menu_sp->GetDrawWidth();
166644d93782SGreg Clayton       menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
166744d93782SGreg Clayton       if (m_menu_window_sp)
166844d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
166944d93782SGreg Clayton 
1670b9c1b51eSKate Stone       m_menu_window_sp = window.GetParent()->CreateSubWindow(
1671b9c1b51eSKate Stone           run_menu_sp->GetName().c_str(), menu_bounds, true);
167244d93782SGreg Clayton       m_menu_window_sp->SetDelegate(run_menu_sp);
167344d93782SGreg Clayton     }
1674b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Item) {
1675b9c1b51eSKate Stone     switch (key) {
167644d93782SGreg Clayton     case KEY_DOWN:
1677b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
167844d93782SGreg Clayton         const int start_select = m_selected;
1679b9c1b51eSKate Stone         while (++m_selected != start_select) {
16803985c8c6SSaleem Abdulrasool           if (static_cast<size_t>(m_selected) >= num_submenus)
168144d93782SGreg Clayton             m_selected = 0;
168244d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
168344d93782SGreg Clayton             continue;
168444d93782SGreg Clayton           else
168544d93782SGreg Clayton             break;
168644d93782SGreg Clayton         }
168744d93782SGreg Clayton         return eKeyHandled;
168844d93782SGreg Clayton       }
168944d93782SGreg Clayton       break;
169044d93782SGreg Clayton 
169144d93782SGreg Clayton     case KEY_UP:
1692b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
169344d93782SGreg Clayton         const int start_select = m_selected;
1694b9c1b51eSKate Stone         while (--m_selected != start_select) {
16953985c8c6SSaleem Abdulrasool           if (m_selected < static_cast<int>(0))
169644d93782SGreg Clayton             m_selected = num_submenus - 1;
169744d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
169844d93782SGreg Clayton             continue;
169944d93782SGreg Clayton           else
170044d93782SGreg Clayton             break;
170144d93782SGreg Clayton         }
170244d93782SGreg Clayton         return eKeyHandled;
170344d93782SGreg Clayton       }
170444d93782SGreg Clayton       break;
170544d93782SGreg Clayton 
170644d93782SGreg Clayton     case KEY_RETURN:
1707b9c1b51eSKate Stone       if (static_cast<size_t>(selected_idx) < num_submenus) {
170844d93782SGreg Clayton         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
170944d93782SGreg Clayton           return eQuitApplication;
171044d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
171144d93782SGreg Clayton         return eKeyHandled;
171244d93782SGreg Clayton       }
171344d93782SGreg Clayton       break;
171444d93782SGreg Clayton 
1715b9c1b51eSKate Stone     case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in
1716b9c1b51eSKate Stone                      // case other chars are entered for escaped sequences
171744d93782SGreg Clayton       window.GetParent()->RemoveSubWindow(&window);
171844d93782SGreg Clayton       return eKeyHandled;
171944d93782SGreg Clayton 
172044d93782SGreg Clayton     default:
1721b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
172244d93782SGreg Clayton         Menu *menu = submenus[i].get();
1723b9c1b51eSKate Stone         if (menu->GetKeyValue() == key) {
172444d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
172544d93782SGreg Clayton           window.GetParent()->RemoveSubWindow(&window);
172644d93782SGreg Clayton           if (menu->Action() == MenuActionResult::Quit)
172744d93782SGreg Clayton             return eQuitApplication;
172844d93782SGreg Clayton           return eKeyHandled;
172944d93782SGreg Clayton         }
173044d93782SGreg Clayton       }
173144d93782SGreg Clayton       break;
173244d93782SGreg Clayton     }
1733b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Separator) {
173444d93782SGreg Clayton   }
173544d93782SGreg Clayton   return result;
173644d93782SGreg Clayton }
173744d93782SGreg Clayton 
1738b9c1b51eSKate Stone class Application {
173944d93782SGreg Clayton public:
1740b9c1b51eSKate Stone   Application(FILE *in, FILE *out)
1741b9c1b51eSKate Stone       : m_window_sp(), m_screen(nullptr), m_in(in), m_out(out) {}
174244d93782SGreg Clayton 
1743b9c1b51eSKate Stone   ~Application() {
174444d93782SGreg Clayton     m_window_delegates.clear();
174544d93782SGreg Clayton     m_window_sp.reset();
1746b9c1b51eSKate Stone     if (m_screen) {
174744d93782SGreg Clayton       ::delscreen(m_screen);
1748c5dac77aSEugene Zelenko       m_screen = nullptr;
174944d93782SGreg Clayton     }
175044d93782SGreg Clayton   }
175144d93782SGreg Clayton 
1752b9c1b51eSKate Stone   void Initialize() {
175344d93782SGreg Clayton     ::setlocale(LC_ALL, "");
175444d93782SGreg Clayton     ::setlocale(LC_CTYPE, "");
175544d93782SGreg Clayton #if 0
175644d93782SGreg Clayton             ::initscr();
175744d93782SGreg Clayton #else
1758c5dac77aSEugene Zelenko     m_screen = ::newterm(nullptr, m_out, m_in);
175944d93782SGreg Clayton #endif
176044d93782SGreg Clayton     ::start_color();
176144d93782SGreg Clayton     ::curs_set(0);
176244d93782SGreg Clayton     ::noecho();
176344d93782SGreg Clayton     ::keypad(stdscr, TRUE);
176444d93782SGreg Clayton   }
176544d93782SGreg Clayton 
1766b9c1b51eSKate Stone   void Terminate() { ::endwin(); }
176744d93782SGreg Clayton 
1768b9c1b51eSKate Stone   void Run(Debugger &debugger) {
176944d93782SGreg Clayton     bool done = false;
177044d93782SGreg Clayton     int delay_in_tenths_of_a_second = 1;
177144d93782SGreg Clayton 
177205097246SAdrian Prantl     // Alas the threading model in curses is a bit lame so we need to resort to
177305097246SAdrian Prantl     // polling every 0.5 seconds. We could poll for stdin ourselves and then
177405097246SAdrian Prantl     // pass the keys down but then we need to translate all of the escape
177505097246SAdrian Prantl     // sequences ourselves. So we resort to polling for input because we need
177605097246SAdrian Prantl     // to receive async process events while in this loop.
177744d93782SGreg Clayton 
1778b9c1b51eSKate Stone     halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths
1779b9c1b51eSKate Stone                                             // of seconds seconds when calling
1780b9c1b51eSKate Stone                                             // Window::GetChar()
178144d93782SGreg Clayton 
1782b9c1b51eSKate Stone     ListenerSP listener_sp(
1783b9c1b51eSKate Stone         Listener::MakeListener("lldb.IOHandler.curses.Application"));
178444d93782SGreg Clayton     ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
178544d93782SGreg Clayton     ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
178644d93782SGreg Clayton     ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
178744d93782SGreg Clayton     debugger.EnableForwardEvents(listener_sp);
178844d93782SGreg Clayton 
178944d93782SGreg Clayton     bool update = true;
179044d93782SGreg Clayton #if defined(__APPLE__)
179144d93782SGreg Clayton     std::deque<int> escape_chars;
179244d93782SGreg Clayton #endif
179344d93782SGreg Clayton 
1794b9c1b51eSKate Stone     while (!done) {
1795b9c1b51eSKate Stone       if (update) {
179644d93782SGreg Clayton         m_window_sp->Draw(false);
179705097246SAdrian Prantl         // All windows should be calling Window::DeferredRefresh() instead of
179805097246SAdrian Prantl         // Window::Refresh() so we can do a single update and avoid any screen
179905097246SAdrian Prantl         // blinking
180044d93782SGreg Clayton         update_panels();
180144d93782SGreg Clayton 
1802b9c1b51eSKate Stone         // Cursor hiding isn't working on MacOSX, so hide it in the top left
1803b9c1b51eSKate Stone         // corner
180444d93782SGreg Clayton         m_window_sp->MoveCursor(0, 0);
180544d93782SGreg Clayton 
180644d93782SGreg Clayton         doupdate();
180744d93782SGreg Clayton         update = false;
180844d93782SGreg Clayton       }
180944d93782SGreg Clayton 
181044d93782SGreg Clayton #if defined(__APPLE__)
181105097246SAdrian Prantl       // Terminal.app doesn't map its function keys correctly, F1-F4 default
181205097246SAdrian Prantl       // to: \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if
1813b9c1b51eSKate Stone       // possible
181444d93782SGreg Clayton       int ch;
181544d93782SGreg Clayton       if (escape_chars.empty())
181644d93782SGreg Clayton         ch = m_window_sp->GetChar();
1817b9c1b51eSKate Stone       else {
181844d93782SGreg Clayton         ch = escape_chars.front();
181944d93782SGreg Clayton         escape_chars.pop_front();
182044d93782SGreg Clayton       }
1821b9c1b51eSKate Stone       if (ch == KEY_ESCAPE) {
182244d93782SGreg Clayton         int ch2 = m_window_sp->GetChar();
1823b9c1b51eSKate Stone         if (ch2 == 'O') {
182444d93782SGreg Clayton           int ch3 = m_window_sp->GetChar();
1825b9c1b51eSKate Stone           switch (ch3) {
1826b9c1b51eSKate Stone           case 'P':
1827b9c1b51eSKate Stone             ch = KEY_F(1);
1828b9c1b51eSKate Stone             break;
1829b9c1b51eSKate Stone           case 'Q':
1830b9c1b51eSKate Stone             ch = KEY_F(2);
1831b9c1b51eSKate Stone             break;
1832b9c1b51eSKate Stone           case 'R':
1833b9c1b51eSKate Stone             ch = KEY_F(3);
1834b9c1b51eSKate Stone             break;
1835b9c1b51eSKate Stone           case 'S':
1836b9c1b51eSKate Stone             ch = KEY_F(4);
1837b9c1b51eSKate Stone             break;
183844d93782SGreg Clayton           default:
183944d93782SGreg Clayton             escape_chars.push_back(ch2);
184044d93782SGreg Clayton             if (ch3 != -1)
184144d93782SGreg Clayton               escape_chars.push_back(ch3);
184244d93782SGreg Clayton             break;
184344d93782SGreg Clayton           }
1844b9c1b51eSKate Stone         } else if (ch2 != -1)
184544d93782SGreg Clayton           escape_chars.push_back(ch2);
184644d93782SGreg Clayton       }
184744d93782SGreg Clayton #else
184844d93782SGreg Clayton       int ch = m_window_sp->GetChar();
184944d93782SGreg Clayton 
185044d93782SGreg Clayton #endif
1851b9c1b51eSKate Stone       if (ch == -1) {
1852b9c1b51eSKate Stone         if (feof(m_in) || ferror(m_in)) {
185344d93782SGreg Clayton           done = true;
1854b9c1b51eSKate Stone         } else {
185544d93782SGreg Clayton           // Just a timeout from using halfdelay(), check for events
185644d93782SGreg Clayton           EventSP event_sp;
1857b9c1b51eSKate Stone           while (listener_sp->PeekAtNextEvent()) {
1858d35031e1SPavel Labath             listener_sp->GetEvent(event_sp, std::chrono::seconds(0));
185944d93782SGreg Clayton 
1860b9c1b51eSKate Stone             if (event_sp) {
186144d93782SGreg Clayton               Broadcaster *broadcaster = event_sp->GetBroadcaster();
1862b9c1b51eSKate Stone               if (broadcaster) {
186344d93782SGreg Clayton                 // uint32_t event_type = event_sp->GetType();
1864b9c1b51eSKate Stone                 ConstString broadcaster_class(
1865b9c1b51eSKate Stone                     broadcaster->GetBroadcasterClass());
1866b9c1b51eSKate Stone                 if (broadcaster_class == broadcaster_class_process) {
1867b9c1b51eSKate Stone                   debugger.GetCommandInterpreter().UpdateExecutionContext(
1868b9c1b51eSKate Stone                       nullptr);
186944d93782SGreg Clayton                   update = true;
187044d93782SGreg Clayton                   continue; // Don't get any key, just update our view
187144d93782SGreg Clayton                 }
187244d93782SGreg Clayton               }
187344d93782SGreg Clayton             }
187444d93782SGreg Clayton           }
187544d93782SGreg Clayton         }
1876b9c1b51eSKate Stone       } else {
187744d93782SGreg Clayton         HandleCharResult key_result = m_window_sp->HandleChar(ch);
1878b9c1b51eSKate Stone         switch (key_result) {
187944d93782SGreg Clayton         case eKeyHandled:
1880c5dac77aSEugene Zelenko           debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
188144d93782SGreg Clayton           update = true;
188244d93782SGreg Clayton           break;
188344d93782SGreg Clayton         case eKeyNotHandled:
188444d93782SGreg Clayton           break;
188544d93782SGreg Clayton         case eQuitApplication:
188644d93782SGreg Clayton           done = true;
188744d93782SGreg Clayton           break;
188844d93782SGreg Clayton         }
188944d93782SGreg Clayton       }
189044d93782SGreg Clayton     }
189144d93782SGreg Clayton 
189244d93782SGreg Clayton     debugger.CancelForwardEvents(listener_sp);
189344d93782SGreg Clayton   }
189444d93782SGreg Clayton 
1895b9c1b51eSKate Stone   WindowSP &GetMainWindow() {
189644d93782SGreg Clayton     if (!m_window_sp)
189744d93782SGreg Clayton       m_window_sp.reset(new Window("main", stdscr, false));
189844d93782SGreg Clayton     return m_window_sp;
189944d93782SGreg Clayton   }
190044d93782SGreg Clayton 
1901b9c1b51eSKate Stone   WindowDelegates &GetWindowDelegates() { return m_window_delegates; }
190244d93782SGreg Clayton 
190344d93782SGreg Clayton protected:
190444d93782SGreg Clayton   WindowSP m_window_sp;
190544d93782SGreg Clayton   WindowDelegates m_window_delegates;
190644d93782SGreg Clayton   SCREEN *m_screen;
190744d93782SGreg Clayton   FILE *m_in;
190844d93782SGreg Clayton   FILE *m_out;
190944d93782SGreg Clayton };
191044d93782SGreg Clayton 
191144d93782SGreg Clayton } // namespace curses
191244d93782SGreg Clayton 
191344d93782SGreg Clayton using namespace curses;
191444d93782SGreg Clayton 
1915b9c1b51eSKate Stone struct Row {
19168369b28dSGreg Clayton   ValueObjectManager value;
191744d93782SGreg Clayton   Row *parent;
19188369b28dSGreg Clayton   // The process stop ID when the children were calculated.
19198369b28dSGreg Clayton   uint32_t children_stop_id;
192044d93782SGreg Clayton   int row_idx;
192144d93782SGreg Clayton   int x;
192244d93782SGreg Clayton   int y;
192344d93782SGreg Clayton   bool might_have_children;
192444d93782SGreg Clayton   bool expanded;
192544d93782SGreg Clayton   bool calculated_children;
192644d93782SGreg Clayton   std::vector<Row> children;
192744d93782SGreg Clayton 
1928b9c1b51eSKate Stone   Row(const ValueObjectSP &v, Row *p)
19298369b28dSGreg Clayton       : value(v, lldb::eDynamicDontRunTarget, true), parent(p), row_idx(0),
19308369b28dSGreg Clayton         x(1), y(1), might_have_children(v ? v->MightHaveChildren() : false),
1931b9c1b51eSKate Stone         expanded(false), calculated_children(false), children() {}
193244d93782SGreg Clayton 
1933b9c1b51eSKate Stone   size_t GetDepth() const {
193444d93782SGreg Clayton     if (parent)
193544d93782SGreg Clayton       return 1 + parent->GetDepth();
193644d93782SGreg Clayton     return 0;
193744d93782SGreg Clayton   }
193844d93782SGreg Clayton 
1939b9c1b51eSKate Stone   void Expand() {
194044d93782SGreg Clayton     expanded = true;
19418369b28dSGreg Clayton   }
19428369b28dSGreg Clayton 
19438369b28dSGreg Clayton   std::vector<Row> &GetChildren() {
19448369b28dSGreg Clayton     ProcessSP process_sp = value.GetProcessSP();
19458369b28dSGreg Clayton     auto stop_id = process_sp->GetStopID();
19468369b28dSGreg Clayton     if (process_sp && stop_id != children_stop_id) {
19478369b28dSGreg Clayton       children_stop_id = stop_id;
19488369b28dSGreg Clayton       calculated_children = false;
19498369b28dSGreg Clayton     }
1950b9c1b51eSKate Stone     if (!calculated_children) {
19518369b28dSGreg Clayton       children.clear();
195244d93782SGreg Clayton       calculated_children = true;
19538369b28dSGreg Clayton       ValueObjectSP valobj = value.GetSP();
1954b9c1b51eSKate Stone       if (valobj) {
195544d93782SGreg Clayton         const size_t num_children = valobj->GetNumChildren();
1956b9c1b51eSKate Stone         for (size_t i = 0; i < num_children; ++i) {
195744d93782SGreg Clayton           children.push_back(Row(valobj->GetChildAtIndex(i, true), this));
195844d93782SGreg Clayton         }
195944d93782SGreg Clayton       }
196044d93782SGreg Clayton     }
19618369b28dSGreg Clayton     return children;
196244d93782SGreg Clayton   }
196344d93782SGreg Clayton 
19648369b28dSGreg Clayton   void Unexpand() {
19658369b28dSGreg Clayton     expanded = false;
19668369b28dSGreg Clayton     calculated_children = false;
19678369b28dSGreg Clayton     children.clear();
19688369b28dSGreg Clayton   }
196944d93782SGreg Clayton 
1970b9c1b51eSKate Stone   void DrawTree(Window &window) {
197144d93782SGreg Clayton     if (parent)
197244d93782SGreg Clayton       parent->DrawTreeForChild(window, this, 0);
197344d93782SGreg Clayton 
1974b9c1b51eSKate Stone     if (might_have_children) {
197505097246SAdrian Prantl       // It we can get UTF8 characters to work we should try to use the
197605097246SAdrian Prantl       // "symbol" UTF8 string below
197744d93782SGreg Clayton       //            const char *symbol = "";
197844d93782SGreg Clayton       //            if (row.expanded)
197944d93782SGreg Clayton       //                symbol = "\xe2\x96\xbd ";
198044d93782SGreg Clayton       //            else
198144d93782SGreg Clayton       //                symbol = "\xe2\x96\xb7 ";
198244d93782SGreg Clayton       //            window.PutCString (symbol);
198344d93782SGreg Clayton 
198405097246SAdrian Prantl       // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 'v'
198505097246SAdrian Prantl       // or '>' character...
198644d93782SGreg Clayton       //            if (expanded)
198744d93782SGreg Clayton       //                window.PutChar (ACS_DARROW);
198844d93782SGreg Clayton       //            else
198944d93782SGreg Clayton       //                window.PutChar (ACS_RARROW);
199005097246SAdrian Prantl       // Since we can't find any good looking right arrow/down arrow symbols,
199105097246SAdrian Prantl       // just use a diamond...
199244d93782SGreg Clayton       window.PutChar(ACS_DIAMOND);
199344d93782SGreg Clayton       window.PutChar(ACS_HLINE);
199444d93782SGreg Clayton     }
199544d93782SGreg Clayton   }
199644d93782SGreg Clayton 
1997b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, Row *child, uint32_t reverse_depth) {
199844d93782SGreg Clayton     if (parent)
199944d93782SGreg Clayton       parent->DrawTreeForChild(window, this, reverse_depth + 1);
200044d93782SGreg Clayton 
20018369b28dSGreg Clayton     if (&GetChildren().back() == child) {
200244d93782SGreg Clayton       // Last child
2003b9c1b51eSKate Stone       if (reverse_depth == 0) {
200444d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
200544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2006b9c1b51eSKate Stone       } else {
200744d93782SGreg Clayton         window.PutChar(' ');
200844d93782SGreg Clayton         window.PutChar(' ');
200944d93782SGreg Clayton       }
2010b9c1b51eSKate Stone     } else {
2011b9c1b51eSKate Stone       if (reverse_depth == 0) {
201244d93782SGreg Clayton         window.PutChar(ACS_LTEE);
201344d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2014b9c1b51eSKate Stone       } else {
201544d93782SGreg Clayton         window.PutChar(ACS_VLINE);
201644d93782SGreg Clayton         window.PutChar(' ');
201744d93782SGreg Clayton       }
201844d93782SGreg Clayton     }
201944d93782SGreg Clayton   }
202044d93782SGreg Clayton };
202144d93782SGreg Clayton 
2022b9c1b51eSKate Stone struct DisplayOptions {
202344d93782SGreg Clayton   bool show_types;
202444d93782SGreg Clayton };
202544d93782SGreg Clayton 
202644d93782SGreg Clayton class TreeItem;
202744d93782SGreg Clayton 
2028b9c1b51eSKate Stone class TreeDelegate {
202944d93782SGreg Clayton public:
2030c5dac77aSEugene Zelenko   TreeDelegate() = default;
2031315b6884SEugene Zelenko   virtual ~TreeDelegate() = default;
2032315b6884SEugene Zelenko 
203344d93782SGreg Clayton   virtual void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) = 0;
203444d93782SGreg Clayton   virtual void TreeDelegateGenerateChildren(TreeItem &item) = 0;
2035b9c1b51eSKate Stone   virtual bool TreeDelegateItemSelected(
2036b9c1b51eSKate Stone       TreeItem &item) = 0; // Return true if we need to update views
203744d93782SGreg Clayton };
2038315b6884SEugene Zelenko 
203944d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
204044d93782SGreg Clayton 
2041b9c1b51eSKate Stone class TreeItem {
204244d93782SGreg Clayton public:
2043b9c1b51eSKate Stone   TreeItem(TreeItem *parent, TreeDelegate &delegate, bool might_have_children)
2044b9c1b51eSKate Stone       : m_parent(parent), m_delegate(delegate), m_user_data(nullptr),
2045b9c1b51eSKate Stone         m_identifier(0), m_row_idx(-1), m_children(),
2046b9c1b51eSKate Stone         m_might_have_children(might_have_children), m_is_expanded(false) {}
204744d93782SGreg Clayton 
2048b9c1b51eSKate Stone   TreeItem &operator=(const TreeItem &rhs) {
2049b9c1b51eSKate Stone     if (this != &rhs) {
205044d93782SGreg Clayton       m_parent = rhs.m_parent;
205144d93782SGreg Clayton       m_delegate = rhs.m_delegate;
2052ec990867SGreg Clayton       m_user_data = rhs.m_user_data;
205344d93782SGreg Clayton       m_identifier = rhs.m_identifier;
205444d93782SGreg Clayton       m_row_idx = rhs.m_row_idx;
205544d93782SGreg Clayton       m_children = rhs.m_children;
205644d93782SGreg Clayton       m_might_have_children = rhs.m_might_have_children;
205744d93782SGreg Clayton       m_is_expanded = rhs.m_is_expanded;
205844d93782SGreg Clayton     }
205944d93782SGreg Clayton     return *this;
206044d93782SGreg Clayton   }
206144d93782SGreg Clayton 
2062b9c1b51eSKate Stone   size_t GetDepth() const {
206344d93782SGreg Clayton     if (m_parent)
206444d93782SGreg Clayton       return 1 + m_parent->GetDepth();
206544d93782SGreg Clayton     return 0;
206644d93782SGreg Clayton   }
206744d93782SGreg Clayton 
2068b9c1b51eSKate Stone   int GetRowIndex() const { return m_row_idx; }
206944d93782SGreg Clayton 
2070b9c1b51eSKate Stone   void ClearChildren() { m_children.clear(); }
207144d93782SGreg Clayton 
2072b9c1b51eSKate Stone   void Resize(size_t n, const TreeItem &t) { m_children.resize(n, t); }
207344d93782SGreg Clayton 
2074b9c1b51eSKate Stone   TreeItem &operator[](size_t i) { return m_children[i]; }
207544d93782SGreg Clayton 
2076b9c1b51eSKate Stone   void SetRowIndex(int row_idx) { m_row_idx = row_idx; }
207744d93782SGreg Clayton 
2078b9c1b51eSKate Stone   size_t GetNumChildren() {
207944d93782SGreg Clayton     m_delegate.TreeDelegateGenerateChildren(*this);
208044d93782SGreg Clayton     return m_children.size();
208144d93782SGreg Clayton   }
208244d93782SGreg Clayton 
2083b9c1b51eSKate Stone   void ItemWasSelected() { m_delegate.TreeDelegateItemSelected(*this); }
2084315b6884SEugene Zelenko 
2085b9c1b51eSKate Stone   void CalculateRowIndexes(int &row_idx) {
208644d93782SGreg Clayton     SetRowIndex(row_idx);
208744d93782SGreg Clayton     ++row_idx;
208844d93782SGreg Clayton 
2089ec990867SGreg Clayton     const bool expanded = IsExpanded();
2090ec990867SGreg Clayton 
209105097246SAdrian Prantl     // The root item must calculate its children, or we must calculate the
209205097246SAdrian Prantl     // number of children if the item is expanded
2093c5dac77aSEugene Zelenko     if (m_parent == nullptr || expanded)
209444d93782SGreg Clayton       GetNumChildren();
209544d93782SGreg Clayton 
2096b9c1b51eSKate Stone     for (auto &item : m_children) {
209744d93782SGreg Clayton       if (expanded)
209844d93782SGreg Clayton         item.CalculateRowIndexes(row_idx);
209944d93782SGreg Clayton       else
210044d93782SGreg Clayton         item.SetRowIndex(-1);
210144d93782SGreg Clayton     }
210244d93782SGreg Clayton   }
210344d93782SGreg Clayton 
2104b9c1b51eSKate Stone   TreeItem *GetParent() { return m_parent; }
210544d93782SGreg Clayton 
2106b9c1b51eSKate Stone   bool IsExpanded() const { return m_is_expanded; }
210744d93782SGreg Clayton 
2108b9c1b51eSKate Stone   void Expand() { m_is_expanded = true; }
210944d93782SGreg Clayton 
2110b9c1b51eSKate Stone   void Unexpand() { m_is_expanded = false; }
211144d93782SGreg Clayton 
2112b9c1b51eSKate Stone   bool Draw(Window &window, const int first_visible_row,
2113b9c1b51eSKate Stone             const uint32_t selected_row_idx, int &row_idx, int &num_rows_left) {
211444d93782SGreg Clayton     if (num_rows_left <= 0)
211544d93782SGreg Clayton       return false;
211644d93782SGreg Clayton 
2117b9c1b51eSKate Stone     if (m_row_idx >= first_visible_row) {
211844d93782SGreg Clayton       window.MoveCursor(2, row_idx + 1);
211944d93782SGreg Clayton 
212044d93782SGreg Clayton       if (m_parent)
212144d93782SGreg Clayton         m_parent->DrawTreeForChild(window, this, 0);
212244d93782SGreg Clayton 
2123b9c1b51eSKate Stone       if (m_might_have_children) {
2124b9c1b51eSKate Stone         // It we can get UTF8 characters to work we should try to use the
212505097246SAdrian Prantl         // "symbol" UTF8 string below
212644d93782SGreg Clayton         //            const char *symbol = "";
212744d93782SGreg Clayton         //            if (row.expanded)
212844d93782SGreg Clayton         //                symbol = "\xe2\x96\xbd ";
212944d93782SGreg Clayton         //            else
213044d93782SGreg Clayton         //                symbol = "\xe2\x96\xb7 ";
213144d93782SGreg Clayton         //            window.PutCString (symbol);
213244d93782SGreg Clayton 
213344d93782SGreg Clayton         // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
213444d93782SGreg Clayton         // 'v' or '>' character...
213544d93782SGreg Clayton         //            if (expanded)
213644d93782SGreg Clayton         //                window.PutChar (ACS_DARROW);
213744d93782SGreg Clayton         //            else
213844d93782SGreg Clayton         //                window.PutChar (ACS_RARROW);
213905097246SAdrian Prantl         // Since we can't find any good looking right arrow/down arrow symbols,
214005097246SAdrian Prantl         // just use a diamond...
214144d93782SGreg Clayton         window.PutChar(ACS_DIAMOND);
214244d93782SGreg Clayton         window.PutChar(ACS_HLINE);
214344d93782SGreg Clayton       }
2144b9c1b51eSKate Stone       bool highlight = (selected_row_idx == static_cast<size_t>(m_row_idx)) &&
2145b9c1b51eSKate Stone                        window.IsActive();
214644d93782SGreg Clayton 
214744d93782SGreg Clayton       if (highlight)
214844d93782SGreg Clayton         window.AttributeOn(A_REVERSE);
214944d93782SGreg Clayton 
215044d93782SGreg Clayton       m_delegate.TreeDelegateDrawTreeItem(*this, window);
215144d93782SGreg Clayton 
215244d93782SGreg Clayton       if (highlight)
215344d93782SGreg Clayton         window.AttributeOff(A_REVERSE);
215444d93782SGreg Clayton       ++row_idx;
215544d93782SGreg Clayton       --num_rows_left;
215644d93782SGreg Clayton     }
215744d93782SGreg Clayton 
215844d93782SGreg Clayton     if (num_rows_left <= 0)
215944d93782SGreg Clayton       return false; // We are done drawing...
216044d93782SGreg Clayton 
2161b9c1b51eSKate Stone     if (IsExpanded()) {
2162b9c1b51eSKate Stone       for (auto &item : m_children) {
216305097246SAdrian Prantl         // If we displayed all the rows and item.Draw() returns false we are
216405097246SAdrian Prantl         // done drawing and can exit this for loop
2165b9c1b51eSKate Stone         if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx,
2166b9c1b51eSKate Stone                        num_rows_left))
216744d93782SGreg Clayton           break;
216844d93782SGreg Clayton       }
216944d93782SGreg Clayton     }
217044d93782SGreg Clayton     return num_rows_left >= 0; // Return true if not done drawing yet
217144d93782SGreg Clayton   }
217244d93782SGreg Clayton 
2173b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, TreeItem *child,
2174b9c1b51eSKate Stone                         uint32_t reverse_depth) {
217544d93782SGreg Clayton     if (m_parent)
217644d93782SGreg Clayton       m_parent->DrawTreeForChild(window, this, reverse_depth + 1);
217744d93782SGreg Clayton 
2178b9c1b51eSKate Stone     if (&m_children.back() == child) {
217944d93782SGreg Clayton       // Last child
2180b9c1b51eSKate Stone       if (reverse_depth == 0) {
218144d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
218244d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2183b9c1b51eSKate Stone       } else {
218444d93782SGreg Clayton         window.PutChar(' ');
218544d93782SGreg Clayton         window.PutChar(' ');
218644d93782SGreg Clayton       }
2187b9c1b51eSKate Stone     } else {
2188b9c1b51eSKate Stone       if (reverse_depth == 0) {
218944d93782SGreg Clayton         window.PutChar(ACS_LTEE);
219044d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2191b9c1b51eSKate Stone       } else {
219244d93782SGreg Clayton         window.PutChar(ACS_VLINE);
219344d93782SGreg Clayton         window.PutChar(' ');
219444d93782SGreg Clayton       }
219544d93782SGreg Clayton     }
219644d93782SGreg Clayton   }
219744d93782SGreg Clayton 
2198b9c1b51eSKate Stone   TreeItem *GetItemForRowIndex(uint32_t row_idx) {
21993985c8c6SSaleem Abdulrasool     if (static_cast<uint32_t>(m_row_idx) == row_idx)
220044d93782SGreg Clayton       return this;
220144d93782SGreg Clayton     if (m_children.empty())
2202c5dac77aSEugene Zelenko       return nullptr;
2203b9c1b51eSKate Stone     if (IsExpanded()) {
2204b9c1b51eSKate Stone       for (auto &item : m_children) {
220544d93782SGreg Clayton         TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
220644d93782SGreg Clayton         if (selected_item_ptr)
220744d93782SGreg Clayton           return selected_item_ptr;
220844d93782SGreg Clayton       }
220944d93782SGreg Clayton     }
2210c5dac77aSEugene Zelenko     return nullptr;
221144d93782SGreg Clayton   }
221244d93782SGreg Clayton 
2213b9c1b51eSKate Stone   void *GetUserData() const { return m_user_data; }
2214ec990867SGreg Clayton 
2215b9c1b51eSKate Stone   void SetUserData(void *user_data) { m_user_data = user_data; }
2216ec990867SGreg Clayton 
2217b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
221844d93782SGreg Clayton 
2219b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
222044d93782SGreg Clayton 
2221b9c1b51eSKate Stone   void SetMightHaveChildren(bool b) { m_might_have_children = b; }
2222ec990867SGreg Clayton 
222344d93782SGreg Clayton protected:
222444d93782SGreg Clayton   TreeItem *m_parent;
222544d93782SGreg Clayton   TreeDelegate &m_delegate;
2226ec990867SGreg Clayton   void *m_user_data;
222744d93782SGreg Clayton   uint64_t m_identifier;
2228b9c1b51eSKate Stone   int m_row_idx; // Zero based visible row index, -1 if not visible or for the
2229b9c1b51eSKate Stone                  // root item
223044d93782SGreg Clayton   std::vector<TreeItem> m_children;
223144d93782SGreg Clayton   bool m_might_have_children;
223244d93782SGreg Clayton   bool m_is_expanded;
223344d93782SGreg Clayton };
223444d93782SGreg Clayton 
2235b9c1b51eSKate Stone class TreeWindowDelegate : public WindowDelegate {
223644d93782SGreg Clayton public:
2237b9c1b51eSKate Stone   TreeWindowDelegate(Debugger &debugger, const TreeDelegateSP &delegate_sp)
2238b9c1b51eSKate Stone       : m_debugger(debugger), m_delegate_sp(delegate_sp),
2239b9c1b51eSKate Stone         m_root(nullptr, *delegate_sp, true), m_selected_item(nullptr),
2240b9c1b51eSKate Stone         m_num_rows(0), m_selected_row_idx(0), m_first_visible_row(0),
2241b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
224244d93782SGreg Clayton 
2243b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
224444d93782SGreg Clayton 
2245b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2246b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2247b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
224844d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
224944d93782SGreg Clayton 
225044d93782SGreg Clayton     bool display_content = false;
2251b9c1b51eSKate Stone     if (process) {
225244d93782SGreg Clayton       StateType state = process->GetState();
2253b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
225444d93782SGreg Clayton         // We are stopped, so it is ok to
225544d93782SGreg Clayton         display_content = true;
2256b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
225744d93782SGreg Clayton         return true; // Don't do any updating when we are running
225844d93782SGreg Clayton       }
225944d93782SGreg Clayton     }
226044d93782SGreg Clayton 
226144d93782SGreg Clayton     m_min_x = 2;
226244d93782SGreg Clayton     m_min_y = 1;
226344d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
226444d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
226544d93782SGreg Clayton 
226644d93782SGreg Clayton     window.Erase();
226744d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
226844d93782SGreg Clayton 
2269b9c1b51eSKate Stone     if (display_content) {
227044d93782SGreg Clayton       const int num_visible_rows = NumVisibleRows();
227144d93782SGreg Clayton       m_num_rows = 0;
227244d93782SGreg Clayton       m_root.CalculateRowIndexes(m_num_rows);
227344d93782SGreg Clayton 
227405097246SAdrian Prantl       // If we unexpanded while having something selected our total number of
227505097246SAdrian Prantl       // rows is less than the num visible rows, then make sure we show all the
227605097246SAdrian Prantl       // rows by setting the first visible row accordingly.
227744d93782SGreg Clayton       if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
227844d93782SGreg Clayton         m_first_visible_row = 0;
227944d93782SGreg Clayton 
228044d93782SGreg Clayton       // Make sure the selected row is always visible
228144d93782SGreg Clayton       if (m_selected_row_idx < m_first_visible_row)
228244d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx;
228344d93782SGreg Clayton       else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
228444d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
228544d93782SGreg Clayton 
228644d93782SGreg Clayton       int row_idx = 0;
228744d93782SGreg Clayton       int num_rows_left = num_visible_rows;
2288b9c1b51eSKate Stone       m_root.Draw(window, m_first_visible_row, m_selected_row_idx, row_idx,
2289b9c1b51eSKate Stone                   num_rows_left);
229044d93782SGreg Clayton       // Get the selected row
229144d93782SGreg Clayton       m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
2292b9c1b51eSKate Stone     } else {
2293c5dac77aSEugene Zelenko       m_selected_item = nullptr;
229444d93782SGreg Clayton     }
229544d93782SGreg Clayton 
229644d93782SGreg Clayton     window.DeferredRefresh();
229744d93782SGreg Clayton 
229844d93782SGreg Clayton     return true; // Drawing handled
229944d93782SGreg Clayton   }
230044d93782SGreg Clayton 
2301b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
230244d93782SGreg Clayton     return "Thread window keyboard shortcuts:";
230344d93782SGreg Clayton   }
230444d93782SGreg Clayton 
2305b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
230644d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
230744d93782SGreg Clayton         {KEY_UP, "Select previous item"},
230844d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
230944d93782SGreg Clayton         {KEY_RIGHT, "Expand the selected item"},
2310b9c1b51eSKate Stone         {KEY_LEFT,
2311b9c1b51eSKate Stone          "Unexpand the selected item or select parent if not expanded"},
231244d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
231344d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
231444d93782SGreg Clayton         {'h', "Show help dialog"},
231544d93782SGreg Clayton         {' ', "Toggle item expansion"},
231644d93782SGreg Clayton         {',', "Page up"},
231744d93782SGreg Clayton         {'.', "Page down"},
2318b9c1b51eSKate Stone         {'\0', nullptr}};
231944d93782SGreg Clayton     return g_source_view_key_help;
232044d93782SGreg Clayton   }
232144d93782SGreg Clayton 
2322b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2323b9c1b51eSKate Stone     switch (c) {
232444d93782SGreg Clayton     case ',':
232544d93782SGreg Clayton     case KEY_PPAGE:
232644d93782SGreg Clayton       // Page up key
2327b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
232844d93782SGreg Clayton         if (m_first_visible_row > m_max_y)
232944d93782SGreg Clayton           m_first_visible_row -= m_max_y;
233044d93782SGreg Clayton         else
233144d93782SGreg Clayton           m_first_visible_row = 0;
233244d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
233344d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
233444d93782SGreg Clayton         if (m_selected_item)
233544d93782SGreg Clayton           m_selected_item->ItemWasSelected();
233644d93782SGreg Clayton       }
233744d93782SGreg Clayton       return eKeyHandled;
233844d93782SGreg Clayton 
233944d93782SGreg Clayton     case '.':
234044d93782SGreg Clayton     case KEY_NPAGE:
234144d93782SGreg Clayton       // Page down key
2342b9c1b51eSKate Stone       if (m_num_rows > m_max_y) {
2343b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
234444d93782SGreg Clayton           m_first_visible_row += m_max_y;
234544d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
234644d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
234744d93782SGreg Clayton           if (m_selected_item)
234844d93782SGreg Clayton             m_selected_item->ItemWasSelected();
234944d93782SGreg Clayton         }
235044d93782SGreg Clayton       }
235144d93782SGreg Clayton       return eKeyHandled;
235244d93782SGreg Clayton 
235344d93782SGreg Clayton     case KEY_UP:
2354b9c1b51eSKate Stone       if (m_selected_row_idx > 0) {
235544d93782SGreg Clayton         --m_selected_row_idx;
235644d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
235744d93782SGreg Clayton         if (m_selected_item)
235844d93782SGreg Clayton           m_selected_item->ItemWasSelected();
235944d93782SGreg Clayton       }
236044d93782SGreg Clayton       return eKeyHandled;
2361315b6884SEugene Zelenko 
236244d93782SGreg Clayton     case KEY_DOWN:
2363b9c1b51eSKate Stone       if (m_selected_row_idx + 1 < m_num_rows) {
236444d93782SGreg Clayton         ++m_selected_row_idx;
236544d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
236644d93782SGreg Clayton         if (m_selected_item)
236744d93782SGreg Clayton           m_selected_item->ItemWasSelected();
236844d93782SGreg Clayton       }
236944d93782SGreg Clayton       return eKeyHandled;
237044d93782SGreg Clayton 
237144d93782SGreg Clayton     case KEY_RIGHT:
2372b9c1b51eSKate Stone       if (m_selected_item) {
237344d93782SGreg Clayton         if (!m_selected_item->IsExpanded())
237444d93782SGreg Clayton           m_selected_item->Expand();
237544d93782SGreg Clayton       }
237644d93782SGreg Clayton       return eKeyHandled;
237744d93782SGreg Clayton 
237844d93782SGreg Clayton     case KEY_LEFT:
2379b9c1b51eSKate Stone       if (m_selected_item) {
238044d93782SGreg Clayton         if (m_selected_item->IsExpanded())
238144d93782SGreg Clayton           m_selected_item->Unexpand();
2382b9c1b51eSKate Stone         else if (m_selected_item->GetParent()) {
238344d93782SGreg Clayton           m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
238444d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
238544d93782SGreg Clayton           if (m_selected_item)
238644d93782SGreg Clayton             m_selected_item->ItemWasSelected();
238744d93782SGreg Clayton         }
238844d93782SGreg Clayton       }
238944d93782SGreg Clayton       return eKeyHandled;
239044d93782SGreg Clayton 
239144d93782SGreg Clayton     case ' ':
239244d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2393b9c1b51eSKate Stone       if (m_selected_item) {
239444d93782SGreg Clayton         if (m_selected_item->IsExpanded())
239544d93782SGreg Clayton           m_selected_item->Unexpand();
239644d93782SGreg Clayton         else
239744d93782SGreg Clayton           m_selected_item->Expand();
239844d93782SGreg Clayton       }
239944d93782SGreg Clayton       return eKeyHandled;
240044d93782SGreg Clayton 
240144d93782SGreg Clayton     case 'h':
240244d93782SGreg Clayton       window.CreateHelpSubwindow();
240344d93782SGreg Clayton       return eKeyHandled;
240444d93782SGreg Clayton 
240544d93782SGreg Clayton     default:
240644d93782SGreg Clayton       break;
240744d93782SGreg Clayton     }
240844d93782SGreg Clayton     return eKeyNotHandled;
240944d93782SGreg Clayton   }
241044d93782SGreg Clayton 
241144d93782SGreg Clayton protected:
241244d93782SGreg Clayton   Debugger &m_debugger;
241344d93782SGreg Clayton   TreeDelegateSP m_delegate_sp;
241444d93782SGreg Clayton   TreeItem m_root;
241544d93782SGreg Clayton   TreeItem *m_selected_item;
241644d93782SGreg Clayton   int m_num_rows;
241744d93782SGreg Clayton   int m_selected_row_idx;
241844d93782SGreg Clayton   int m_first_visible_row;
241944d93782SGreg Clayton   int m_min_x;
242044d93782SGreg Clayton   int m_min_y;
242144d93782SGreg Clayton   int m_max_x;
242244d93782SGreg Clayton   int m_max_y;
242344d93782SGreg Clayton };
242444d93782SGreg Clayton 
2425b9c1b51eSKate Stone class FrameTreeDelegate : public TreeDelegate {
242644d93782SGreg Clayton public:
2427b9c1b51eSKate Stone   FrameTreeDelegate() : TreeDelegate() {
2428b9c1b51eSKate Stone     FormatEntity::Parse(
2429b9c1b51eSKate Stone         "frame #${frame.index}: {${function.name}${function.pc-offset}}}",
2430554f68d3SGreg Clayton         m_format);
243144d93782SGreg Clayton   }
243244d93782SGreg Clayton 
2433315b6884SEugene Zelenko   ~FrameTreeDelegate() override = default;
243444d93782SGreg Clayton 
2435b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2436ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2437b9c1b51eSKate Stone     if (thread) {
243844d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2439ec990867SGreg Clayton       StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
2440b9c1b51eSKate Stone       if (frame_sp) {
244144d93782SGreg Clayton         StreamString strm;
2442b9c1b51eSKate Stone         const SymbolContext &sc =
2443b9c1b51eSKate Stone             frame_sp->GetSymbolContext(eSymbolContextEverything);
244444d93782SGreg Clayton         ExecutionContext exe_ctx(frame_sp);
2445b9c1b51eSKate Stone         if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr,
2446b9c1b51eSKate Stone                                  nullptr, false, false)) {
244744d93782SGreg Clayton           int right_pad = 1;
2448c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
244944d93782SGreg Clayton         }
245044d93782SGreg Clayton       }
245144d93782SGreg Clayton     }
245244d93782SGreg Clayton   }
2453315b6884SEugene Zelenko 
2454b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
245544d93782SGreg Clayton     // No children for frames yet...
245644d93782SGreg Clayton   }
245744d93782SGreg Clayton 
2458b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2459ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2460b9c1b51eSKate Stone     if (thread) {
2461b9c1b51eSKate Stone       thread->GetProcess()->GetThreadList().SetSelectedThreadByID(
2462b9c1b51eSKate Stone           thread->GetID());
246344d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2464ec990867SGreg Clayton       thread->SetSelectedFrameByIndex(frame_idx);
246544d93782SGreg Clayton       return true;
246644d93782SGreg Clayton     }
246744d93782SGreg Clayton     return false;
246844d93782SGreg Clayton   }
2469315b6884SEugene Zelenko 
2470554f68d3SGreg Clayton protected:
2471554f68d3SGreg Clayton   FormatEntity::Entry m_format;
247244d93782SGreg Clayton };
247344d93782SGreg Clayton 
2474b9c1b51eSKate Stone class ThreadTreeDelegate : public TreeDelegate {
247544d93782SGreg Clayton public:
2476b9c1b51eSKate Stone   ThreadTreeDelegate(Debugger &debugger)
2477b9c1b51eSKate Stone       : TreeDelegate(), m_debugger(debugger), m_tid(LLDB_INVALID_THREAD_ID),
2478b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2479b9c1b51eSKate Stone     FormatEntity::Parse("thread #${thread.index}: tid = ${thread.id}{, stop "
2480b9c1b51eSKate Stone                         "reason = ${thread.stop-reason}}",
2481554f68d3SGreg Clayton                         m_format);
248244d93782SGreg Clayton   }
248344d93782SGreg Clayton 
2484315b6884SEugene Zelenko   ~ThreadTreeDelegate() override = default;
248544d93782SGreg Clayton 
2486b9c1b51eSKate Stone   ProcessSP GetProcess() {
2487b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2488b9c1b51eSKate Stone         .GetExecutionContext()
2489b9c1b51eSKate Stone         .GetProcessSP();
2490ec990867SGreg Clayton   }
2491ec990867SGreg Clayton 
2492b9c1b51eSKate Stone   ThreadSP GetThread(const TreeItem &item) {
2493ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2494ec990867SGreg Clayton     if (process_sp)
2495ec990867SGreg Clayton       return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
2496ec990867SGreg Clayton     return ThreadSP();
2497ec990867SGreg Clayton   }
2498ec990867SGreg Clayton 
2499b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2500ec990867SGreg Clayton     ThreadSP thread_sp = GetThread(item);
2501b9c1b51eSKate Stone     if (thread_sp) {
250244d93782SGreg Clayton       StreamString strm;
250344d93782SGreg Clayton       ExecutionContext exe_ctx(thread_sp);
2504b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2505b9c1b51eSKate Stone                                nullptr, false, false)) {
250644d93782SGreg Clayton         int right_pad = 1;
2507c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
250844d93782SGreg Clayton       }
250944d93782SGreg Clayton     }
251044d93782SGreg Clayton   }
2511315b6884SEugene Zelenko 
2512b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2513ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2514b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
251544d93782SGreg Clayton       StateType state = process_sp->GetState();
2516b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2517ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2518b9c1b51eSKate Stone         if (thread_sp) {
2519b9c1b51eSKate Stone           if (m_stop_id == process_sp->GetStopID() &&
2520b9c1b51eSKate Stone               thread_sp->GetID() == m_tid)
252144d93782SGreg Clayton             return; // Children are already up to date
2522b9c1b51eSKate Stone           if (!m_frame_delegate_sp) {
252344d93782SGreg Clayton             // Always expand the thread item the first time we show it
2524ec990867SGreg Clayton             m_frame_delegate_sp.reset(new FrameTreeDelegate());
252544d93782SGreg Clayton           }
252644d93782SGreg Clayton 
252744d93782SGreg Clayton           m_stop_id = process_sp->GetStopID();
252844d93782SGreg Clayton           m_tid = thread_sp->GetID();
252944d93782SGreg Clayton 
253044d93782SGreg Clayton           TreeItem t(&item, *m_frame_delegate_sp, false);
253144d93782SGreg Clayton           size_t num_frames = thread_sp->GetStackFrameCount();
253244d93782SGreg Clayton           item.Resize(num_frames, t);
2533b9c1b51eSKate Stone           for (size_t i = 0; i < num_frames; ++i) {
2534ec990867SGreg Clayton             item[i].SetUserData(thread_sp.get());
253544d93782SGreg Clayton             item[i].SetIdentifier(i);
253644d93782SGreg Clayton           }
253744d93782SGreg Clayton         }
253844d93782SGreg Clayton         return;
253944d93782SGreg Clayton       }
254044d93782SGreg Clayton     }
254144d93782SGreg Clayton     item.ClearChildren();
254244d93782SGreg Clayton   }
254344d93782SGreg Clayton 
2544b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2545ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2546b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2547ec990867SGreg Clayton       StateType state = process_sp->GetState();
2548b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2549ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2550b9c1b51eSKate Stone         if (thread_sp) {
255144d93782SGreg Clayton           ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
2552bb19a13cSSaleem Abdulrasool           std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
255344d93782SGreg Clayton           ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
2554b9c1b51eSKate Stone           if (selected_thread_sp->GetID() != thread_sp->GetID()) {
255544d93782SGreg Clayton             thread_list.SetSelectedThreadByID(thread_sp->GetID());
255644d93782SGreg Clayton             return true;
255744d93782SGreg Clayton           }
255844d93782SGreg Clayton         }
2559ec990867SGreg Clayton       }
2560ec990867SGreg Clayton     }
256144d93782SGreg Clayton     return false;
256244d93782SGreg Clayton   }
256344d93782SGreg Clayton 
256444d93782SGreg Clayton protected:
256544d93782SGreg Clayton   Debugger &m_debugger;
256644d93782SGreg Clayton   std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
256744d93782SGreg Clayton   lldb::user_id_t m_tid;
256844d93782SGreg Clayton   uint32_t m_stop_id;
2569554f68d3SGreg Clayton   FormatEntity::Entry m_format;
257044d93782SGreg Clayton };
257144d93782SGreg Clayton 
2572b9c1b51eSKate Stone class ThreadsTreeDelegate : public TreeDelegate {
2573ec990867SGreg Clayton public:
2574b9c1b51eSKate Stone   ThreadsTreeDelegate(Debugger &debugger)
2575b9c1b51eSKate Stone       : TreeDelegate(), m_thread_delegate_sp(), m_debugger(debugger),
2576b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2577554f68d3SGreg Clayton     FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
2578554f68d3SGreg Clayton                         m_format);
2579ec990867SGreg Clayton   }
2580ec990867SGreg Clayton 
2581315b6884SEugene Zelenko   ~ThreadsTreeDelegate() override = default;
2582ec990867SGreg Clayton 
2583b9c1b51eSKate Stone   ProcessSP GetProcess() {
2584b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2585b9c1b51eSKate Stone         .GetExecutionContext()
2586b9c1b51eSKate Stone         .GetProcessSP();
2587ec990867SGreg Clayton   }
2588ec990867SGreg Clayton 
2589b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2590ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2591b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2592ec990867SGreg Clayton       StreamString strm;
2593ec990867SGreg Clayton       ExecutionContext exe_ctx(process_sp);
2594b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2595b9c1b51eSKate Stone                                nullptr, false, false)) {
2596ec990867SGreg Clayton         int right_pad = 1;
2597c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
2598ec990867SGreg Clayton       }
2599ec990867SGreg Clayton     }
2600ec990867SGreg Clayton   }
2601ec990867SGreg Clayton 
2602b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2603ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2604b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2605ec990867SGreg Clayton       StateType state = process_sp->GetState();
2606b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2607ec990867SGreg Clayton         const uint32_t stop_id = process_sp->GetStopID();
2608ec990867SGreg Clayton         if (m_stop_id == stop_id)
2609ec990867SGreg Clayton           return; // Children are already up to date
2610ec990867SGreg Clayton 
2611ec990867SGreg Clayton         m_stop_id = stop_id;
2612ec990867SGreg Clayton 
2613b9c1b51eSKate Stone         if (!m_thread_delegate_sp) {
2614ec990867SGreg Clayton           // Always expand the thread item the first time we show it
2615ec990867SGreg Clayton           // item.Expand();
2616ec990867SGreg Clayton           m_thread_delegate_sp.reset(new ThreadTreeDelegate(m_debugger));
2617ec990867SGreg Clayton         }
2618ec990867SGreg Clayton 
2619ec990867SGreg Clayton         TreeItem t(&item, *m_thread_delegate_sp, false);
2620ec990867SGreg Clayton         ThreadList &threads = process_sp->GetThreadList();
2621bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
2622ec990867SGreg Clayton         size_t num_threads = threads.GetSize();
2623ec990867SGreg Clayton         item.Resize(num_threads, t);
2624b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
2625ec990867SGreg Clayton           item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
2626ec990867SGreg Clayton           item[i].SetMightHaveChildren(true);
2627ec990867SGreg Clayton         }
2628ec990867SGreg Clayton         return;
2629ec990867SGreg Clayton       }
2630ec990867SGreg Clayton     }
2631ec990867SGreg Clayton     item.ClearChildren();
2632ec990867SGreg Clayton   }
2633ec990867SGreg Clayton 
2634b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override { return false; }
2635ec990867SGreg Clayton 
2636ec990867SGreg Clayton protected:
2637ec990867SGreg Clayton   std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
2638ec990867SGreg Clayton   Debugger &m_debugger;
2639ec990867SGreg Clayton   uint32_t m_stop_id;
2640554f68d3SGreg Clayton   FormatEntity::Entry m_format;
2641ec990867SGreg Clayton };
2642ec990867SGreg Clayton 
2643b9c1b51eSKate Stone class ValueObjectListDelegate : public WindowDelegate {
264444d93782SGreg Clayton public:
2645b9c1b51eSKate Stone   ValueObjectListDelegate()
26468369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2647b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2648b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {}
264944d93782SGreg Clayton 
2650b9c1b51eSKate Stone   ValueObjectListDelegate(ValueObjectList &valobj_list)
26518369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2652b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2653b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {
265444d93782SGreg Clayton     SetValues(valobj_list);
265544d93782SGreg Clayton   }
265644d93782SGreg Clayton 
2657315b6884SEugene Zelenko   ~ValueObjectListDelegate() override = default;
265844d93782SGreg Clayton 
2659b9c1b51eSKate Stone   void SetValues(ValueObjectList &valobj_list) {
2660c5dac77aSEugene Zelenko     m_selected_row = nullptr;
266144d93782SGreg Clayton     m_selected_row_idx = 0;
266244d93782SGreg Clayton     m_first_visible_row = 0;
266344d93782SGreg Clayton     m_num_rows = 0;
266444d93782SGreg Clayton     m_rows.clear();
26658369b28dSGreg Clayton     for (auto &valobj_sp : valobj_list.GetObjects())
26668369b28dSGreg Clayton       m_rows.push_back(Row(valobj_sp, nullptr));
266744d93782SGreg Clayton   }
266844d93782SGreg Clayton 
2669b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
267044d93782SGreg Clayton     m_num_rows = 0;
267144d93782SGreg Clayton     m_min_x = 2;
267244d93782SGreg Clayton     m_min_y = 1;
267344d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
267444d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
267544d93782SGreg Clayton 
267644d93782SGreg Clayton     window.Erase();
267744d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
267844d93782SGreg Clayton 
267944d93782SGreg Clayton     const int num_visible_rows = NumVisibleRows();
268044d93782SGreg Clayton     const int num_rows = CalculateTotalNumberRows(m_rows);
268144d93782SGreg Clayton 
268205097246SAdrian Prantl     // If we unexpanded while having something selected our total number of
268305097246SAdrian Prantl     // rows is less than the num visible rows, then make sure we show all the
268405097246SAdrian Prantl     // rows by setting the first visible row accordingly.
268544d93782SGreg Clayton     if (m_first_visible_row > 0 && num_rows < num_visible_rows)
268644d93782SGreg Clayton       m_first_visible_row = 0;
268744d93782SGreg Clayton 
268844d93782SGreg Clayton     // Make sure the selected row is always visible
268944d93782SGreg Clayton     if (m_selected_row_idx < m_first_visible_row)
269044d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx;
269144d93782SGreg Clayton     else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
269244d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
269344d93782SGreg Clayton 
269444d93782SGreg Clayton     DisplayRows(window, m_rows, g_options);
269544d93782SGreg Clayton 
269644d93782SGreg Clayton     window.DeferredRefresh();
269744d93782SGreg Clayton 
269844d93782SGreg Clayton     // Get the selected row
269944d93782SGreg Clayton     m_selected_row = GetRowForRowIndex(m_selected_row_idx);
270005097246SAdrian Prantl     // Keep the cursor on the selected row so the highlight and the cursor are
270105097246SAdrian Prantl     // always on the same line
270244d93782SGreg Clayton     if (m_selected_row)
2703b9c1b51eSKate Stone       window.MoveCursor(m_selected_row->x, m_selected_row->y);
270444d93782SGreg Clayton 
270544d93782SGreg Clayton     return true; // Drawing handled
270644d93782SGreg Clayton   }
270744d93782SGreg Clayton 
2708b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
270944d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
271044d93782SGreg Clayton         {KEY_UP, "Select previous item"},
271144d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
271244d93782SGreg Clayton         {KEY_RIGHT, "Expand selected item"},
271344d93782SGreg Clayton         {KEY_LEFT, "Unexpand selected item or select parent if not expanded"},
271444d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
271544d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
271644d93782SGreg Clayton         {'A', "Format as annotated address"},
271744d93782SGreg Clayton         {'b', "Format as binary"},
271844d93782SGreg Clayton         {'B', "Format as hex bytes with ASCII"},
271944d93782SGreg Clayton         {'c', "Format as character"},
272044d93782SGreg Clayton         {'d', "Format as a signed integer"},
272144d93782SGreg Clayton         {'D', "Format selected value using the default format for the type"},
272244d93782SGreg Clayton         {'f', "Format as float"},
272344d93782SGreg Clayton         {'h', "Show help dialog"},
272444d93782SGreg Clayton         {'i', "Format as instructions"},
272544d93782SGreg Clayton         {'o', "Format as octal"},
272644d93782SGreg Clayton         {'p', "Format as pointer"},
272744d93782SGreg Clayton         {'s', "Format as C string"},
272844d93782SGreg Clayton         {'t', "Toggle showing/hiding type names"},
272944d93782SGreg Clayton         {'u', "Format as an unsigned integer"},
273044d93782SGreg Clayton         {'x', "Format as hex"},
273144d93782SGreg Clayton         {'X', "Format as uppercase hex"},
273244d93782SGreg Clayton         {' ', "Toggle item expansion"},
273344d93782SGreg Clayton         {',', "Page up"},
273444d93782SGreg Clayton         {'.', "Page down"},
2735b9c1b51eSKate Stone         {'\0', nullptr}};
273644d93782SGreg Clayton     return g_source_view_key_help;
273744d93782SGreg Clayton   }
273844d93782SGreg Clayton 
2739b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2740b9c1b51eSKate Stone     switch (c) {
274144d93782SGreg Clayton     case 'x':
274244d93782SGreg Clayton     case 'X':
274344d93782SGreg Clayton     case 'o':
274444d93782SGreg Clayton     case 's':
274544d93782SGreg Clayton     case 'u':
274644d93782SGreg Clayton     case 'd':
274744d93782SGreg Clayton     case 'D':
274844d93782SGreg Clayton     case 'i':
274944d93782SGreg Clayton     case 'A':
275044d93782SGreg Clayton     case 'p':
275144d93782SGreg Clayton     case 'c':
275244d93782SGreg Clayton     case 'b':
275344d93782SGreg Clayton     case 'B':
275444d93782SGreg Clayton     case 'f':
275544d93782SGreg Clayton       // Change the format for the currently selected item
27568369b28dSGreg Clayton       if (m_selected_row) {
27578369b28dSGreg Clayton         auto valobj_sp = m_selected_row->value.GetSP();
27588369b28dSGreg Clayton         if (valobj_sp)
27598369b28dSGreg Clayton           valobj_sp->SetFormat(FormatForChar(c));
27608369b28dSGreg Clayton       }
276144d93782SGreg Clayton       return eKeyHandled;
276244d93782SGreg Clayton 
276344d93782SGreg Clayton     case 't':
276444d93782SGreg Clayton       // Toggle showing type names
276544d93782SGreg Clayton       g_options.show_types = !g_options.show_types;
276644d93782SGreg Clayton       return eKeyHandled;
276744d93782SGreg Clayton 
276844d93782SGreg Clayton     case ',':
276944d93782SGreg Clayton     case KEY_PPAGE:
277044d93782SGreg Clayton       // Page up key
2771b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
27723985c8c6SSaleem Abdulrasool         if (static_cast<int>(m_first_visible_row) > m_max_y)
277344d93782SGreg Clayton           m_first_visible_row -= m_max_y;
277444d93782SGreg Clayton         else
277544d93782SGreg Clayton           m_first_visible_row = 0;
277644d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
277744d93782SGreg Clayton       }
277844d93782SGreg Clayton       return eKeyHandled;
277944d93782SGreg Clayton 
278044d93782SGreg Clayton     case '.':
278144d93782SGreg Clayton     case KEY_NPAGE:
278244d93782SGreg Clayton       // Page down key
2783b9c1b51eSKate Stone       if (m_num_rows > static_cast<size_t>(m_max_y)) {
2784b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
278544d93782SGreg Clayton           m_first_visible_row += m_max_y;
278644d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
278744d93782SGreg Clayton         }
278844d93782SGreg Clayton       }
278944d93782SGreg Clayton       return eKeyHandled;
279044d93782SGreg Clayton 
279144d93782SGreg Clayton     case KEY_UP:
279244d93782SGreg Clayton       if (m_selected_row_idx > 0)
279344d93782SGreg Clayton         --m_selected_row_idx;
279444d93782SGreg Clayton       return eKeyHandled;
2795315b6884SEugene Zelenko 
279644d93782SGreg Clayton     case KEY_DOWN:
279744d93782SGreg Clayton       if (m_selected_row_idx + 1 < m_num_rows)
279844d93782SGreg Clayton         ++m_selected_row_idx;
279944d93782SGreg Clayton       return eKeyHandled;
280044d93782SGreg Clayton 
280144d93782SGreg Clayton     case KEY_RIGHT:
2802b9c1b51eSKate Stone       if (m_selected_row) {
280344d93782SGreg Clayton         if (!m_selected_row->expanded)
280444d93782SGreg Clayton           m_selected_row->Expand();
280544d93782SGreg Clayton       }
280644d93782SGreg Clayton       return eKeyHandled;
280744d93782SGreg Clayton 
280844d93782SGreg Clayton     case KEY_LEFT:
2809b9c1b51eSKate Stone       if (m_selected_row) {
281044d93782SGreg Clayton         if (m_selected_row->expanded)
281144d93782SGreg Clayton           m_selected_row->Unexpand();
281244d93782SGreg Clayton         else if (m_selected_row->parent)
281344d93782SGreg Clayton           m_selected_row_idx = m_selected_row->parent->row_idx;
281444d93782SGreg Clayton       }
281544d93782SGreg Clayton       return eKeyHandled;
281644d93782SGreg Clayton 
281744d93782SGreg Clayton     case ' ':
281844d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2819b9c1b51eSKate Stone       if (m_selected_row) {
282044d93782SGreg Clayton         if (m_selected_row->expanded)
282144d93782SGreg Clayton           m_selected_row->Unexpand();
282244d93782SGreg Clayton         else
282344d93782SGreg Clayton           m_selected_row->Expand();
282444d93782SGreg Clayton       }
282544d93782SGreg Clayton       return eKeyHandled;
282644d93782SGreg Clayton 
282744d93782SGreg Clayton     case 'h':
282844d93782SGreg Clayton       window.CreateHelpSubwindow();
282944d93782SGreg Clayton       return eKeyHandled;
283044d93782SGreg Clayton 
283144d93782SGreg Clayton     default:
283244d93782SGreg Clayton       break;
283344d93782SGreg Clayton     }
283444d93782SGreg Clayton     return eKeyNotHandled;
283544d93782SGreg Clayton   }
283644d93782SGreg Clayton 
283744d93782SGreg Clayton protected:
283844d93782SGreg Clayton   std::vector<Row> m_rows;
283944d93782SGreg Clayton   Row *m_selected_row;
284044d93782SGreg Clayton   uint32_t m_selected_row_idx;
284144d93782SGreg Clayton   uint32_t m_first_visible_row;
284244d93782SGreg Clayton   uint32_t m_num_rows;
284344d93782SGreg Clayton   int m_min_x;
284444d93782SGreg Clayton   int m_min_y;
284544d93782SGreg Clayton   int m_max_x;
284644d93782SGreg Clayton   int m_max_y;
284744d93782SGreg Clayton 
2848b9c1b51eSKate Stone   static Format FormatForChar(int c) {
2849b9c1b51eSKate Stone     switch (c) {
2850b9c1b51eSKate Stone     case 'x':
2851b9c1b51eSKate Stone       return eFormatHex;
2852b9c1b51eSKate Stone     case 'X':
2853b9c1b51eSKate Stone       return eFormatHexUppercase;
2854b9c1b51eSKate Stone     case 'o':
2855b9c1b51eSKate Stone       return eFormatOctal;
2856b9c1b51eSKate Stone     case 's':
2857b9c1b51eSKate Stone       return eFormatCString;
2858b9c1b51eSKate Stone     case 'u':
2859b9c1b51eSKate Stone       return eFormatUnsigned;
2860b9c1b51eSKate Stone     case 'd':
2861b9c1b51eSKate Stone       return eFormatDecimal;
2862b9c1b51eSKate Stone     case 'D':
2863b9c1b51eSKate Stone       return eFormatDefault;
2864b9c1b51eSKate Stone     case 'i':
2865b9c1b51eSKate Stone       return eFormatInstruction;
2866b9c1b51eSKate Stone     case 'A':
2867b9c1b51eSKate Stone       return eFormatAddressInfo;
2868b9c1b51eSKate Stone     case 'p':
2869b9c1b51eSKate Stone       return eFormatPointer;
2870b9c1b51eSKate Stone     case 'c':
2871b9c1b51eSKate Stone       return eFormatChar;
2872b9c1b51eSKate Stone     case 'b':
2873b9c1b51eSKate Stone       return eFormatBinary;
2874b9c1b51eSKate Stone     case 'B':
2875b9c1b51eSKate Stone       return eFormatBytesWithASCII;
2876b9c1b51eSKate Stone     case 'f':
2877b9c1b51eSKate Stone       return eFormatFloat;
287844d93782SGreg Clayton     }
287944d93782SGreg Clayton     return eFormatDefault;
288044d93782SGreg Clayton   }
288144d93782SGreg Clayton 
2882b9c1b51eSKate Stone   bool DisplayRowObject(Window &window, Row &row, DisplayOptions &options,
2883b9c1b51eSKate Stone                         bool highlight, bool last_child) {
28848369b28dSGreg Clayton     ValueObject *valobj = row.value.GetSP().get();
288544d93782SGreg Clayton 
2886c5dac77aSEugene Zelenko     if (valobj == nullptr)
288744d93782SGreg Clayton       return false;
288844d93782SGreg Clayton 
2889b9c1b51eSKate Stone     const char *type_name =
2890b9c1b51eSKate Stone         options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
289144d93782SGreg Clayton     const char *name = valobj->GetName().GetCString();
289244d93782SGreg Clayton     const char *value = valobj->GetValueAsCString();
289344d93782SGreg Clayton     const char *summary = valobj->GetSummaryAsCString();
289444d93782SGreg Clayton 
289544d93782SGreg Clayton     window.MoveCursor(row.x, row.y);
289644d93782SGreg Clayton 
289744d93782SGreg Clayton     row.DrawTree(window);
289844d93782SGreg Clayton 
289944d93782SGreg Clayton     if (highlight)
290044d93782SGreg Clayton       window.AttributeOn(A_REVERSE);
290144d93782SGreg Clayton 
290244d93782SGreg Clayton     if (type_name && type_name[0])
290344d93782SGreg Clayton       window.Printf("(%s) ", type_name);
290444d93782SGreg Clayton 
290544d93782SGreg Clayton     if (name && name[0])
290644d93782SGreg Clayton       window.PutCString(name);
290744d93782SGreg Clayton 
290844d93782SGreg Clayton     attr_t changd_attr = 0;
290944d93782SGreg Clayton     if (valobj->GetValueDidChange())
291044d93782SGreg Clayton       changd_attr = COLOR_PAIR(5) | A_BOLD;
291144d93782SGreg Clayton 
2912b9c1b51eSKate Stone     if (value && value[0]) {
291344d93782SGreg Clayton       window.PutCString(" = ");
291444d93782SGreg Clayton       if (changd_attr)
291544d93782SGreg Clayton         window.AttributeOn(changd_attr);
291644d93782SGreg Clayton       window.PutCString(value);
291744d93782SGreg Clayton       if (changd_attr)
291844d93782SGreg Clayton         window.AttributeOff(changd_attr);
291944d93782SGreg Clayton     }
292044d93782SGreg Clayton 
2921b9c1b51eSKate Stone     if (summary && summary[0]) {
292244d93782SGreg Clayton       window.PutChar(' ');
292344d93782SGreg Clayton       if (changd_attr)
292444d93782SGreg Clayton         window.AttributeOn(changd_attr);
292544d93782SGreg Clayton       window.PutCString(summary);
292644d93782SGreg Clayton       if (changd_attr)
292744d93782SGreg Clayton         window.AttributeOff(changd_attr);
292844d93782SGreg Clayton     }
292944d93782SGreg Clayton 
293044d93782SGreg Clayton     if (highlight)
293144d93782SGreg Clayton       window.AttributeOff(A_REVERSE);
293244d93782SGreg Clayton 
293344d93782SGreg Clayton     return true;
293444d93782SGreg Clayton   }
2935315b6884SEugene Zelenko 
2936b9c1b51eSKate Stone   void DisplayRows(Window &window, std::vector<Row> &rows,
2937b9c1b51eSKate Stone                    DisplayOptions &options) {
293844d93782SGreg Clayton     // >   0x25B7
293944d93782SGreg Clayton     // \/  0x25BD
294044d93782SGreg Clayton 
294144d93782SGreg Clayton     bool window_is_active = window.IsActive();
2942b9c1b51eSKate Stone     for (auto &row : rows) {
294344d93782SGreg Clayton       const bool last_child = row.parent && &rows[rows.size() - 1] == &row;
294444d93782SGreg Clayton       // Save the row index in each Row structure
294544d93782SGreg Clayton       row.row_idx = m_num_rows;
294644d93782SGreg Clayton       if ((m_num_rows >= m_first_visible_row) &&
2947b9c1b51eSKate Stone           ((m_num_rows - m_first_visible_row) <
2948b9c1b51eSKate Stone            static_cast<size_t>(NumVisibleRows()))) {
294944d93782SGreg Clayton         row.x = m_min_x;
295044d93782SGreg Clayton         row.y = m_num_rows - m_first_visible_row + 1;
2951b9c1b51eSKate Stone         if (DisplayRowObject(window, row, options,
2952b9c1b51eSKate Stone                              window_is_active &&
2953b9c1b51eSKate Stone                                  m_num_rows == m_selected_row_idx,
2954b9c1b51eSKate Stone                              last_child)) {
295544d93782SGreg Clayton           ++m_num_rows;
2956b9c1b51eSKate Stone         } else {
295744d93782SGreg Clayton           row.x = 0;
295844d93782SGreg Clayton           row.y = 0;
295944d93782SGreg Clayton         }
2960b9c1b51eSKate Stone       } else {
296144d93782SGreg Clayton         row.x = 0;
296244d93782SGreg Clayton         row.y = 0;
296344d93782SGreg Clayton         ++m_num_rows;
296444d93782SGreg Clayton       }
296544d93782SGreg Clayton 
29668369b28dSGreg Clayton       auto &children = row.GetChildren();
29678369b28dSGreg Clayton       if (row.expanded && !children.empty()) {
29688369b28dSGreg Clayton         DisplayRows(window, children, options);
296944d93782SGreg Clayton       }
297044d93782SGreg Clayton     }
297144d93782SGreg Clayton   }
297244d93782SGreg Clayton 
29738369b28dSGreg Clayton   int CalculateTotalNumberRows(std::vector<Row> &rows) {
297444d93782SGreg Clayton     int row_count = 0;
29758369b28dSGreg Clayton     for (auto &row : rows) {
297644d93782SGreg Clayton       ++row_count;
297744d93782SGreg Clayton       if (row.expanded)
29788369b28dSGreg Clayton         row_count += CalculateTotalNumberRows(row.GetChildren());
297944d93782SGreg Clayton     }
298044d93782SGreg Clayton     return row_count;
298144d93782SGreg Clayton   }
2982315b6884SEugene Zelenko 
2983b9c1b51eSKate Stone   static Row *GetRowForRowIndexImpl(std::vector<Row> &rows, size_t &row_index) {
2984b9c1b51eSKate Stone     for (auto &row : rows) {
298544d93782SGreg Clayton       if (row_index == 0)
298644d93782SGreg Clayton         return &row;
2987b9c1b51eSKate Stone       else {
298844d93782SGreg Clayton         --row_index;
29898369b28dSGreg Clayton         auto &children = row.GetChildren();
29908369b28dSGreg Clayton         if (row.expanded && !children.empty()) {
29918369b28dSGreg Clayton           Row *result = GetRowForRowIndexImpl(children, row_index);
299244d93782SGreg Clayton           if (result)
299344d93782SGreg Clayton             return result;
299444d93782SGreg Clayton         }
299544d93782SGreg Clayton       }
299644d93782SGreg Clayton     }
2997c5dac77aSEugene Zelenko     return nullptr;
299844d93782SGreg Clayton   }
299944d93782SGreg Clayton 
3000b9c1b51eSKate Stone   Row *GetRowForRowIndex(size_t row_index) {
300144d93782SGreg Clayton     return GetRowForRowIndexImpl(m_rows, row_index);
300244d93782SGreg Clayton   }
300344d93782SGreg Clayton 
3004b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
300544d93782SGreg Clayton 
300644d93782SGreg Clayton   static DisplayOptions g_options;
300744d93782SGreg Clayton };
300844d93782SGreg Clayton 
3009b9c1b51eSKate Stone class FrameVariablesWindowDelegate : public ValueObjectListDelegate {
301044d93782SGreg Clayton public:
3011b9c1b51eSKate Stone   FrameVariablesWindowDelegate(Debugger &debugger)
3012b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger),
3013b9c1b51eSKate Stone         m_frame_block(nullptr) {}
301444d93782SGreg Clayton 
3015315b6884SEugene Zelenko   ~FrameVariablesWindowDelegate() override = default;
301644d93782SGreg Clayton 
3017b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
301844d93782SGreg Clayton     return "Frame variable window keyboard shortcuts:";
301944d93782SGreg Clayton   }
302044d93782SGreg Clayton 
3021b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3022b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3023b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
302444d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3025c5dac77aSEugene Zelenko     Block *frame_block = nullptr;
3026c5dac77aSEugene Zelenko     StackFrame *frame = nullptr;
302744d93782SGreg Clayton 
3028b9c1b51eSKate Stone     if (process) {
302944d93782SGreg Clayton       StateType state = process->GetState();
3030b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
303144d93782SGreg Clayton         frame = exe_ctx.GetFramePtr();
303244d93782SGreg Clayton         if (frame)
303344d93782SGreg Clayton           frame_block = frame->GetFrameBlock();
3034b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
303544d93782SGreg Clayton         return true; // Don't do any updating when we are running
303644d93782SGreg Clayton       }
303744d93782SGreg Clayton     }
303844d93782SGreg Clayton 
303944d93782SGreg Clayton     ValueObjectList local_values;
3040b9c1b51eSKate Stone     if (frame_block) {
304144d93782SGreg Clayton       // Only update the variables if they have changed
3042b9c1b51eSKate Stone       if (m_frame_block != frame_block) {
304344d93782SGreg Clayton         m_frame_block = frame_block;
304444d93782SGreg Clayton 
304544d93782SGreg Clayton         VariableList *locals = frame->GetVariableList(true);
3046b9c1b51eSKate Stone         if (locals) {
304744d93782SGreg Clayton           const DynamicValueType use_dynamic = eDynamicDontRunTarget;
304844d93782SGreg Clayton           const size_t num_locals = locals->GetSize();
3049b9c1b51eSKate Stone           for (size_t i = 0; i < num_locals; ++i) {
3050b9c1b51eSKate Stone             ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable(
3051b9c1b51eSKate Stone                 locals->GetVariableAtIndex(i), use_dynamic);
3052b9c1b51eSKate Stone             if (value_sp) {
3053eb72dc7dSGreg Clayton               ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3054eb72dc7dSGreg Clayton               if (synthetic_value_sp)
3055eb72dc7dSGreg Clayton                 local_values.Append(synthetic_value_sp);
3056eb72dc7dSGreg Clayton               else
3057eb72dc7dSGreg Clayton                 local_values.Append(value_sp);
3058eb72dc7dSGreg Clayton             }
3059eb72dc7dSGreg Clayton           }
306044d93782SGreg Clayton           // Update the values
306144d93782SGreg Clayton           SetValues(local_values);
306244d93782SGreg Clayton         }
306344d93782SGreg Clayton       }
3064b9c1b51eSKate Stone     } else {
3065c5dac77aSEugene Zelenko       m_frame_block = nullptr;
306644d93782SGreg Clayton       // Update the values with an empty list if there is no frame
306744d93782SGreg Clayton       SetValues(local_values);
306844d93782SGreg Clayton     }
306944d93782SGreg Clayton 
307044d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
307144d93782SGreg Clayton   }
307244d93782SGreg Clayton 
307344d93782SGreg Clayton protected:
307444d93782SGreg Clayton   Debugger &m_debugger;
307544d93782SGreg Clayton   Block *m_frame_block;
307644d93782SGreg Clayton };
307744d93782SGreg Clayton 
3078b9c1b51eSKate Stone class RegistersWindowDelegate : public ValueObjectListDelegate {
307944d93782SGreg Clayton public:
3080b9c1b51eSKate Stone   RegistersWindowDelegate(Debugger &debugger)
3081b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger) {}
308244d93782SGreg Clayton 
3083315b6884SEugene Zelenko   ~RegistersWindowDelegate() override = default;
308444d93782SGreg Clayton 
3085b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
308644d93782SGreg Clayton     return "Register window keyboard shortcuts:";
308744d93782SGreg Clayton   }
308844d93782SGreg Clayton 
3089b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3090b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3091b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
309244d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
309344d93782SGreg Clayton 
309444d93782SGreg Clayton     ValueObjectList value_list;
3095b9c1b51eSKate Stone     if (frame) {
3096b9c1b51eSKate Stone       if (frame->GetStackID() != m_stack_id) {
309744d93782SGreg Clayton         m_stack_id = frame->GetStackID();
309844d93782SGreg Clayton         RegisterContextSP reg_ctx(frame->GetRegisterContext());
3099b9c1b51eSKate Stone         if (reg_ctx) {
310044d93782SGreg Clayton           const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
3101b9c1b51eSKate Stone           for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) {
3102b9c1b51eSKate Stone             value_list.Append(
3103b9c1b51eSKate Stone                 ValueObjectRegisterSet::Create(frame, reg_ctx, set_idx));
310444d93782SGreg Clayton           }
310544d93782SGreg Clayton         }
310644d93782SGreg Clayton         SetValues(value_list);
310744d93782SGreg Clayton       }
3108b9c1b51eSKate Stone     } else {
310944d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
311044d93782SGreg Clayton       if (process && process->IsAlive())
311144d93782SGreg Clayton         return true; // Don't do any updating if we are running
3112b9c1b51eSKate Stone       else {
311305097246SAdrian Prantl         // Update the values with an empty list if there is no process or the
311405097246SAdrian Prantl         // process isn't alive anymore
311544d93782SGreg Clayton         SetValues(value_list);
311644d93782SGreg Clayton       }
311744d93782SGreg Clayton     }
311844d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
311944d93782SGreg Clayton   }
312044d93782SGreg Clayton 
312144d93782SGreg Clayton protected:
312244d93782SGreg Clayton   Debugger &m_debugger;
312344d93782SGreg Clayton   StackID m_stack_id;
312444d93782SGreg Clayton };
312544d93782SGreg Clayton 
3126b9c1b51eSKate Stone static const char *CursesKeyToCString(int ch) {
312744d93782SGreg Clayton   static char g_desc[32];
3128b9c1b51eSKate Stone   if (ch >= KEY_F0 && ch < KEY_F0 + 64) {
312944d93782SGreg Clayton     snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
313044d93782SGreg Clayton     return g_desc;
313144d93782SGreg Clayton   }
3132b9c1b51eSKate Stone   switch (ch) {
3133b9c1b51eSKate Stone   case KEY_DOWN:
3134b9c1b51eSKate Stone     return "down";
3135b9c1b51eSKate Stone   case KEY_UP:
3136b9c1b51eSKate Stone     return "up";
3137b9c1b51eSKate Stone   case KEY_LEFT:
3138b9c1b51eSKate Stone     return "left";
3139b9c1b51eSKate Stone   case KEY_RIGHT:
3140b9c1b51eSKate Stone     return "right";
3141b9c1b51eSKate Stone   case KEY_HOME:
3142b9c1b51eSKate Stone     return "home";
3143b9c1b51eSKate Stone   case KEY_BACKSPACE:
3144b9c1b51eSKate Stone     return "backspace";
3145b9c1b51eSKate Stone   case KEY_DL:
3146b9c1b51eSKate Stone     return "delete-line";
3147b9c1b51eSKate Stone   case KEY_IL:
3148b9c1b51eSKate Stone     return "insert-line";
3149b9c1b51eSKate Stone   case KEY_DC:
3150b9c1b51eSKate Stone     return "delete-char";
3151b9c1b51eSKate Stone   case KEY_IC:
3152b9c1b51eSKate Stone     return "insert-char";
3153b9c1b51eSKate Stone   case KEY_CLEAR:
3154b9c1b51eSKate Stone     return "clear";
3155b9c1b51eSKate Stone   case KEY_EOS:
3156b9c1b51eSKate Stone     return "clear-to-eos";
3157b9c1b51eSKate Stone   case KEY_EOL:
3158b9c1b51eSKate Stone     return "clear-to-eol";
3159b9c1b51eSKate Stone   case KEY_SF:
3160b9c1b51eSKate Stone     return "scroll-forward";
3161b9c1b51eSKate Stone   case KEY_SR:
3162b9c1b51eSKate Stone     return "scroll-backward";
3163b9c1b51eSKate Stone   case KEY_NPAGE:
3164b9c1b51eSKate Stone     return "page-down";
3165b9c1b51eSKate Stone   case KEY_PPAGE:
3166b9c1b51eSKate Stone     return "page-up";
3167b9c1b51eSKate Stone   case KEY_STAB:
3168b9c1b51eSKate Stone     return "set-tab";
3169b9c1b51eSKate Stone   case KEY_CTAB:
3170b9c1b51eSKate Stone     return "clear-tab";
3171b9c1b51eSKate Stone   case KEY_CATAB:
3172b9c1b51eSKate Stone     return "clear-all-tabs";
3173b9c1b51eSKate Stone   case KEY_ENTER:
3174b9c1b51eSKate Stone     return "enter";
3175b9c1b51eSKate Stone   case KEY_PRINT:
3176b9c1b51eSKate Stone     return "print";
3177b9c1b51eSKate Stone   case KEY_LL:
3178b9c1b51eSKate Stone     return "lower-left key";
3179b9c1b51eSKate Stone   case KEY_A1:
3180b9c1b51eSKate Stone     return "upper left of keypad";
3181b9c1b51eSKate Stone   case KEY_A3:
3182b9c1b51eSKate Stone     return "upper right of keypad";
3183b9c1b51eSKate Stone   case KEY_B2:
3184b9c1b51eSKate Stone     return "center of keypad";
3185b9c1b51eSKate Stone   case KEY_C1:
3186b9c1b51eSKate Stone     return "lower left of keypad";
3187b9c1b51eSKate Stone   case KEY_C3:
3188b9c1b51eSKate Stone     return "lower right of keypad";
3189b9c1b51eSKate Stone   case KEY_BTAB:
3190b9c1b51eSKate Stone     return "back-tab key";
3191b9c1b51eSKate Stone   case KEY_BEG:
3192b9c1b51eSKate Stone     return "begin key";
3193b9c1b51eSKate Stone   case KEY_CANCEL:
3194b9c1b51eSKate Stone     return "cancel key";
3195b9c1b51eSKate Stone   case KEY_CLOSE:
3196b9c1b51eSKate Stone     return "close key";
3197b9c1b51eSKate Stone   case KEY_COMMAND:
3198b9c1b51eSKate Stone     return "command key";
3199b9c1b51eSKate Stone   case KEY_COPY:
3200b9c1b51eSKate Stone     return "copy key";
3201b9c1b51eSKate Stone   case KEY_CREATE:
3202b9c1b51eSKate Stone     return "create key";
3203b9c1b51eSKate Stone   case KEY_END:
3204b9c1b51eSKate Stone     return "end key";
3205b9c1b51eSKate Stone   case KEY_EXIT:
3206b9c1b51eSKate Stone     return "exit key";
3207b9c1b51eSKate Stone   case KEY_FIND:
3208b9c1b51eSKate Stone     return "find key";
3209b9c1b51eSKate Stone   case KEY_HELP:
3210b9c1b51eSKate Stone     return "help key";
3211b9c1b51eSKate Stone   case KEY_MARK:
3212b9c1b51eSKate Stone     return "mark key";
3213b9c1b51eSKate Stone   case KEY_MESSAGE:
3214b9c1b51eSKate Stone     return "message key";
3215b9c1b51eSKate Stone   case KEY_MOVE:
3216b9c1b51eSKate Stone     return "move key";
3217b9c1b51eSKate Stone   case KEY_NEXT:
3218b9c1b51eSKate Stone     return "next key";
3219b9c1b51eSKate Stone   case KEY_OPEN:
3220b9c1b51eSKate Stone     return "open key";
3221b9c1b51eSKate Stone   case KEY_OPTIONS:
3222b9c1b51eSKate Stone     return "options key";
3223b9c1b51eSKate Stone   case KEY_PREVIOUS:
3224b9c1b51eSKate Stone     return "previous key";
3225b9c1b51eSKate Stone   case KEY_REDO:
3226b9c1b51eSKate Stone     return "redo key";
3227b9c1b51eSKate Stone   case KEY_REFERENCE:
3228b9c1b51eSKate Stone     return "reference key";
3229b9c1b51eSKate Stone   case KEY_REFRESH:
3230b9c1b51eSKate Stone     return "refresh key";
3231b9c1b51eSKate Stone   case KEY_REPLACE:
3232b9c1b51eSKate Stone     return "replace key";
3233b9c1b51eSKate Stone   case KEY_RESTART:
3234b9c1b51eSKate Stone     return "restart key";
3235b9c1b51eSKate Stone   case KEY_RESUME:
3236b9c1b51eSKate Stone     return "resume key";
3237b9c1b51eSKate Stone   case KEY_SAVE:
3238b9c1b51eSKate Stone     return "save key";
3239b9c1b51eSKate Stone   case KEY_SBEG:
3240b9c1b51eSKate Stone     return "shifted begin key";
3241b9c1b51eSKate Stone   case KEY_SCANCEL:
3242b9c1b51eSKate Stone     return "shifted cancel key";
3243b9c1b51eSKate Stone   case KEY_SCOMMAND:
3244b9c1b51eSKate Stone     return "shifted command key";
3245b9c1b51eSKate Stone   case KEY_SCOPY:
3246b9c1b51eSKate Stone     return "shifted copy key";
3247b9c1b51eSKate Stone   case KEY_SCREATE:
3248b9c1b51eSKate Stone     return "shifted create key";
3249b9c1b51eSKate Stone   case KEY_SDC:
3250b9c1b51eSKate Stone     return "shifted delete-character key";
3251b9c1b51eSKate Stone   case KEY_SDL:
3252b9c1b51eSKate Stone     return "shifted delete-line key";
3253b9c1b51eSKate Stone   case KEY_SELECT:
3254b9c1b51eSKate Stone     return "select key";
3255b9c1b51eSKate Stone   case KEY_SEND:
3256b9c1b51eSKate Stone     return "shifted end key";
3257b9c1b51eSKate Stone   case KEY_SEOL:
3258b9c1b51eSKate Stone     return "shifted clear-to-end-of-line key";
3259b9c1b51eSKate Stone   case KEY_SEXIT:
3260b9c1b51eSKate Stone     return "shifted exit key";
3261b9c1b51eSKate Stone   case KEY_SFIND:
3262b9c1b51eSKate Stone     return "shifted find key";
3263b9c1b51eSKate Stone   case KEY_SHELP:
3264b9c1b51eSKate Stone     return "shifted help key";
3265b9c1b51eSKate Stone   case KEY_SHOME:
3266b9c1b51eSKate Stone     return "shifted home key";
3267b9c1b51eSKate Stone   case KEY_SIC:
3268b9c1b51eSKate Stone     return "shifted insert-character key";
3269b9c1b51eSKate Stone   case KEY_SLEFT:
3270b9c1b51eSKate Stone     return "shifted left-arrow key";
3271b9c1b51eSKate Stone   case KEY_SMESSAGE:
3272b9c1b51eSKate Stone     return "shifted message key";
3273b9c1b51eSKate Stone   case KEY_SMOVE:
3274b9c1b51eSKate Stone     return "shifted move key";
3275b9c1b51eSKate Stone   case KEY_SNEXT:
3276b9c1b51eSKate Stone     return "shifted next key";
3277b9c1b51eSKate Stone   case KEY_SOPTIONS:
3278b9c1b51eSKate Stone     return "shifted options key";
3279b9c1b51eSKate Stone   case KEY_SPREVIOUS:
3280b9c1b51eSKate Stone     return "shifted previous key";
3281b9c1b51eSKate Stone   case KEY_SPRINT:
3282b9c1b51eSKate Stone     return "shifted print key";
3283b9c1b51eSKate Stone   case KEY_SREDO:
3284b9c1b51eSKate Stone     return "shifted redo key";
3285b9c1b51eSKate Stone   case KEY_SREPLACE:
3286b9c1b51eSKate Stone     return "shifted replace key";
3287b9c1b51eSKate Stone   case KEY_SRIGHT:
3288b9c1b51eSKate Stone     return "shifted right-arrow key";
3289b9c1b51eSKate Stone   case KEY_SRSUME:
3290b9c1b51eSKate Stone     return "shifted resume key";
3291b9c1b51eSKate Stone   case KEY_SSAVE:
3292b9c1b51eSKate Stone     return "shifted save key";
3293b9c1b51eSKate Stone   case KEY_SSUSPEND:
3294b9c1b51eSKate Stone     return "shifted suspend key";
3295b9c1b51eSKate Stone   case KEY_SUNDO:
3296b9c1b51eSKate Stone     return "shifted undo key";
3297b9c1b51eSKate Stone   case KEY_SUSPEND:
3298b9c1b51eSKate Stone     return "suspend key";
3299b9c1b51eSKate Stone   case KEY_UNDO:
3300b9c1b51eSKate Stone     return "undo key";
3301b9c1b51eSKate Stone   case KEY_MOUSE:
3302b9c1b51eSKate Stone     return "Mouse event has occurred";
3303b9c1b51eSKate Stone   case KEY_RESIZE:
3304b9c1b51eSKate Stone     return "Terminal resize event";
330527801f4fSBruce Mitchener #ifdef KEY_EVENT
3306b9c1b51eSKate Stone   case KEY_EVENT:
3307b9c1b51eSKate Stone     return "We were interrupted by an event";
330827801f4fSBruce Mitchener #endif
3309b9c1b51eSKate Stone   case KEY_RETURN:
3310b9c1b51eSKate Stone     return "return";
3311b9c1b51eSKate Stone   case ' ':
3312b9c1b51eSKate Stone     return "space";
3313b9c1b51eSKate Stone   case '\t':
3314b9c1b51eSKate Stone     return "tab";
3315b9c1b51eSKate Stone   case KEY_ESCAPE:
3316b9c1b51eSKate Stone     return "escape";
331744d93782SGreg Clayton   default:
331844d93782SGreg Clayton     if (isprint(ch))
331944d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "%c", ch);
332044d93782SGreg Clayton     else
332144d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
332244d93782SGreg Clayton     return g_desc;
332344d93782SGreg Clayton   }
3324c5dac77aSEugene Zelenko   return nullptr;
332544d93782SGreg Clayton }
332644d93782SGreg Clayton 
3327b9c1b51eSKate Stone HelpDialogDelegate::HelpDialogDelegate(const char *text,
3328b9c1b51eSKate Stone                                        KeyHelp *key_help_array)
3329b9c1b51eSKate Stone     : m_text(), m_first_visible_line(0) {
3330b9c1b51eSKate Stone   if (text && text[0]) {
333144d93782SGreg Clayton     m_text.SplitIntoLines(text);
333244d93782SGreg Clayton     m_text.AppendString("");
333344d93782SGreg Clayton   }
3334b9c1b51eSKate Stone   if (key_help_array) {
3335b9c1b51eSKate Stone     for (KeyHelp *key = key_help_array; key->ch; ++key) {
333644d93782SGreg Clayton       StreamString key_description;
3337b9c1b51eSKate Stone       key_description.Printf("%10s - %s", CursesKeyToCString(key->ch),
3338b9c1b51eSKate Stone                              key->description);
3339c156427dSZachary Turner       m_text.AppendString(key_description.GetString());
334044d93782SGreg Clayton     }
334144d93782SGreg Clayton   }
334244d93782SGreg Clayton }
334344d93782SGreg Clayton 
3344315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
334544d93782SGreg Clayton 
3346b9c1b51eSKate Stone bool HelpDialogDelegate::WindowDelegateDraw(Window &window, bool force) {
334744d93782SGreg Clayton   window.Erase();
334844d93782SGreg Clayton   const int window_height = window.GetHeight();
334944d93782SGreg Clayton   int x = 2;
335044d93782SGreg Clayton   int y = 1;
335144d93782SGreg Clayton   const int min_y = y;
335244d93782SGreg Clayton   const int max_y = window_height - 1 - y;
33533985c8c6SSaleem Abdulrasool   const size_t num_visible_lines = max_y - min_y + 1;
335444d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
335544d93782SGreg Clayton   const char *bottom_message;
335644d93782SGreg Clayton   if (num_lines <= num_visible_lines)
335744d93782SGreg Clayton     bottom_message = "Press any key to exit";
335844d93782SGreg Clayton   else
335944d93782SGreg Clayton     bottom_message = "Use arrows to scroll, any other key to exit";
336044d93782SGreg Clayton   window.DrawTitleBox(window.GetName(), bottom_message);
3361b9c1b51eSKate Stone   while (y <= max_y) {
336244d93782SGreg Clayton     window.MoveCursor(x, y);
3363b9c1b51eSKate Stone     window.PutCStringTruncated(
3364b9c1b51eSKate Stone         m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
336544d93782SGreg Clayton     ++y;
336644d93782SGreg Clayton   }
336744d93782SGreg Clayton   return true;
336844d93782SGreg Clayton }
336944d93782SGreg Clayton 
3370b9c1b51eSKate Stone HandleCharResult HelpDialogDelegate::WindowDelegateHandleChar(Window &window,
3371b9c1b51eSKate Stone                                                               int key) {
337244d93782SGreg Clayton   bool done = false;
337344d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
337444d93782SGreg Clayton   const size_t num_visible_lines = window.GetHeight() - 2;
337544d93782SGreg Clayton 
3376b9c1b51eSKate Stone   if (num_lines <= num_visible_lines) {
337744d93782SGreg Clayton     done = true;
337805097246SAdrian Prantl     // If we have all lines visible and don't need scrolling, then any key
337905097246SAdrian Prantl     // press will cause us to exit
3380b9c1b51eSKate Stone   } else {
3381b9c1b51eSKate Stone     switch (key) {
338244d93782SGreg Clayton     case KEY_UP:
338344d93782SGreg Clayton       if (m_first_visible_line > 0)
338444d93782SGreg Clayton         --m_first_visible_line;
338544d93782SGreg Clayton       break;
338644d93782SGreg Clayton 
338744d93782SGreg Clayton     case KEY_DOWN:
338844d93782SGreg Clayton       if (m_first_visible_line + num_visible_lines < num_lines)
338944d93782SGreg Clayton         ++m_first_visible_line;
339044d93782SGreg Clayton       break;
339144d93782SGreg Clayton 
339244d93782SGreg Clayton     case KEY_PPAGE:
339344d93782SGreg Clayton     case ',':
3394b9c1b51eSKate Stone       if (m_first_visible_line > 0) {
33953985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
339644d93782SGreg Clayton           m_first_visible_line -= num_visible_lines;
339744d93782SGreg Clayton         else
339844d93782SGreg Clayton           m_first_visible_line = 0;
339944d93782SGreg Clayton       }
340044d93782SGreg Clayton       break;
3401315b6884SEugene Zelenko 
340244d93782SGreg Clayton     case KEY_NPAGE:
340344d93782SGreg Clayton     case '.':
3404b9c1b51eSKate Stone       if (m_first_visible_line + num_visible_lines < num_lines) {
340544d93782SGreg Clayton         m_first_visible_line += num_visible_lines;
34063985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > num_lines)
340744d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
340844d93782SGreg Clayton       }
340944d93782SGreg Clayton       break;
3410315b6884SEugene Zelenko 
341144d93782SGreg Clayton     default:
341244d93782SGreg Clayton       done = true;
341344d93782SGreg Clayton       break;
341444d93782SGreg Clayton     }
341544d93782SGreg Clayton   }
341644d93782SGreg Clayton   if (done)
341744d93782SGreg Clayton     window.GetParent()->RemoveSubWindow(&window);
341844d93782SGreg Clayton   return eKeyHandled;
341944d93782SGreg Clayton }
342044d93782SGreg Clayton 
3421b9c1b51eSKate Stone class ApplicationDelegate : public WindowDelegate, public MenuDelegate {
342244d93782SGreg Clayton public:
342344d93782SGreg Clayton   enum {
342444d93782SGreg Clayton     eMenuID_LLDB = 1,
342544d93782SGreg Clayton     eMenuID_LLDBAbout,
342644d93782SGreg Clayton     eMenuID_LLDBExit,
342744d93782SGreg Clayton 
342844d93782SGreg Clayton     eMenuID_Target,
342944d93782SGreg Clayton     eMenuID_TargetCreate,
343044d93782SGreg Clayton     eMenuID_TargetDelete,
343144d93782SGreg Clayton 
343244d93782SGreg Clayton     eMenuID_Process,
343344d93782SGreg Clayton     eMenuID_ProcessAttach,
343444d93782SGreg Clayton     eMenuID_ProcessDetach,
343544d93782SGreg Clayton     eMenuID_ProcessLaunch,
343644d93782SGreg Clayton     eMenuID_ProcessContinue,
343744d93782SGreg Clayton     eMenuID_ProcessHalt,
343844d93782SGreg Clayton     eMenuID_ProcessKill,
343944d93782SGreg Clayton 
344044d93782SGreg Clayton     eMenuID_Thread,
344144d93782SGreg Clayton     eMenuID_ThreadStepIn,
344244d93782SGreg Clayton     eMenuID_ThreadStepOver,
344344d93782SGreg Clayton     eMenuID_ThreadStepOut,
344444d93782SGreg Clayton 
344544d93782SGreg Clayton     eMenuID_View,
344644d93782SGreg Clayton     eMenuID_ViewBacktrace,
344744d93782SGreg Clayton     eMenuID_ViewRegisters,
344844d93782SGreg Clayton     eMenuID_ViewSource,
344944d93782SGreg Clayton     eMenuID_ViewVariables,
345044d93782SGreg Clayton 
345144d93782SGreg Clayton     eMenuID_Help,
345244d93782SGreg Clayton     eMenuID_HelpGUIHelp
345344d93782SGreg Clayton   };
345444d93782SGreg Clayton 
3455b9c1b51eSKate Stone   ApplicationDelegate(Application &app, Debugger &debugger)
3456b9c1b51eSKate Stone       : WindowDelegate(), MenuDelegate(), m_app(app), m_debugger(debugger) {}
345744d93782SGreg Clayton 
3458315b6884SEugene Zelenko   ~ApplicationDelegate() override = default;
3459bd5ae6b4SGreg Clayton 
3460b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
346144d93782SGreg Clayton     return false; // Drawing not handled, let standard window drawing happen
346244d93782SGreg Clayton   }
346344d93782SGreg Clayton 
3464b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override {
3465b9c1b51eSKate Stone     switch (key) {
34665fdb09bbSGreg Clayton     case '\t':
346744d93782SGreg Clayton       window.SelectNextWindowAsActive();
346844d93782SGreg Clayton       return eKeyHandled;
34695fdb09bbSGreg Clayton 
34705fdb09bbSGreg Clayton     case 'h':
34715fdb09bbSGreg Clayton       window.CreateHelpSubwindow();
34725fdb09bbSGreg Clayton       return eKeyHandled;
34735fdb09bbSGreg Clayton 
34745fdb09bbSGreg Clayton     case KEY_ESCAPE:
34755fdb09bbSGreg Clayton       return eQuitApplication;
34765fdb09bbSGreg Clayton 
34775fdb09bbSGreg Clayton     default:
34785fdb09bbSGreg Clayton       break;
347944d93782SGreg Clayton     }
348044d93782SGreg Clayton     return eKeyNotHandled;
348144d93782SGreg Clayton   }
348244d93782SGreg Clayton 
3483b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
34845fdb09bbSGreg Clayton     return "Welcome to the LLDB curses GUI.\n\n"
34855fdb09bbSGreg Clayton            "Press the TAB key to change the selected view.\n"
3486b9c1b51eSKate Stone            "Each view has its own keyboard shortcuts, press 'h' to open a "
3487b9c1b51eSKate Stone            "dialog to display them.\n\n"
34885fdb09bbSGreg Clayton            "Common key bindings for all views:";
34895fdb09bbSGreg Clayton   }
34905fdb09bbSGreg Clayton 
3491b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
34925fdb09bbSGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
34935fdb09bbSGreg Clayton         {'\t', "Select next view"},
34945fdb09bbSGreg Clayton         {'h', "Show help dialog with view specific key bindings"},
34955fdb09bbSGreg Clayton         {',', "Page up"},
34965fdb09bbSGreg Clayton         {'.', "Page down"},
34975fdb09bbSGreg Clayton         {KEY_UP, "Select previous"},
34985fdb09bbSGreg Clayton         {KEY_DOWN, "Select next"},
34995fdb09bbSGreg Clayton         {KEY_LEFT, "Unexpand or select parent"},
35005fdb09bbSGreg Clayton         {KEY_RIGHT, "Expand"},
35015fdb09bbSGreg Clayton         {KEY_PPAGE, "Page up"},
35025fdb09bbSGreg Clayton         {KEY_NPAGE, "Page down"},
3503b9c1b51eSKate Stone         {'\0', nullptr}};
35045fdb09bbSGreg Clayton     return g_source_view_key_help;
35055fdb09bbSGreg Clayton   }
35065fdb09bbSGreg Clayton 
3507b9c1b51eSKate Stone   MenuActionResult MenuDelegateAction(Menu &menu) override {
3508b9c1b51eSKate Stone     switch (menu.GetIdentifier()) {
3509b9c1b51eSKate Stone     case eMenuID_ThreadStepIn: {
3510b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3511b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3512b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
351344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3514b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3515b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
35164b4b2478SJim Ingham           exe_ctx.GetThreadRef().StepIn(true);
351744d93782SGreg Clayton       }
351844d93782SGreg Clayton     }
351944d93782SGreg Clayton       return MenuActionResult::Handled;
352044d93782SGreg Clayton 
3521b9c1b51eSKate Stone     case eMenuID_ThreadStepOut: {
3522b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3523b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3524b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
352544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3526b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3527b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
352844d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
352944d93782SGreg Clayton       }
353044d93782SGreg Clayton     }
353144d93782SGreg Clayton       return MenuActionResult::Handled;
353244d93782SGreg Clayton 
3533b9c1b51eSKate Stone     case eMenuID_ThreadStepOver: {
3534b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3535b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3536b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
353744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3538b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3539b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
354044d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOver(true);
354144d93782SGreg Clayton       }
354244d93782SGreg Clayton     }
354344d93782SGreg Clayton       return MenuActionResult::Handled;
354444d93782SGreg Clayton 
3545b9c1b51eSKate Stone     case eMenuID_ProcessContinue: {
3546b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3547b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3548b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
354944d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3550b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3551b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
355244d93782SGreg Clayton           process->Resume();
355344d93782SGreg Clayton       }
355444d93782SGreg Clayton     }
355544d93782SGreg Clayton       return MenuActionResult::Handled;
355644d93782SGreg Clayton 
3557b9c1b51eSKate Stone     case eMenuID_ProcessKill: {
3558b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3559b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3560b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
356144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
356244d93782SGreg Clayton         if (process && process->IsAlive())
3563ede3193bSJason Molenda           process->Destroy(false);
356444d93782SGreg Clayton       }
356544d93782SGreg Clayton     }
356644d93782SGreg Clayton       return MenuActionResult::Handled;
356744d93782SGreg Clayton 
3568b9c1b51eSKate Stone     case eMenuID_ProcessHalt: {
3569b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3570b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3571b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
357244d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
357344d93782SGreg Clayton         if (process && process->IsAlive())
357444d93782SGreg Clayton           process->Halt();
357544d93782SGreg Clayton       }
357644d93782SGreg Clayton     }
357744d93782SGreg Clayton       return MenuActionResult::Handled;
357844d93782SGreg Clayton 
3579b9c1b51eSKate Stone     case eMenuID_ProcessDetach: {
3580b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3581b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3582b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
358344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
358444d93782SGreg Clayton         if (process && process->IsAlive())
358544d93782SGreg Clayton           process->Detach(false);
358644d93782SGreg Clayton       }
358744d93782SGreg Clayton     }
358844d93782SGreg Clayton       return MenuActionResult::Handled;
358944d93782SGreg Clayton 
3590b9c1b51eSKate Stone     case eMenuID_Process: {
3591b9c1b51eSKate Stone       // Populate the menu with all of the threads if the process is stopped
359205097246SAdrian Prantl       // when the Process menu gets selected and is about to display its
359305097246SAdrian Prantl       // submenu.
359444d93782SGreg Clayton       Menus &submenus = menu.GetSubmenus();
3595b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3596b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
359744d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
3598b9c1b51eSKate Stone       if (process && process->IsAlive() &&
3599b9c1b51eSKate Stone           StateIsStoppedState(process->GetState(), true)) {
360044d93782SGreg Clayton         if (submenus.size() == 7)
360144d93782SGreg Clayton           menu.AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
360244d93782SGreg Clayton         else if (submenus.size() > 8)
360344d93782SGreg Clayton           submenus.erase(submenus.begin() + 8, submenus.end());
360444d93782SGreg Clayton 
360544d93782SGreg Clayton         ThreadList &threads = process->GetThreadList();
3606bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
360744d93782SGreg Clayton         size_t num_threads = threads.GetSize();
3608b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
360944d93782SGreg Clayton           ThreadSP thread_sp = threads.GetThreadAtIndex(i);
361044d93782SGreg Clayton           char menu_char = '\0';
361144d93782SGreg Clayton           if (i < 9)
361244d93782SGreg Clayton             menu_char = '1' + i;
361344d93782SGreg Clayton           StreamString thread_menu_title;
361444d93782SGreg Clayton           thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
361544d93782SGreg Clayton           const char *thread_name = thread_sp->GetName();
361644d93782SGreg Clayton           if (thread_name && thread_name[0])
361744d93782SGreg Clayton             thread_menu_title.Printf(" %s", thread_name);
3618b9c1b51eSKate Stone           else {
361944d93782SGreg Clayton             const char *queue_name = thread_sp->GetQueueName();
362044d93782SGreg Clayton             if (queue_name && queue_name[0])
362144d93782SGreg Clayton               thread_menu_title.Printf(" %s", queue_name);
362244d93782SGreg Clayton           }
3623b9c1b51eSKate Stone           menu.AddSubmenu(
3624c156427dSZachary Turner               MenuSP(new Menu(thread_menu_title.GetString().str().c_str(),
3625c156427dSZachary Turner                               nullptr, menu_char, thread_sp->GetID())));
362644d93782SGreg Clayton         }
3627b9c1b51eSKate Stone       } else if (submenus.size() > 7) {
362805097246SAdrian Prantl         // Remove the separator and any other thread submenu items that were
362905097246SAdrian Prantl         // previously added
363044d93782SGreg Clayton         submenus.erase(submenus.begin() + 7, submenus.end());
363144d93782SGreg Clayton       }
3632b9c1b51eSKate Stone       // Since we are adding and removing items we need to recalculate the name
3633b9c1b51eSKate Stone       // lengths
363444d93782SGreg Clayton       menu.RecalculateNameLengths();
363544d93782SGreg Clayton     }
363644d93782SGreg Clayton       return MenuActionResult::Handled;
363744d93782SGreg Clayton 
3638b9c1b51eSKate Stone     case eMenuID_ViewVariables: {
363944d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
364044d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
364144d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
364244d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
364344d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
364444d93782SGreg Clayton 
3645b9c1b51eSKate Stone       if (variables_window_sp) {
364644d93782SGreg Clayton         const Rect variables_bounds = variables_window_sp->GetBounds();
364744d93782SGreg Clayton 
364844d93782SGreg Clayton         main_window_sp->RemoveSubWindow(variables_window_sp.get());
364944d93782SGreg Clayton 
3650b9c1b51eSKate Stone         if (registers_window_sp) {
3651b9c1b51eSKate Stone           // We have a registers window, so give all the area back to the
3652b9c1b51eSKate Stone           // registers window
365344d93782SGreg Clayton           Rect registers_bounds = variables_bounds;
365444d93782SGreg Clayton           registers_bounds.size.width = source_bounds.size.width;
365544d93782SGreg Clayton           registers_window_sp->SetBounds(registers_bounds);
3656b9c1b51eSKate Stone         } else {
365705097246SAdrian Prantl           // We have no registers window showing so give the bottom area back
365805097246SAdrian Prantl           // to the source view
365944d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3660b9c1b51eSKate Stone                                    source_bounds.size.height +
3661b9c1b51eSKate Stone                                        variables_bounds.size.height);
366244d93782SGreg Clayton         }
3663b9c1b51eSKate Stone       } else {
366444d93782SGreg Clayton         Rect new_variables_rect;
3665b9c1b51eSKate Stone         if (registers_window_sp) {
366644d93782SGreg Clayton           // We have a registers window so split the area of the registers
366744d93782SGreg Clayton           // window into two columns where the left hand side will be the
366844d93782SGreg Clayton           // variables and the right hand side will be the registers
366944d93782SGreg Clayton           const Rect variables_bounds = registers_window_sp->GetBounds();
367044d93782SGreg Clayton           Rect new_registers_rect;
3671b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_variables_rect,
3672b9c1b51eSKate Stone                                                    new_registers_rect);
367344d93782SGreg Clayton           registers_window_sp->SetBounds(new_registers_rect);
3674b9c1b51eSKate Stone         } else {
367544d93782SGreg Clayton           // No variables window, grab the bottom part of the source window
367644d93782SGreg Clayton           Rect new_source_rect;
3677b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3678b9c1b51eSKate Stone                                                   new_variables_rect);
367944d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
368044d93782SGreg Clayton         }
3681b9c1b51eSKate Stone         WindowSP new_window_sp = main_window_sp->CreateSubWindow(
3682b9c1b51eSKate Stone             "Variables", new_variables_rect, false);
3683b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3684b9c1b51eSKate Stone             WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
368544d93782SGreg Clayton       }
368644d93782SGreg Clayton       touchwin(stdscr);
368744d93782SGreg Clayton     }
368844d93782SGreg Clayton       return MenuActionResult::Handled;
368944d93782SGreg Clayton 
3690b9c1b51eSKate Stone     case eMenuID_ViewRegisters: {
369144d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
369244d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
369344d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
369444d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
369544d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
369644d93782SGreg Clayton 
3697b9c1b51eSKate Stone       if (registers_window_sp) {
3698b9c1b51eSKate Stone         if (variables_window_sp) {
369944d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
370044d93782SGreg Clayton 
3701b9c1b51eSKate Stone           // We have a variables window, so give all the area back to the
3702b9c1b51eSKate Stone           // variables window
3703b9c1b51eSKate Stone           variables_window_sp->Resize(variables_bounds.size.width +
3704b9c1b51eSKate Stone                                           registers_window_sp->GetWidth(),
370544d93782SGreg Clayton                                       variables_bounds.size.height);
3706b9c1b51eSKate Stone         } else {
370705097246SAdrian Prantl           // We have no variables window showing so give the bottom area back
370805097246SAdrian Prantl           // to the source view
370944d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3710b9c1b51eSKate Stone                                    source_bounds.size.height +
3711b9c1b51eSKate Stone                                        registers_window_sp->GetHeight());
371244d93782SGreg Clayton         }
371344d93782SGreg Clayton         main_window_sp->RemoveSubWindow(registers_window_sp.get());
3714b9c1b51eSKate Stone       } else {
371544d93782SGreg Clayton         Rect new_regs_rect;
3716b9c1b51eSKate Stone         if (variables_window_sp) {
371705097246SAdrian Prantl           // We have a variables window, split it into two columns where the
371805097246SAdrian Prantl           // left hand side will be the variables and the right hand side will
371905097246SAdrian Prantl           // be the registers
372044d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
372144d93782SGreg Clayton           Rect new_vars_rect;
3722b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_vars_rect,
3723b9c1b51eSKate Stone                                                    new_regs_rect);
372444d93782SGreg Clayton           variables_window_sp->SetBounds(new_vars_rect);
3725b9c1b51eSKate Stone         } else {
372644d93782SGreg Clayton           // No registers window, grab the bottom part of the source window
372744d93782SGreg Clayton           Rect new_source_rect;
3728b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3729b9c1b51eSKate Stone                                                   new_regs_rect);
373044d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
373144d93782SGreg Clayton         }
3732b9c1b51eSKate Stone         WindowSP new_window_sp =
3733b9c1b51eSKate Stone             main_window_sp->CreateSubWindow("Registers", new_regs_rect, false);
3734b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3735b9c1b51eSKate Stone             WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
373644d93782SGreg Clayton       }
373744d93782SGreg Clayton       touchwin(stdscr);
373844d93782SGreg Clayton     }
373944d93782SGreg Clayton       return MenuActionResult::Handled;
374044d93782SGreg Clayton 
374144d93782SGreg Clayton     case eMenuID_HelpGUIHelp:
37425fdb09bbSGreg Clayton       m_app.GetMainWindow()->CreateHelpSubwindow();
374344d93782SGreg Clayton       return MenuActionResult::Handled;
374444d93782SGreg Clayton 
374544d93782SGreg Clayton     default:
374644d93782SGreg Clayton       break;
374744d93782SGreg Clayton     }
374844d93782SGreg Clayton 
374944d93782SGreg Clayton     return MenuActionResult::NotHandled;
375044d93782SGreg Clayton   }
3751b9c1b51eSKate Stone 
375244d93782SGreg Clayton protected:
375344d93782SGreg Clayton   Application &m_app;
375444d93782SGreg Clayton   Debugger &m_debugger;
375544d93782SGreg Clayton };
375644d93782SGreg Clayton 
3757b9c1b51eSKate Stone class StatusBarWindowDelegate : public WindowDelegate {
375844d93782SGreg Clayton public:
3759b9c1b51eSKate Stone   StatusBarWindowDelegate(Debugger &debugger) : m_debugger(debugger) {
3760b9c1b51eSKate Stone     FormatEntity::Parse("Thread: ${thread.id%tid}", m_format);
376144d93782SGreg Clayton   }
376244d93782SGreg Clayton 
3763315b6884SEugene Zelenko   ~StatusBarWindowDelegate() override = default;
3764bd5ae6b4SGreg Clayton 
3765b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3766b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3767b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
376844d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
376944d93782SGreg Clayton     Thread *thread = exe_ctx.GetThreadPtr();
377044d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
377144d93782SGreg Clayton     window.Erase();
377244d93782SGreg Clayton     window.SetBackground(2);
377344d93782SGreg Clayton     window.MoveCursor(0, 0);
3774b9c1b51eSKate Stone     if (process) {
377544d93782SGreg Clayton       const StateType state = process->GetState();
3776b9c1b51eSKate Stone       window.Printf("Process: %5" PRIu64 " %10s", process->GetID(),
3777b9c1b51eSKate Stone                     StateAsCString(state));
377844d93782SGreg Clayton 
3779b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
37805b031ebcSEd Maste         StreamString strm;
3781b9c1b51eSKate Stone         if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx,
3782b9c1b51eSKate Stone                                            nullptr, nullptr, false, false)) {
378344d93782SGreg Clayton           window.MoveCursor(40, 0);
3784c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), 1);
37855b031ebcSEd Maste         }
378644d93782SGreg Clayton 
378744d93782SGreg Clayton         window.MoveCursor(60, 0);
378844d93782SGreg Clayton         if (frame)
3789b9c1b51eSKate Stone           window.Printf("Frame: %3u  PC = 0x%16.16" PRIx64,
3790b9c1b51eSKate Stone                         frame->GetFrameIndex(),
3791b9c1b51eSKate Stone                         frame->GetFrameCodeAddress().GetOpcodeLoadAddress(
3792b9c1b51eSKate Stone                             exe_ctx.GetTargetPtr()));
3793b9c1b51eSKate Stone       } else if (state == eStateExited) {
379444d93782SGreg Clayton         const char *exit_desc = process->GetExitDescription();
379544d93782SGreg Clayton         const int exit_status = process->GetExitStatus();
379644d93782SGreg Clayton         if (exit_desc && exit_desc[0])
379744d93782SGreg Clayton           window.Printf(" with status = %i (%s)", exit_status, exit_desc);
379844d93782SGreg Clayton         else
379944d93782SGreg Clayton           window.Printf(" with status = %i", exit_status);
380044d93782SGreg Clayton       }
380144d93782SGreg Clayton     }
380244d93782SGreg Clayton     window.DeferredRefresh();
380344d93782SGreg Clayton     return true;
380444d93782SGreg Clayton   }
380544d93782SGreg Clayton 
380644d93782SGreg Clayton protected:
380744d93782SGreg Clayton   Debugger &m_debugger;
3808554f68d3SGreg Clayton   FormatEntity::Entry m_format;
380944d93782SGreg Clayton };
381044d93782SGreg Clayton 
3811b9c1b51eSKate Stone class SourceFileWindowDelegate : public WindowDelegate {
381244d93782SGreg Clayton public:
3813b9c1b51eSKate Stone   SourceFileWindowDelegate(Debugger &debugger)
3814b9c1b51eSKate Stone       : WindowDelegate(), m_debugger(debugger), m_sc(), m_file_sp(),
3815b9c1b51eSKate Stone         m_disassembly_scope(nullptr), m_disassembly_sp(), m_disassembly_range(),
3816b9c1b51eSKate Stone         m_title(), m_line_width(4), m_selected_line(0), m_pc_line(0),
3817b9c1b51eSKate Stone         m_stop_id(0), m_frame_idx(UINT32_MAX), m_first_visible_line(0),
3818b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
381944d93782SGreg Clayton 
3820315b6884SEugene Zelenko   ~SourceFileWindowDelegate() override = default;
382144d93782SGreg Clayton 
3822b9c1b51eSKate Stone   void Update(const SymbolContext &sc) { m_sc = sc; }
382344d93782SGreg Clayton 
3824b9c1b51eSKate Stone   uint32_t NumVisibleLines() const { return m_max_y - m_min_y; }
382544d93782SGreg Clayton 
3826b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
382744d93782SGreg Clayton     return "Source/Disassembly window keyboard shortcuts:";
382844d93782SGreg Clayton   }
382944d93782SGreg Clayton 
3830b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
383144d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
383244d93782SGreg Clayton         {KEY_RETURN, "Run to selected line with one shot breakpoint"},
383344d93782SGreg Clayton         {KEY_UP, "Select previous source line"},
383444d93782SGreg Clayton         {KEY_DOWN, "Select next source line"},
383544d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
383644d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
383744d93782SGreg Clayton         {'b', "Set breakpoint on selected source/disassembly line"},
383844d93782SGreg Clayton         {'c', "Continue process"},
383944d93782SGreg Clayton         {'d', "Detach and resume process"},
384044d93782SGreg Clayton         {'D', "Detach with process suspended"},
384144d93782SGreg Clayton         {'h', "Show help dialog"},
384244d93782SGreg Clayton         {'k', "Kill process"},
384344d93782SGreg Clayton         {'n', "Step over (source line)"},
384444d93782SGreg Clayton         {'N', "Step over (single instruction)"},
384544d93782SGreg Clayton         {'o', "Step out"},
384644d93782SGreg Clayton         {'s', "Step in (source line)"},
384744d93782SGreg Clayton         {'S', "Step in (single instruction)"},
384844d93782SGreg Clayton         {',', "Page up"},
384944d93782SGreg Clayton         {'.', "Page down"},
3850b9c1b51eSKate Stone         {'\0', nullptr}};
385144d93782SGreg Clayton     return g_source_view_key_help;
385244d93782SGreg Clayton   }
385344d93782SGreg Clayton 
3854b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3855b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3856b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
385744d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3858c5dac77aSEugene Zelenko     Thread *thread = nullptr;
385944d93782SGreg Clayton 
386044d93782SGreg Clayton     bool update_location = false;
3861b9c1b51eSKate Stone     if (process) {
386244d93782SGreg Clayton       StateType state = process->GetState();
3863b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
386444d93782SGreg Clayton         // We are stopped, so it is ok to
386544d93782SGreg Clayton         update_location = true;
386644d93782SGreg Clayton       }
386744d93782SGreg Clayton     }
386844d93782SGreg Clayton 
386944d93782SGreg Clayton     m_min_x = 1;
3870ec990867SGreg Clayton     m_min_y = 2;
387144d93782SGreg Clayton     m_max_x = window.GetMaxX() - 1;
387244d93782SGreg Clayton     m_max_y = window.GetMaxY() - 1;
387344d93782SGreg Clayton 
387444d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
387544d93782SGreg Clayton     StackFrameSP frame_sp;
387644d93782SGreg Clayton     bool set_selected_line_to_pc = false;
387744d93782SGreg Clayton 
3878b9c1b51eSKate Stone     if (update_location) {
387944d93782SGreg Clayton       const bool process_alive = process ? process->IsAlive() : false;
388044d93782SGreg Clayton       bool thread_changed = false;
3881b9c1b51eSKate Stone       if (process_alive) {
388244d93782SGreg Clayton         thread = exe_ctx.GetThreadPtr();
3883b9c1b51eSKate Stone         if (thread) {
388444d93782SGreg Clayton           frame_sp = thread->GetSelectedFrame();
388544d93782SGreg Clayton           auto tid = thread->GetID();
388644d93782SGreg Clayton           thread_changed = tid != m_tid;
388744d93782SGreg Clayton           m_tid = tid;
3888b9c1b51eSKate Stone         } else {
3889b9c1b51eSKate Stone           if (m_tid != LLDB_INVALID_THREAD_ID) {
389044d93782SGreg Clayton             thread_changed = true;
389144d93782SGreg Clayton             m_tid = LLDB_INVALID_THREAD_ID;
389244d93782SGreg Clayton           }
389344d93782SGreg Clayton         }
389444d93782SGreg Clayton       }
389544d93782SGreg Clayton       const uint32_t stop_id = process ? process->GetStopID() : 0;
389644d93782SGreg Clayton       const bool stop_id_changed = stop_id != m_stop_id;
389744d93782SGreg Clayton       bool frame_changed = false;
389844d93782SGreg Clayton       m_stop_id = stop_id;
3899ec990867SGreg Clayton       m_title.Clear();
3900b9c1b51eSKate Stone       if (frame_sp) {
390144d93782SGreg Clayton         m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
3902b9c1b51eSKate Stone         if (m_sc.module_sp) {
3903b9c1b51eSKate Stone           m_title.Printf(
3904b9c1b51eSKate Stone               "%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
3905ec990867SGreg Clayton           ConstString func_name = m_sc.GetFunctionName();
3906ec990867SGreg Clayton           if (func_name)
3907ec990867SGreg Clayton             m_title.Printf("`%s", func_name.GetCString());
3908ec990867SGreg Clayton         }
390944d93782SGreg Clayton         const uint32_t frame_idx = frame_sp->GetFrameIndex();
391044d93782SGreg Clayton         frame_changed = frame_idx != m_frame_idx;
391144d93782SGreg Clayton         m_frame_idx = frame_idx;
3912b9c1b51eSKate Stone       } else {
391344d93782SGreg Clayton         m_sc.Clear(true);
391444d93782SGreg Clayton         frame_changed = m_frame_idx != UINT32_MAX;
391544d93782SGreg Clayton         m_frame_idx = UINT32_MAX;
391644d93782SGreg Clayton       }
391744d93782SGreg Clayton 
3918b9c1b51eSKate Stone       const bool context_changed =
3919b9c1b51eSKate Stone           thread_changed || frame_changed || stop_id_changed;
392044d93782SGreg Clayton 
3921b9c1b51eSKate Stone       if (process_alive) {
3922b9c1b51eSKate Stone         if (m_sc.line_entry.IsValid()) {
392344d93782SGreg Clayton           m_pc_line = m_sc.line_entry.line;
392444d93782SGreg Clayton           if (m_pc_line != UINT32_MAX)
392544d93782SGreg Clayton             --m_pc_line; // Convert to zero based line number...
392644d93782SGreg Clayton           // Update the selected line if the stop ID changed...
392744d93782SGreg Clayton           if (context_changed)
392844d93782SGreg Clayton             m_selected_line = m_pc_line;
392944d93782SGreg Clayton 
3930b9c1b51eSKate Stone           if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) {
393105097246SAdrian Prantl             // Same file, nothing to do, we should either have the lines or not
393205097246SAdrian Prantl             // (source file missing)
3933b9c1b51eSKate Stone             if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) {
393444d93782SGreg Clayton               if (m_selected_line >= m_first_visible_line + num_visible_lines)
393544d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
3936b9c1b51eSKate Stone             } else {
393744d93782SGreg Clayton               if (m_selected_line > 10)
393844d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
393944d93782SGreg Clayton               else
394044d93782SGreg Clayton                 m_first_visible_line = 0;
394144d93782SGreg Clayton             }
3942b9c1b51eSKate Stone           } else {
394344d93782SGreg Clayton             // File changed, set selected line to the line with the PC
394444d93782SGreg Clayton             m_selected_line = m_pc_line;
3945b9c1b51eSKate Stone             m_file_sp =
3946b9c1b51eSKate Stone                 m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
3947b9c1b51eSKate Stone             if (m_file_sp) {
394844d93782SGreg Clayton               const size_t num_lines = m_file_sp->GetNumLines();
394944d93782SGreg Clayton               int m_line_width = 1;
395044d93782SGreg Clayton               for (size_t n = num_lines; n >= 10; n = n / 10)
395144d93782SGreg Clayton                 ++m_line_width;
395244d93782SGreg Clayton 
3953b9c1b51eSKate Stone               snprintf(m_line_format, sizeof(m_line_format), " %%%iu ",
3954b9c1b51eSKate Stone                        m_line_width);
3955b9c1b51eSKate Stone               if (num_lines < num_visible_lines ||
3956b9c1b51eSKate Stone                   m_selected_line < num_visible_lines)
395744d93782SGreg Clayton                 m_first_visible_line = 0;
395844d93782SGreg Clayton               else
395944d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
396044d93782SGreg Clayton             }
396144d93782SGreg Clayton           }
3962b9c1b51eSKate Stone         } else {
396344d93782SGreg Clayton           m_file_sp.reset();
396444d93782SGreg Clayton         }
396544d93782SGreg Clayton 
3966b9c1b51eSKate Stone         if (!m_file_sp || m_file_sp->GetNumLines() == 0) {
396744d93782SGreg Clayton           // Show disassembly
396844d93782SGreg Clayton           bool prefer_file_cache = false;
3969b9c1b51eSKate Stone           if (m_sc.function) {
3970b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.function) {
397144d93782SGreg Clayton               m_disassembly_scope = m_sc.function;
3972b9c1b51eSKate Stone               m_disassembly_sp = m_sc.function->GetInstructions(
3973b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3974b9c1b51eSKate Stone               if (m_disassembly_sp) {
397544d93782SGreg Clayton                 set_selected_line_to_pc = true;
397644d93782SGreg Clayton                 m_disassembly_range = m_sc.function->GetAddressRange();
3977b9c1b51eSKate Stone               } else {
397844d93782SGreg Clayton                 m_disassembly_range.Clear();
397944d93782SGreg Clayton               }
3980b9c1b51eSKate Stone             } else {
398144d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
398244d93782SGreg Clayton             }
3983b9c1b51eSKate Stone           } else if (m_sc.symbol) {
3984b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.symbol) {
398544d93782SGreg Clayton               m_disassembly_scope = m_sc.symbol;
3986b9c1b51eSKate Stone               m_disassembly_sp = m_sc.symbol->GetInstructions(
3987b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3988b9c1b51eSKate Stone               if (m_disassembly_sp) {
398944d93782SGreg Clayton                 set_selected_line_to_pc = true;
3990b9c1b51eSKate Stone                 m_disassembly_range.GetBaseAddress() =
3991b9c1b51eSKate Stone                     m_sc.symbol->GetAddress();
399244d93782SGreg Clayton                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
3993b9c1b51eSKate Stone               } else {
399444d93782SGreg Clayton                 m_disassembly_range.Clear();
399544d93782SGreg Clayton               }
3996b9c1b51eSKate Stone             } else {
399744d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
399844d93782SGreg Clayton             }
399944d93782SGreg Clayton           }
400044d93782SGreg Clayton         }
4001b9c1b51eSKate Stone       } else {
400244d93782SGreg Clayton         m_pc_line = UINT32_MAX;
400344d93782SGreg Clayton       }
400444d93782SGreg Clayton     }
400544d93782SGreg Clayton 
4006ec990867SGreg Clayton     const int window_width = window.GetWidth();
400744d93782SGreg Clayton     window.Erase();
400844d93782SGreg Clayton     window.DrawTitleBox("Sources");
4009b9c1b51eSKate Stone     if (!m_title.GetString().empty()) {
4010ec990867SGreg Clayton       window.AttributeOn(A_REVERSE);
4011ec990867SGreg Clayton       window.MoveCursor(1, 1);
4012ec990867SGreg Clayton       window.PutChar(' ');
4013c156427dSZachary Turner       window.PutCStringTruncated(m_title.GetString().str().c_str(), 1);
4014ec990867SGreg Clayton       int x = window.GetCursorX();
4015b9c1b51eSKate Stone       if (x < window_width - 1) {
4016ec990867SGreg Clayton         window.Printf("%*s", window_width - x - 1, "");
4017ec990867SGreg Clayton       }
4018ec990867SGreg Clayton       window.AttributeOff(A_REVERSE);
4019ec990867SGreg Clayton     }
402044d93782SGreg Clayton 
402144d93782SGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
402244d93782SGreg Clayton     const size_t num_source_lines = GetNumSourceLines();
4023b9c1b51eSKate Stone     if (num_source_lines > 0) {
402444d93782SGreg Clayton       // Display source
402544d93782SGreg Clayton       BreakpointLines bp_lines;
4026b9c1b51eSKate Stone       if (target) {
402744d93782SGreg Clayton         BreakpointList &bp_list = target->GetBreakpointList();
402844d93782SGreg Clayton         const size_t num_bps = bp_list.GetSize();
4029b9c1b51eSKate Stone         for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
403044d93782SGreg Clayton           BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
403144d93782SGreg Clayton           const size_t num_bps_locs = bp_sp->GetNumLocations();
4032b9c1b51eSKate Stone           for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs; ++bp_loc_idx) {
4033b9c1b51eSKate Stone             BreakpointLocationSP bp_loc_sp =
4034b9c1b51eSKate Stone                 bp_sp->GetLocationAtIndex(bp_loc_idx);
403544d93782SGreg Clayton             LineEntry bp_loc_line_entry;
4036b9c1b51eSKate Stone             if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry(
4037b9c1b51eSKate Stone                     bp_loc_line_entry)) {
4038b9c1b51eSKate Stone               if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) {
403944d93782SGreg Clayton                 bp_lines.insert(bp_loc_line_entry.line);
404044d93782SGreg Clayton               }
404144d93782SGreg Clayton             }
404244d93782SGreg Clayton           }
404344d93782SGreg Clayton         }
404444d93782SGreg Clayton       }
404544d93782SGreg Clayton 
404644d93782SGreg Clayton       const attr_t selected_highlight_attr = A_REVERSE;
404744d93782SGreg Clayton       const attr_t pc_highlight_attr = COLOR_PAIR(1);
404844d93782SGreg Clayton 
4049b9c1b51eSKate Stone       for (size_t i = 0; i < num_visible_lines; ++i) {
405044d93782SGreg Clayton         const uint32_t curr_line = m_first_visible_line + i;
4051b9c1b51eSKate Stone         if (curr_line < num_source_lines) {
4052ec990867SGreg Clayton           const int line_y = m_min_y + i;
405344d93782SGreg Clayton           window.MoveCursor(1, line_y);
405444d93782SGreg Clayton           const bool is_pc_line = curr_line == m_pc_line;
405544d93782SGreg Clayton           const bool line_is_selected = m_selected_line == curr_line;
405644d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
405744d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
405844d93782SGreg Clayton           attr_t highlight_attr = 0;
405944d93782SGreg Clayton           attr_t bp_attr = 0;
406044d93782SGreg Clayton           if (is_pc_line)
406144d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
406244d93782SGreg Clayton           else if (line_is_selected)
406344d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
406444d93782SGreg Clayton 
406544d93782SGreg Clayton           if (bp_lines.find(curr_line + 1) != bp_lines.end())
406644d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
406744d93782SGreg Clayton 
406844d93782SGreg Clayton           if (bp_attr)
406944d93782SGreg Clayton             window.AttributeOn(bp_attr);
407044d93782SGreg Clayton 
407144d93782SGreg Clayton           window.Printf(m_line_format, curr_line + 1);
407244d93782SGreg Clayton 
407344d93782SGreg Clayton           if (bp_attr)
407444d93782SGreg Clayton             window.AttributeOff(bp_attr);
407544d93782SGreg Clayton 
407644d93782SGreg Clayton           window.PutChar(ACS_VLINE);
407744d93782SGreg Clayton           // Mark the line with the PC with a diamond
407844d93782SGreg Clayton           if (is_pc_line)
407944d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
408044d93782SGreg Clayton           else
408144d93782SGreg Clayton             window.PutChar(' ');
408244d93782SGreg Clayton 
408344d93782SGreg Clayton           if (highlight_attr)
408444d93782SGreg Clayton             window.AttributeOn(highlight_attr);
4085b9c1b51eSKate Stone           const uint32_t line_len =
4086b9c1b51eSKate Stone               m_file_sp->GetLineLength(curr_line + 1, false);
408744d93782SGreg Clayton           if (line_len > 0)
408844d93782SGreg Clayton             window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
408944d93782SGreg Clayton 
4090b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4091b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
409244d93782SGreg Clayton             StopInfoSP stop_info_sp;
409344d93782SGreg Clayton             if (thread)
409444d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4095b9c1b51eSKate Stone             if (stop_info_sp) {
409644d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4097b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
409844d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4099ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
410044d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4101b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4102b9c1b51eSKate Stone                 // line_y);
4103b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4104b9c1b51eSKate Stone                               stop_description);
410544d93782SGreg Clayton               }
4106b9c1b51eSKate Stone             } else {
4107ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
410844d93782SGreg Clayton             }
410944d93782SGreg Clayton           }
411044d93782SGreg Clayton           if (highlight_attr)
411144d93782SGreg Clayton             window.AttributeOff(highlight_attr);
4112b9c1b51eSKate Stone         } else {
411344d93782SGreg Clayton           break;
411444d93782SGreg Clayton         }
411544d93782SGreg Clayton       }
4116b9c1b51eSKate Stone     } else {
411744d93782SGreg Clayton       size_t num_disassembly_lines = GetNumDisassemblyLines();
4118b9c1b51eSKate Stone       if (num_disassembly_lines > 0) {
411944d93782SGreg Clayton         // Display disassembly
412044d93782SGreg Clayton         BreakpointAddrs bp_file_addrs;
412144d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
4122b9c1b51eSKate Stone         if (target) {
412344d93782SGreg Clayton           BreakpointList &bp_list = target->GetBreakpointList();
412444d93782SGreg Clayton           const size_t num_bps = bp_list.GetSize();
4125b9c1b51eSKate Stone           for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
412644d93782SGreg Clayton             BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
412744d93782SGreg Clayton             const size_t num_bps_locs = bp_sp->GetNumLocations();
4128b9c1b51eSKate Stone             for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs;
4129b9c1b51eSKate Stone                  ++bp_loc_idx) {
4130b9c1b51eSKate Stone               BreakpointLocationSP bp_loc_sp =
4131b9c1b51eSKate Stone                   bp_sp->GetLocationAtIndex(bp_loc_idx);
413244d93782SGreg Clayton               LineEntry bp_loc_line_entry;
4133b9c1b51eSKate Stone               const lldb::addr_t file_addr =
4134b9c1b51eSKate Stone                   bp_loc_sp->GetAddress().GetFileAddress();
4135b9c1b51eSKate Stone               if (file_addr != LLDB_INVALID_ADDRESS) {
413644d93782SGreg Clayton                 if (m_disassembly_range.ContainsFileAddress(file_addr))
413744d93782SGreg Clayton                   bp_file_addrs.insert(file_addr);
413844d93782SGreg Clayton               }
413944d93782SGreg Clayton             }
414044d93782SGreg Clayton           }
414144d93782SGreg Clayton         }
414244d93782SGreg Clayton 
414344d93782SGreg Clayton         const attr_t selected_highlight_attr = A_REVERSE;
414444d93782SGreg Clayton         const attr_t pc_highlight_attr = COLOR_PAIR(1);
414544d93782SGreg Clayton 
414644d93782SGreg Clayton         StreamString strm;
414744d93782SGreg Clayton 
414844d93782SGreg Clayton         InstructionList &insts = m_disassembly_sp->GetInstructionList();
414944d93782SGreg Clayton         Address pc_address;
415044d93782SGreg Clayton 
415144d93782SGreg Clayton         if (frame_sp)
415244d93782SGreg Clayton           pc_address = frame_sp->GetFrameCodeAddress();
4153b9c1b51eSKate Stone         const uint32_t pc_idx =
4154b9c1b51eSKate Stone             pc_address.IsValid()
4155b9c1b51eSKate Stone                 ? insts.GetIndexOfInstructionAtAddress(pc_address)
4156b9c1b51eSKate Stone                 : UINT32_MAX;
4157b9c1b51eSKate Stone         if (set_selected_line_to_pc) {
415844d93782SGreg Clayton           m_selected_line = pc_idx;
415944d93782SGreg Clayton         }
416044d93782SGreg Clayton 
416144d93782SGreg Clayton         const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
41623985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
416344d93782SGreg Clayton           m_first_visible_line = 0;
416444d93782SGreg Clayton 
4165b9c1b51eSKate Stone         if (pc_idx < num_disassembly_lines) {
41663985c8c6SSaleem Abdulrasool           if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
416744d93782SGreg Clayton               pc_idx >= m_first_visible_line + num_visible_lines)
416844d93782SGreg Clayton             m_first_visible_line = pc_idx - non_visible_pc_offset;
416944d93782SGreg Clayton         }
417044d93782SGreg Clayton 
4171b9c1b51eSKate Stone         for (size_t i = 0; i < num_visible_lines; ++i) {
417244d93782SGreg Clayton           const uint32_t inst_idx = m_first_visible_line + i;
417344d93782SGreg Clayton           Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
417444d93782SGreg Clayton           if (!inst)
417544d93782SGreg Clayton             break;
417644d93782SGreg Clayton 
4177ec990867SGreg Clayton           const int line_y = m_min_y + i;
4178ec990867SGreg Clayton           window.MoveCursor(1, line_y);
417944d93782SGreg Clayton           const bool is_pc_line = frame_sp && inst_idx == pc_idx;
418044d93782SGreg Clayton           const bool line_is_selected = m_selected_line == inst_idx;
418144d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
418244d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
418344d93782SGreg Clayton           attr_t highlight_attr = 0;
418444d93782SGreg Clayton           attr_t bp_attr = 0;
418544d93782SGreg Clayton           if (is_pc_line)
418644d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
418744d93782SGreg Clayton           else if (line_is_selected)
418844d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
418944d93782SGreg Clayton 
4190b9c1b51eSKate Stone           if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) !=
4191b9c1b51eSKate Stone               bp_file_addrs.end())
419244d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
419344d93782SGreg Clayton 
419444d93782SGreg Clayton           if (bp_attr)
419544d93782SGreg Clayton             window.AttributeOn(bp_attr);
419644d93782SGreg Clayton 
4197324a1036SSaleem Abdulrasool           window.Printf(" 0x%16.16llx ",
4198b9c1b51eSKate Stone                         static_cast<unsigned long long>(
4199b9c1b51eSKate Stone                             inst->GetAddress().GetLoadAddress(target)));
420044d93782SGreg Clayton 
420144d93782SGreg Clayton           if (bp_attr)
420244d93782SGreg Clayton             window.AttributeOff(bp_attr);
420344d93782SGreg Clayton 
420444d93782SGreg Clayton           window.PutChar(ACS_VLINE);
420544d93782SGreg Clayton           // Mark the line with the PC with a diamond
420644d93782SGreg Clayton           if (is_pc_line)
420744d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
420844d93782SGreg Clayton           else
420944d93782SGreg Clayton             window.PutChar(' ');
421044d93782SGreg Clayton 
421144d93782SGreg Clayton           if (highlight_attr)
421244d93782SGreg Clayton             window.AttributeOn(highlight_attr);
421344d93782SGreg Clayton 
421444d93782SGreg Clayton           const char *mnemonic = inst->GetMnemonic(&exe_ctx);
421544d93782SGreg Clayton           const char *operands = inst->GetOperands(&exe_ctx);
421644d93782SGreg Clayton           const char *comment = inst->GetComment(&exe_ctx);
421744d93782SGreg Clayton 
4218c5dac77aSEugene Zelenko           if (mnemonic != nullptr && mnemonic[0] == '\0')
4219c5dac77aSEugene Zelenko             mnemonic = nullptr;
4220c5dac77aSEugene Zelenko           if (operands != nullptr && operands[0] == '\0')
4221c5dac77aSEugene Zelenko             operands = nullptr;
4222c5dac77aSEugene Zelenko           if (comment != nullptr && comment[0] == '\0')
4223c5dac77aSEugene Zelenko             comment = nullptr;
422444d93782SGreg Clayton 
422544d93782SGreg Clayton           strm.Clear();
422644d93782SGreg Clayton 
4227c5dac77aSEugene Zelenko           if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
422844d93782SGreg Clayton             strm.Printf("%-8s %-25s ; %s", mnemonic, operands, comment);
4229c5dac77aSEugene Zelenko           else if (mnemonic != nullptr && operands != nullptr)
423044d93782SGreg Clayton             strm.Printf("%-8s %s", mnemonic, operands);
4231c5dac77aSEugene Zelenko           else if (mnemonic != nullptr)
423244d93782SGreg Clayton             strm.Printf("%s", mnemonic);
423344d93782SGreg Clayton 
423444d93782SGreg Clayton           int right_pad = 1;
4235c156427dSZachary Turner           window.PutCStringTruncated(strm.GetData(), right_pad);
423644d93782SGreg Clayton 
4237b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4238b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
423944d93782SGreg Clayton             StopInfoSP stop_info_sp;
424044d93782SGreg Clayton             if (thread)
424144d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4242b9c1b51eSKate Stone             if (stop_info_sp) {
424344d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4244b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
424544d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4246ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
424744d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4248b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4249b9c1b51eSKate Stone                 // line_y);
4250b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4251b9c1b51eSKate Stone                               stop_description);
425244d93782SGreg Clayton               }
4253b9c1b51eSKate Stone             } else {
4254ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
425544d93782SGreg Clayton             }
425644d93782SGreg Clayton           }
425744d93782SGreg Clayton           if (highlight_attr)
425844d93782SGreg Clayton             window.AttributeOff(highlight_attr);
425944d93782SGreg Clayton         }
426044d93782SGreg Clayton       }
426144d93782SGreg Clayton     }
426244d93782SGreg Clayton     window.DeferredRefresh();
426344d93782SGreg Clayton     return true; // Drawing handled
426444d93782SGreg Clayton   }
426544d93782SGreg Clayton 
4266b9c1b51eSKate Stone   size_t GetNumLines() {
426744d93782SGreg Clayton     size_t num_lines = GetNumSourceLines();
426844d93782SGreg Clayton     if (num_lines == 0)
426944d93782SGreg Clayton       num_lines = GetNumDisassemblyLines();
427044d93782SGreg Clayton     return num_lines;
427144d93782SGreg Clayton   }
427244d93782SGreg Clayton 
4273b9c1b51eSKate Stone   size_t GetNumSourceLines() const {
427444d93782SGreg Clayton     if (m_file_sp)
427544d93782SGreg Clayton       return m_file_sp->GetNumLines();
427644d93782SGreg Clayton     return 0;
427744d93782SGreg Clayton   }
4278315b6884SEugene Zelenko 
4279b9c1b51eSKate Stone   size_t GetNumDisassemblyLines() const {
428044d93782SGreg Clayton     if (m_disassembly_sp)
428144d93782SGreg Clayton       return m_disassembly_sp->GetInstructionList().GetSize();
428244d93782SGreg Clayton     return 0;
428344d93782SGreg Clayton   }
428444d93782SGreg Clayton 
4285b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
428644d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
428744d93782SGreg Clayton     const size_t num_lines = GetNumLines();
428844d93782SGreg Clayton 
4289b9c1b51eSKate Stone     switch (c) {
429044d93782SGreg Clayton     case ',':
429144d93782SGreg Clayton     case KEY_PPAGE:
429244d93782SGreg Clayton       // Page up key
42933985c8c6SSaleem Abdulrasool       if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
429444d93782SGreg Clayton         m_first_visible_line -= num_visible_lines;
429544d93782SGreg Clayton       else
429644d93782SGreg Clayton         m_first_visible_line = 0;
429744d93782SGreg Clayton       m_selected_line = m_first_visible_line;
429844d93782SGreg Clayton       return eKeyHandled;
429944d93782SGreg Clayton 
430044d93782SGreg Clayton     case '.':
430144d93782SGreg Clayton     case KEY_NPAGE:
430244d93782SGreg Clayton       // Page down key
430344d93782SGreg Clayton       {
430444d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < num_lines)
430544d93782SGreg Clayton           m_first_visible_line += num_visible_lines;
430644d93782SGreg Clayton         else if (num_lines < num_visible_lines)
430744d93782SGreg Clayton           m_first_visible_line = 0;
430844d93782SGreg Clayton         else
430944d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
431044d93782SGreg Clayton         m_selected_line = m_first_visible_line;
431144d93782SGreg Clayton       }
431244d93782SGreg Clayton       return eKeyHandled;
431344d93782SGreg Clayton 
431444d93782SGreg Clayton     case KEY_UP:
4315b9c1b51eSKate Stone       if (m_selected_line > 0) {
431644d93782SGreg Clayton         m_selected_line--;
43173985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
431844d93782SGreg Clayton           m_first_visible_line = m_selected_line;
431944d93782SGreg Clayton       }
432044d93782SGreg Clayton       return eKeyHandled;
432144d93782SGreg Clayton 
432244d93782SGreg Clayton     case KEY_DOWN:
4323b9c1b51eSKate Stone       if (m_selected_line + 1 < num_lines) {
432444d93782SGreg Clayton         m_selected_line++;
432544d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < m_selected_line)
432644d93782SGreg Clayton           m_first_visible_line++;
432744d93782SGreg Clayton       }
432844d93782SGreg Clayton       return eKeyHandled;
432944d93782SGreg Clayton 
433044d93782SGreg Clayton     case '\r':
433144d93782SGreg Clayton     case '\n':
433244d93782SGreg Clayton     case KEY_ENTER:
433344d93782SGreg Clayton       // Set a breakpoint and run to the line using a one shot breakpoint
4334b9c1b51eSKate Stone       if (GetNumSourceLines() > 0) {
4335b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4336b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4337b9c1b51eSKate Stone         if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) {
4338b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4339b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
434044d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4341b9c1b51eSKate Stone               m_selected_line +
4342b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
43432411167fSJim Ingham               0,     // No offset
434444d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
434544d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
434644d93782SGreg Clayton               false,               // internal
4347055ad9beSIlia K               false,               // request_hardware
4348055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
434944d93782SGreg Clayton           // Make breakpoint one shot
435044d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
435144d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
435244d93782SGreg Clayton         }
4353b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4354b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4355b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4356b9c1b51eSKate Stone                                       .get();
4357b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4358b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4359b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
436044d93782SGreg Clayton           Address addr = inst->GetAddress();
4361b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4362b9c1b51eSKate Stone               addr,   // lldb_private::Address
436344d93782SGreg Clayton               false,  // internal
436444d93782SGreg Clayton               false); // request_hardware
436544d93782SGreg Clayton           // Make breakpoint one shot
436644d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
436744d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
436844d93782SGreg Clayton         }
436944d93782SGreg Clayton       }
437044d93782SGreg Clayton       return eKeyHandled;
437144d93782SGreg Clayton 
437244d93782SGreg Clayton     case 'b': // 'b' == toggle breakpoint on currently selected line
4373b9c1b51eSKate Stone       if (m_selected_line < GetNumSourceLines()) {
4374b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4375b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4376b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
4377b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4378b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
437944d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4380b9c1b51eSKate Stone               m_selected_line +
4381b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
43822411167fSJim Ingham               0,     // No offset
438344d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
438444d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
438544d93782SGreg Clayton               false,               // internal
4386055ad9beSIlia K               false,               // request_hardware
4387055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
438844d93782SGreg Clayton         }
4389b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4390b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4391b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4392b9c1b51eSKate Stone                                       .get();
4393b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4394b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4395b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
439644d93782SGreg Clayton           Address addr = inst->GetAddress();
4397b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4398b9c1b51eSKate Stone               addr,   // lldb_private::Address
439944d93782SGreg Clayton               false,  // internal
440044d93782SGreg Clayton               false); // request_hardware
440144d93782SGreg Clayton         }
440244d93782SGreg Clayton       }
440344d93782SGreg Clayton       return eKeyHandled;
440444d93782SGreg Clayton 
440544d93782SGreg Clayton     case 'd': // 'd' == detach and let run
440644d93782SGreg Clayton     case 'D': // 'D' == detach and keep stopped
440744d93782SGreg Clayton     {
4408b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4409b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
441044d93782SGreg Clayton       if (exe_ctx.HasProcessScope())
441144d93782SGreg Clayton         exe_ctx.GetProcessRef().Detach(c == 'D');
441244d93782SGreg Clayton     }
441344d93782SGreg Clayton       return eKeyHandled;
441444d93782SGreg Clayton 
441544d93782SGreg Clayton     case 'k':
441644d93782SGreg Clayton       // 'k' == kill
441744d93782SGreg Clayton       {
4418b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4419b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
442044d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
4421ede3193bSJason Molenda           exe_ctx.GetProcessRef().Destroy(false);
442244d93782SGreg Clayton       }
442344d93782SGreg Clayton       return eKeyHandled;
442444d93782SGreg Clayton 
442544d93782SGreg Clayton     case 'c':
442644d93782SGreg Clayton       // 'c' == continue
442744d93782SGreg Clayton       {
4428b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4429b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
443044d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
443144d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
443244d93782SGreg Clayton       }
443344d93782SGreg Clayton       return eKeyHandled;
443444d93782SGreg Clayton 
443544d93782SGreg Clayton     case 'o':
443644d93782SGreg Clayton       // 'o' == step out
443744d93782SGreg Clayton       {
4438b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4439b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4440b9c1b51eSKate Stone         if (exe_ctx.HasThreadScope() &&
4441b9c1b51eSKate Stone             StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
444244d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
444344d93782SGreg Clayton         }
444444d93782SGreg Clayton       }
444544d93782SGreg Clayton       return eKeyHandled;
4446315b6884SEugene Zelenko 
444744d93782SGreg Clayton     case 'n': // 'n' == step over
444844d93782SGreg Clayton     case 'N': // 'N' == step over instruction
444944d93782SGreg Clayton     {
4450b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4451b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4452b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4453b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
445444d93782SGreg Clayton         bool source_step = (c == 'n');
445544d93782SGreg Clayton         exe_ctx.GetThreadRef().StepOver(source_step);
445644d93782SGreg Clayton       }
445744d93782SGreg Clayton     }
445844d93782SGreg Clayton       return eKeyHandled;
4459315b6884SEugene Zelenko 
446044d93782SGreg Clayton     case 's': // 's' == step into
446144d93782SGreg Clayton     case 'S': // 'S' == step into instruction
446244d93782SGreg Clayton     {
4463b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4464b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4465b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4466b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
446744d93782SGreg Clayton         bool source_step = (c == 's');
44684b4b2478SJim Ingham         exe_ctx.GetThreadRef().StepIn(source_step);
446944d93782SGreg Clayton       }
447044d93782SGreg Clayton     }
447144d93782SGreg Clayton       return eKeyHandled;
447244d93782SGreg Clayton 
447344d93782SGreg Clayton     case 'h':
447444d93782SGreg Clayton       window.CreateHelpSubwindow();
447544d93782SGreg Clayton       return eKeyHandled;
447644d93782SGreg Clayton 
447744d93782SGreg Clayton     default:
447844d93782SGreg Clayton       break;
447944d93782SGreg Clayton     }
448044d93782SGreg Clayton     return eKeyNotHandled;
448144d93782SGreg Clayton   }
448244d93782SGreg Clayton 
448344d93782SGreg Clayton protected:
448444d93782SGreg Clayton   typedef std::set<uint32_t> BreakpointLines;
448544d93782SGreg Clayton   typedef std::set<lldb::addr_t> BreakpointAddrs;
448644d93782SGreg Clayton 
448744d93782SGreg Clayton   Debugger &m_debugger;
448844d93782SGreg Clayton   SymbolContext m_sc;
448944d93782SGreg Clayton   SourceManager::FileSP m_file_sp;
449044d93782SGreg Clayton   SymbolContextScope *m_disassembly_scope;
449144d93782SGreg Clayton   lldb::DisassemblerSP m_disassembly_sp;
449244d93782SGreg Clayton   AddressRange m_disassembly_range;
4493ec990867SGreg Clayton   StreamString m_title;
449444d93782SGreg Clayton   lldb::user_id_t m_tid;
449544d93782SGreg Clayton   char m_line_format[8];
449644d93782SGreg Clayton   int m_line_width;
449744d93782SGreg Clayton   uint32_t m_selected_line; // The selected line
449844d93782SGreg Clayton   uint32_t m_pc_line;       // The line with the PC
449944d93782SGreg Clayton   uint32_t m_stop_id;
450044d93782SGreg Clayton   uint32_t m_frame_idx;
450144d93782SGreg Clayton   int m_first_visible_line;
450244d93782SGreg Clayton   int m_min_x;
450344d93782SGreg Clayton   int m_min_y;
450444d93782SGreg Clayton   int m_max_x;
450544d93782SGreg Clayton   int m_max_y;
450644d93782SGreg Clayton };
450744d93782SGreg Clayton 
450844d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = {true};
450944d93782SGreg Clayton 
4510b9c1b51eSKate Stone IOHandlerCursesGUI::IOHandlerCursesGUI(Debugger &debugger)
4511b9c1b51eSKate Stone     : IOHandler(debugger, IOHandler::Type::Curses) {}
451244d93782SGreg Clayton 
4513b9c1b51eSKate Stone void IOHandlerCursesGUI::Activate() {
451444d93782SGreg Clayton   IOHandler::Activate();
4515b9c1b51eSKate Stone   if (!m_app_ap) {
451644d93782SGreg Clayton     m_app_ap.reset(new Application(GetInputFILE(), GetOutputFILE()));
451744d93782SGreg Clayton 
451844d93782SGreg Clayton     // This is both a window and a menu delegate
4519b9c1b51eSKate Stone     std::shared_ptr<ApplicationDelegate> app_delegate_sp(
4520b9c1b51eSKate Stone         new ApplicationDelegate(*m_app_ap, m_debugger));
452144d93782SGreg Clayton 
4522b9c1b51eSKate Stone     MenuDelegateSP app_menu_delegate_sp =
4523b9c1b51eSKate Stone         std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
4524b9c1b51eSKate Stone     MenuSP lldb_menu_sp(
4525b9c1b51eSKate Stone         new Menu("LLDB", "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
4526b9c1b51eSKate Stone     MenuSP exit_menuitem_sp(
4527b9c1b51eSKate Stone         new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
452844d93782SGreg Clayton     exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
4529b9c1b51eSKate Stone     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(
4530b9c1b51eSKate Stone         "About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
453144d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
453244d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(exit_menuitem_sp);
453344d93782SGreg Clayton 
4534b9c1b51eSKate Stone     MenuSP target_menu_sp(new Menu("Target", "F2", KEY_F(2),
4535b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Target));
4536b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4537b9c1b51eSKate Stone         "Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
4538b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4539b9c1b51eSKate Stone         "Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
454044d93782SGreg Clayton 
4541b9c1b51eSKate Stone     MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3),
4542b9c1b51eSKate Stone                                     ApplicationDelegate::eMenuID_Process));
4543b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4544b9c1b51eSKate Stone         "Attach", nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
4545b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4546b9c1b51eSKate Stone         "Detach", nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
4547b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4548b9c1b51eSKate Stone         "Launch", nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
454944d93782SGreg Clayton     process_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
4550b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(
4551b9c1b51eSKate Stone         MenuSP(new Menu("Continue", nullptr, 'c',
4552b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ProcessContinue)));
4553b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4554b9c1b51eSKate Stone         "Halt", nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
4555b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4556b9c1b51eSKate Stone         "Kill", nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
455744d93782SGreg Clayton 
4558b9c1b51eSKate Stone     MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4),
4559b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Thread));
4560b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4561b9c1b51eSKate Stone         "Step In", nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
4562b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(
4563b9c1b51eSKate Stone         MenuSP(new Menu("Step Over", nullptr, 'v',
4564b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ThreadStepOver)));
4565b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4566b9c1b51eSKate Stone         "Step Out", nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
456744d93782SGreg Clayton 
4568b9c1b51eSKate Stone     MenuSP view_menu_sp(
4569b9c1b51eSKate Stone         new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
4570b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4571b9c1b51eSKate Stone         MenuSP(new Menu("Backtrace", nullptr, 'b',
4572b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewBacktrace)));
4573b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4574b9c1b51eSKate Stone         MenuSP(new Menu("Registers", nullptr, 'r',
4575b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewRegisters)));
4576b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(MenuSP(new Menu(
4577b9c1b51eSKate Stone         "Source", nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
4578b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4579b9c1b51eSKate Stone         MenuSP(new Menu("Variables", nullptr, 'v',
4580b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewVariables)));
458144d93782SGreg Clayton 
4582b9c1b51eSKate Stone     MenuSP help_menu_sp(
4583b9c1b51eSKate Stone         new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
4584b9c1b51eSKate Stone     help_menu_sp->AddSubmenu(MenuSP(new Menu(
4585b9c1b51eSKate Stone         "GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
458644d93782SGreg Clayton 
458744d93782SGreg Clayton     m_app_ap->Initialize();
458844d93782SGreg Clayton     WindowSP &main_window_sp = m_app_ap->GetMainWindow();
458944d93782SGreg Clayton 
459044d93782SGreg Clayton     MenuSP menubar_sp(new Menu(Menu::Type::Bar));
459144d93782SGreg Clayton     menubar_sp->AddSubmenu(lldb_menu_sp);
459244d93782SGreg Clayton     menubar_sp->AddSubmenu(target_menu_sp);
459344d93782SGreg Clayton     menubar_sp->AddSubmenu(process_menu_sp);
459444d93782SGreg Clayton     menubar_sp->AddSubmenu(thread_menu_sp);
459544d93782SGreg Clayton     menubar_sp->AddSubmenu(view_menu_sp);
459644d93782SGreg Clayton     menubar_sp->AddSubmenu(help_menu_sp);
459744d93782SGreg Clayton     menubar_sp->SetDelegate(app_menu_delegate_sp);
459844d93782SGreg Clayton 
459944d93782SGreg Clayton     Rect content_bounds = main_window_sp->GetFrame();
460044d93782SGreg Clayton     Rect menubar_bounds = content_bounds.MakeMenuBar();
460144d93782SGreg Clayton     Rect status_bounds = content_bounds.MakeStatusBar();
460244d93782SGreg Clayton     Rect source_bounds;
460344d93782SGreg Clayton     Rect variables_bounds;
460444d93782SGreg Clayton     Rect threads_bounds;
460544d93782SGreg Clayton     Rect source_variables_bounds;
4606b9c1b51eSKate Stone     content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds,
4607b9c1b51eSKate Stone                                            threads_bounds);
4608b9c1b51eSKate Stone     source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds,
4609b9c1b51eSKate Stone                                                       variables_bounds);
461044d93782SGreg Clayton 
4611b9c1b51eSKate Stone     WindowSP menubar_window_sp =
4612b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
461305097246SAdrian Prantl     // Let the menubar get keys if the active window doesn't handle the keys
461405097246SAdrian Prantl     // that are typed so it can respond to menubar key presses.
4615b9c1b51eSKate Stone     menubar_window_sp->SetCanBeActive(
4616b9c1b51eSKate Stone         false); // Don't let the menubar become the active window
461744d93782SGreg Clayton     menubar_window_sp->SetDelegate(menubar_sp);
461844d93782SGreg Clayton 
4619b9c1b51eSKate Stone     WindowSP source_window_sp(
4620b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Source", source_bounds, true));
4621b9c1b51eSKate Stone     WindowSP variables_window_sp(
4622b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Variables", variables_bounds, false));
4623b9c1b51eSKate Stone     WindowSP threads_window_sp(
4624b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Threads", threads_bounds, false));
4625b9c1b51eSKate Stone     WindowSP status_window_sp(
46266bb7e21fSPavel Labath         main_window_sp->CreateSubWindow("Status", status_bounds, false));
4627b9c1b51eSKate Stone     status_window_sp->SetCanBeActive(
4628b9c1b51eSKate Stone         false); // Don't let the status bar become the active window
4629b9c1b51eSKate Stone     main_window_sp->SetDelegate(
4630b9c1b51eSKate Stone         std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
4631b9c1b51eSKate Stone     source_window_sp->SetDelegate(
4632b9c1b51eSKate Stone         WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
4633b9c1b51eSKate Stone     variables_window_sp->SetDelegate(
4634b9c1b51eSKate Stone         WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
4635ec990867SGreg Clayton     TreeDelegateSP thread_delegate_sp(new ThreadsTreeDelegate(m_debugger));
4636b9c1b51eSKate Stone     threads_window_sp->SetDelegate(WindowDelegateSP(
4637b9c1b51eSKate Stone         new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
4638b9c1b51eSKate Stone     status_window_sp->SetDelegate(
4639b9c1b51eSKate Stone         WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
464044d93782SGreg Clayton 
46415fdb09bbSGreg Clayton     // Show the main help window once the first time the curses GUI is launched
46425fdb09bbSGreg Clayton     static bool g_showed_help = false;
4643b9c1b51eSKate Stone     if (!g_showed_help) {
46445fdb09bbSGreg Clayton       g_showed_help = true;
46455fdb09bbSGreg Clayton       main_window_sp->CreateHelpSubwindow();
46465fdb09bbSGreg Clayton     }
46475fdb09bbSGreg Clayton 
464844d93782SGreg Clayton     init_pair(1, COLOR_WHITE, COLOR_BLUE);
464944d93782SGreg Clayton     init_pair(2, COLOR_BLACK, COLOR_WHITE);
465044d93782SGreg Clayton     init_pair(3, COLOR_MAGENTA, COLOR_WHITE);
465144d93782SGreg Clayton     init_pair(4, COLOR_MAGENTA, COLOR_BLACK);
465244d93782SGreg Clayton     init_pair(5, COLOR_RED, COLOR_BLACK);
465344d93782SGreg Clayton   }
465444d93782SGreg Clayton }
465544d93782SGreg Clayton 
4656b9c1b51eSKate Stone void IOHandlerCursesGUI::Deactivate() { m_app_ap->Terminate(); }
465744d93782SGreg Clayton 
4658b9c1b51eSKate Stone void IOHandlerCursesGUI::Run() {
465944d93782SGreg Clayton   m_app_ap->Run(m_debugger);
466044d93782SGreg Clayton   SetIsDone(true);
466144d93782SGreg Clayton }
466244d93782SGreg Clayton 
4663315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
466444d93782SGreg Clayton 
4665b9c1b51eSKate Stone void IOHandlerCursesGUI::Cancel() {}
466644d93782SGreg Clayton 
4667b9c1b51eSKate Stone bool IOHandlerCursesGUI::Interrupt() { return false; }
466844d93782SGreg Clayton 
4669b9c1b51eSKate Stone void IOHandlerCursesGUI::GotEOF() {}
467044d93782SGreg Clayton 
4671315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
4672