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 1044d93782SGreg Clayton 1144d93782SGreg Clayton #include "lldb/lldb-python.h" 1244d93782SGreg Clayton 1344d93782SGreg Clayton #include <string> 1444d93782SGreg Clayton 1544d93782SGreg Clayton #include "lldb/Breakpoint/BreakpointLocation.h" 1644d93782SGreg Clayton #include "lldb/Core/IOHandler.h" 1744d93782SGreg Clayton #include "lldb/Core/Debugger.h" 18ec990867SGreg Clayton #include "lldb/Core/Module.h" 1944d93782SGreg Clayton #include "lldb/Core/State.h" 2044d93782SGreg Clayton #include "lldb/Core/StreamFile.h" 2144d93782SGreg Clayton #include "lldb/Core/ValueObjectRegister.h" 2244d93782SGreg Clayton #include "lldb/Host/Editline.h" 2344d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h" 2444d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h" 2544d93782SGreg Clayton #include "lldb/Symbol/Block.h" 2644d93782SGreg Clayton #include "lldb/Symbol/Function.h" 2744d93782SGreg Clayton #include "lldb/Symbol/Symbol.h" 2844d93782SGreg Clayton #include "lldb/Target/RegisterContext.h" 2944d93782SGreg Clayton #include "lldb/Target/ThreadPlan.h" 3044d93782SGreg Clayton 31914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES 3244d93782SGreg Clayton #include <ncurses.h> 3344d93782SGreg Clayton #include <panel.h> 34914b8d98SDeepak Panickal #endif 3544d93782SGreg Clayton 3644d93782SGreg Clayton using namespace lldb; 3744d93782SGreg Clayton using namespace lldb_private; 3844d93782SGreg Clayton 3944d93782SGreg Clayton IOHandler::IOHandler (Debugger &debugger) : 4044d93782SGreg Clayton IOHandler (debugger, 4144d93782SGreg Clayton StreamFileSP(), // Adopt STDIN from top input reader 4244d93782SGreg Clayton StreamFileSP(), // Adopt STDOUT from top input reader 43340b0309SGreg Clayton StreamFileSP(), // Adopt STDERR from top input reader 44340b0309SGreg Clayton 0) // Flags 4544d93782SGreg Clayton { 4644d93782SGreg Clayton } 4744d93782SGreg Clayton 4844d93782SGreg Clayton 4944d93782SGreg Clayton IOHandler::IOHandler (Debugger &debugger, 5044d93782SGreg Clayton const lldb::StreamFileSP &input_sp, 5144d93782SGreg Clayton const lldb::StreamFileSP &output_sp, 52340b0309SGreg Clayton const lldb::StreamFileSP &error_sp, 53340b0309SGreg Clayton uint32_t flags) : 5444d93782SGreg Clayton m_debugger (debugger), 5544d93782SGreg Clayton m_input_sp (input_sp), 5644d93782SGreg Clayton m_output_sp (output_sp), 5744d93782SGreg Clayton m_error_sp (error_sp), 58340b0309SGreg Clayton m_flags (flags), 5944d93782SGreg Clayton m_user_data (NULL), 6044d93782SGreg Clayton m_done (false), 6144d93782SGreg Clayton m_active (false) 6244d93782SGreg Clayton { 6344d93782SGreg Clayton // If any files are not specified, then adopt them from the top input reader. 6444d93782SGreg Clayton if (!m_input_sp || !m_output_sp || !m_error_sp) 6544d93782SGreg Clayton debugger.AdoptTopIOHandlerFilesIfInvalid (m_input_sp, 6644d93782SGreg Clayton m_output_sp, 6744d93782SGreg Clayton m_error_sp); 6844d93782SGreg Clayton } 6944d93782SGreg Clayton 7044d93782SGreg Clayton IOHandler::~IOHandler() 7144d93782SGreg Clayton { 7244d93782SGreg Clayton } 7344d93782SGreg Clayton 7444d93782SGreg Clayton 7544d93782SGreg Clayton int 7644d93782SGreg Clayton IOHandler::GetInputFD() 7744d93782SGreg Clayton { 7844d93782SGreg Clayton if (m_input_sp) 7944d93782SGreg Clayton return m_input_sp->GetFile().GetDescriptor(); 8044d93782SGreg Clayton return -1; 8144d93782SGreg Clayton } 8244d93782SGreg Clayton 8344d93782SGreg Clayton int 8444d93782SGreg Clayton IOHandler::GetOutputFD() 8544d93782SGreg Clayton { 8644d93782SGreg Clayton if (m_output_sp) 8744d93782SGreg Clayton return m_output_sp->GetFile().GetDescriptor(); 8844d93782SGreg Clayton return -1; 8944d93782SGreg Clayton } 9044d93782SGreg Clayton 9144d93782SGreg Clayton int 9244d93782SGreg Clayton IOHandler::GetErrorFD() 9344d93782SGreg Clayton { 9444d93782SGreg Clayton if (m_error_sp) 9544d93782SGreg Clayton return m_error_sp->GetFile().GetDescriptor(); 9644d93782SGreg Clayton return -1; 9744d93782SGreg Clayton } 9844d93782SGreg Clayton 9944d93782SGreg Clayton FILE * 10044d93782SGreg Clayton IOHandler::GetInputFILE() 10144d93782SGreg Clayton { 10244d93782SGreg Clayton if (m_input_sp) 10344d93782SGreg Clayton return m_input_sp->GetFile().GetStream(); 10444d93782SGreg Clayton return NULL; 10544d93782SGreg Clayton } 10644d93782SGreg Clayton 10744d93782SGreg Clayton FILE * 10844d93782SGreg Clayton IOHandler::GetOutputFILE() 10944d93782SGreg Clayton { 11044d93782SGreg Clayton if (m_output_sp) 11144d93782SGreg Clayton return m_output_sp->GetFile().GetStream(); 11244d93782SGreg Clayton return NULL; 11344d93782SGreg Clayton } 11444d93782SGreg Clayton 11544d93782SGreg Clayton FILE * 11644d93782SGreg Clayton IOHandler::GetErrorFILE() 11744d93782SGreg Clayton { 11844d93782SGreg Clayton if (m_error_sp) 11944d93782SGreg Clayton return m_error_sp->GetFile().GetStream(); 12044d93782SGreg Clayton return NULL; 12144d93782SGreg Clayton } 12244d93782SGreg Clayton 12344d93782SGreg Clayton StreamFileSP & 12444d93782SGreg Clayton IOHandler::GetInputStreamFile() 12544d93782SGreg Clayton { 12644d93782SGreg Clayton return m_input_sp; 12744d93782SGreg Clayton } 12844d93782SGreg Clayton 12944d93782SGreg Clayton StreamFileSP & 13044d93782SGreg Clayton IOHandler::GetOutputStreamFile() 13144d93782SGreg Clayton { 13244d93782SGreg Clayton return m_output_sp; 13344d93782SGreg Clayton } 13444d93782SGreg Clayton 13544d93782SGreg Clayton 13644d93782SGreg Clayton StreamFileSP & 13744d93782SGreg Clayton IOHandler::GetErrorStreamFile() 13844d93782SGreg Clayton { 13944d93782SGreg Clayton return m_error_sp; 14044d93782SGreg Clayton } 14144d93782SGreg Clayton 142340b0309SGreg Clayton bool 143340b0309SGreg Clayton IOHandler::GetIsInteractive () 144340b0309SGreg Clayton { 145340b0309SGreg Clayton return GetInputStreamFile()->GetFile().GetIsInteractive (); 146340b0309SGreg Clayton } 147340b0309SGreg Clayton 148340b0309SGreg Clayton bool 149340b0309SGreg Clayton IOHandler::GetIsRealTerminal () 150340b0309SGreg Clayton { 151340b0309SGreg Clayton return GetInputStreamFile()->GetFile().GetIsRealTerminal(); 152340b0309SGreg Clayton } 15344d93782SGreg Clayton 15444d93782SGreg Clayton IOHandlerConfirm::IOHandlerConfirm (Debugger &debugger, 15544d93782SGreg Clayton const char *prompt, 15644d93782SGreg Clayton bool default_response) : 15744d93782SGreg Clayton IOHandlerEditline(debugger, 15844d93782SGreg Clayton NULL, // NULL editline_name means no history loaded/saved 15944d93782SGreg Clayton NULL, 16044d93782SGreg Clayton false, // Multi-line 161f6913cd7SGreg Clayton 0, 16244d93782SGreg Clayton *this), 16344d93782SGreg Clayton m_default_response (default_response), 16444d93782SGreg Clayton m_user_response (default_response) 16544d93782SGreg Clayton { 16644d93782SGreg Clayton StreamString prompt_stream; 16744d93782SGreg Clayton prompt_stream.PutCString(prompt); 16844d93782SGreg Clayton if (m_default_response) 16944d93782SGreg Clayton prompt_stream.Printf(": [Y/n] "); 17044d93782SGreg Clayton else 17144d93782SGreg Clayton prompt_stream.Printf(": [y/N] "); 17244d93782SGreg Clayton 17344d93782SGreg Clayton SetPrompt (prompt_stream.GetString().c_str()); 17444d93782SGreg Clayton 17544d93782SGreg Clayton } 17644d93782SGreg Clayton 17744d93782SGreg Clayton 17844d93782SGreg Clayton IOHandlerConfirm::~IOHandlerConfirm () 17944d93782SGreg Clayton { 18044d93782SGreg Clayton } 18144d93782SGreg Clayton 18244d93782SGreg Clayton int 18344d93782SGreg Clayton IOHandlerConfirm::IOHandlerComplete (IOHandler &io_handler, 18444d93782SGreg Clayton const char *current_line, 18544d93782SGreg Clayton const char *cursor, 18644d93782SGreg Clayton const char *last_char, 18744d93782SGreg Clayton int skip_first_n_matches, 18844d93782SGreg Clayton int max_matches, 18944d93782SGreg Clayton StringList &matches) 19044d93782SGreg Clayton { 19144d93782SGreg Clayton if (current_line == cursor) 19244d93782SGreg Clayton { 19344d93782SGreg Clayton if (m_default_response) 19444d93782SGreg Clayton { 19544d93782SGreg Clayton matches.AppendString("y"); 19644d93782SGreg Clayton } 19744d93782SGreg Clayton else 19844d93782SGreg Clayton { 19944d93782SGreg Clayton matches.AppendString("n"); 20044d93782SGreg Clayton } 20144d93782SGreg Clayton } 20244d93782SGreg Clayton return matches.GetSize(); 20344d93782SGreg Clayton } 20444d93782SGreg Clayton 20544d93782SGreg Clayton void 20644d93782SGreg Clayton IOHandlerConfirm::IOHandlerInputComplete (IOHandler &io_handler, std::string &line) 20744d93782SGreg Clayton { 20844d93782SGreg Clayton if (line.empty()) 20944d93782SGreg Clayton { 21044d93782SGreg Clayton // User just hit enter, set the response to the default 21144d93782SGreg Clayton m_user_response = m_default_response; 21244d93782SGreg Clayton io_handler.SetIsDone(true); 21344d93782SGreg Clayton return; 21444d93782SGreg Clayton } 21544d93782SGreg Clayton 21644d93782SGreg Clayton if (line.size() == 1) 21744d93782SGreg Clayton { 21844d93782SGreg Clayton switch (line[0]) 21944d93782SGreg Clayton { 22044d93782SGreg Clayton case 'y': 22144d93782SGreg Clayton case 'Y': 22244d93782SGreg Clayton m_user_response = true; 22344d93782SGreg Clayton io_handler.SetIsDone(true); 22444d93782SGreg Clayton return; 22544d93782SGreg Clayton case 'n': 22644d93782SGreg Clayton case 'N': 22744d93782SGreg Clayton m_user_response = false; 22844d93782SGreg Clayton io_handler.SetIsDone(true); 22944d93782SGreg Clayton return; 23044d93782SGreg Clayton default: 23144d93782SGreg Clayton break; 23244d93782SGreg Clayton } 23344d93782SGreg Clayton } 23444d93782SGreg Clayton 23544d93782SGreg Clayton if (line == "yes" || line == "YES" || line == "Yes") 23644d93782SGreg Clayton { 23744d93782SGreg Clayton m_user_response = true; 23844d93782SGreg Clayton io_handler.SetIsDone(true); 23944d93782SGreg Clayton } 24044d93782SGreg Clayton else if (line == "no" || line == "NO" || line == "No") 24144d93782SGreg Clayton { 24244d93782SGreg Clayton m_user_response = false; 24344d93782SGreg Clayton io_handler.SetIsDone(true); 24444d93782SGreg Clayton } 24544d93782SGreg Clayton } 24644d93782SGreg Clayton 24744d93782SGreg Clayton int 24844d93782SGreg Clayton IOHandlerDelegate::IOHandlerComplete (IOHandler &io_handler, 24944d93782SGreg Clayton const char *current_line, 25044d93782SGreg Clayton const char *cursor, 25144d93782SGreg Clayton const char *last_char, 25244d93782SGreg Clayton int skip_first_n_matches, 25344d93782SGreg Clayton int max_matches, 25444d93782SGreg Clayton StringList &matches) 25544d93782SGreg Clayton { 25644d93782SGreg Clayton switch (m_completion) 25744d93782SGreg Clayton { 25844d93782SGreg Clayton case Completion::None: 25944d93782SGreg Clayton break; 26044d93782SGreg Clayton 26144d93782SGreg Clayton case Completion::LLDBCommand: 26244d93782SGreg Clayton return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion (current_line, 26344d93782SGreg Clayton cursor, 26444d93782SGreg Clayton last_char, 26544d93782SGreg Clayton skip_first_n_matches, 26644d93782SGreg Clayton max_matches, 26744d93782SGreg Clayton matches); 26844d93782SGreg Clayton 26944d93782SGreg Clayton case Completion::Expression: 27044d93782SGreg Clayton { 27144d93782SGreg Clayton bool word_complete = false; 27244d93782SGreg Clayton const char *word_start = cursor; 27344d93782SGreg Clayton if (cursor > current_line) 27444d93782SGreg Clayton --word_start; 27544d93782SGreg Clayton while (word_start > current_line && !isspace(*word_start)) 27644d93782SGreg Clayton --word_start; 27744d93782SGreg Clayton CommandCompletions::InvokeCommonCompletionCallbacks (io_handler.GetDebugger().GetCommandInterpreter(), 27844d93782SGreg Clayton CommandCompletions::eVariablePathCompletion, 27944d93782SGreg Clayton word_start, 28044d93782SGreg Clayton skip_first_n_matches, 28144d93782SGreg Clayton max_matches, 28244d93782SGreg Clayton NULL, 28344d93782SGreg Clayton word_complete, 28444d93782SGreg Clayton matches); 28544d93782SGreg Clayton 28644d93782SGreg Clayton size_t num_matches = matches.GetSize(); 28744d93782SGreg Clayton if (num_matches > 0) 28844d93782SGreg Clayton { 28944d93782SGreg Clayton std::string common_prefix; 29044d93782SGreg Clayton matches.LongestCommonPrefix (common_prefix); 29144d93782SGreg Clayton const size_t partial_name_len = strlen(word_start); 29244d93782SGreg Clayton 29344d93782SGreg Clayton // If we matched a unique single command, add a space... 29444d93782SGreg Clayton // Only do this if the completer told us this was a complete word, however... 29544d93782SGreg Clayton if (num_matches == 1 && word_complete) 29644d93782SGreg Clayton { 29744d93782SGreg Clayton common_prefix.push_back(' '); 29844d93782SGreg Clayton } 29944d93782SGreg Clayton common_prefix.erase (0, partial_name_len); 30044d93782SGreg Clayton matches.InsertStringAtIndex(0, std::move(common_prefix)); 30144d93782SGreg Clayton } 30244d93782SGreg Clayton return num_matches; 30344d93782SGreg Clayton } 30444d93782SGreg Clayton break; 30544d93782SGreg Clayton } 30644d93782SGreg Clayton 30744d93782SGreg Clayton 30844d93782SGreg Clayton return 0; 30944d93782SGreg Clayton } 31044d93782SGreg Clayton 31144d93782SGreg Clayton 31244d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger, 31344d93782SGreg Clayton const char *editline_name, // Used for saving history files 31444d93782SGreg Clayton const char *prompt, 31544d93782SGreg Clayton bool multi_line, 316f6913cd7SGreg Clayton uint32_t line_number_start, 31744d93782SGreg Clayton IOHandlerDelegate &delegate) : 31844d93782SGreg Clayton IOHandlerEditline(debugger, 31944d93782SGreg Clayton StreamFileSP(), // Inherit input from top input reader 32044d93782SGreg Clayton StreamFileSP(), // Inherit output from top input reader 32144d93782SGreg Clayton StreamFileSP(), // Inherit error from top input reader 322340b0309SGreg Clayton 0, // Flags 32344d93782SGreg Clayton editline_name, // Used for saving history files 32444d93782SGreg Clayton prompt, 32544d93782SGreg Clayton multi_line, 326f6913cd7SGreg Clayton line_number_start, 32744d93782SGreg Clayton delegate) 32844d93782SGreg Clayton { 32944d93782SGreg Clayton } 33044d93782SGreg Clayton 33144d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger, 33244d93782SGreg Clayton const lldb::StreamFileSP &input_sp, 33344d93782SGreg Clayton const lldb::StreamFileSP &output_sp, 33444d93782SGreg Clayton const lldb::StreamFileSP &error_sp, 335340b0309SGreg Clayton uint32_t flags, 33644d93782SGreg Clayton const char *editline_name, // Used for saving history files 33744d93782SGreg Clayton const char *prompt, 33844d93782SGreg Clayton bool multi_line, 339f6913cd7SGreg Clayton uint32_t line_number_start, 34044d93782SGreg Clayton IOHandlerDelegate &delegate) : 341340b0309SGreg Clayton IOHandler (debugger, input_sp, output_sp, error_sp, flags), 34244d93782SGreg Clayton m_editline_ap (), 34344d93782SGreg Clayton m_delegate (delegate), 34444d93782SGreg Clayton m_prompt (), 345f6913cd7SGreg Clayton m_base_line_number (line_number_start), 346340b0309SGreg Clayton m_multi_line (multi_line) 34744d93782SGreg Clayton { 34844d93782SGreg Clayton SetPrompt(prompt); 34944d93782SGreg Clayton 350914b8d98SDeepak Panickal bool use_editline = false; 351340b0309SGreg Clayton 352914b8d98SDeepak Panickal #ifndef _MSC_VER 353340b0309SGreg Clayton use_editline = m_input_sp->GetFile().GetIsRealTerminal(); 354914b8d98SDeepak Panickal #else 355914b8d98SDeepak Panickal use_editline = true; 356914b8d98SDeepak Panickal #endif 35744d93782SGreg Clayton 35844d93782SGreg Clayton if (use_editline) 35944d93782SGreg Clayton { 36044d93782SGreg Clayton m_editline_ap.reset(new Editline (editline_name, 36144d93782SGreg Clayton prompt ? prompt : "", 36244d93782SGreg Clayton GetInputFILE (), 36344d93782SGreg Clayton GetOutputFILE (), 36444d93782SGreg Clayton GetErrorFILE ())); 365f6913cd7SGreg Clayton if (m_base_line_number > 0) 366f6913cd7SGreg Clayton m_editline_ap->ShowLineNumbers(true, m_base_line_number); 36744d93782SGreg Clayton m_editline_ap->SetLineCompleteCallback (LineCompletedCallback, this); 36844d93782SGreg Clayton m_editline_ap->SetAutoCompleteCallback (AutoCompleteCallback, this); 36944d93782SGreg Clayton } 37044d93782SGreg Clayton 37144d93782SGreg Clayton } 37244d93782SGreg Clayton 37344d93782SGreg Clayton IOHandlerEditline::~IOHandlerEditline () 37444d93782SGreg Clayton { 37544d93782SGreg Clayton m_editline_ap.reset(); 37644d93782SGreg Clayton } 37744d93782SGreg Clayton 37844d93782SGreg Clayton 37944d93782SGreg Clayton bool 38044d93782SGreg Clayton IOHandlerEditline::GetLine (std::string &line) 38144d93782SGreg Clayton { 38244d93782SGreg Clayton if (m_editline_ap) 38344d93782SGreg Clayton { 38444d93782SGreg Clayton return m_editline_ap->GetLine(line).Success(); 38544d93782SGreg Clayton } 38644d93782SGreg Clayton else 38744d93782SGreg Clayton { 38844d93782SGreg Clayton line.clear(); 38944d93782SGreg Clayton 39044d93782SGreg Clayton FILE *in = GetInputFILE(); 39144d93782SGreg Clayton if (in) 39244d93782SGreg Clayton { 393340b0309SGreg Clayton if (GetIsInteractive()) 39444d93782SGreg Clayton { 39544d93782SGreg Clayton const char *prompt = GetPrompt(); 39644d93782SGreg Clayton if (prompt && prompt[0]) 39744d93782SGreg Clayton { 39844d93782SGreg Clayton FILE *out = GetOutputFILE(); 39944d93782SGreg Clayton if (out) 40044d93782SGreg Clayton { 40144d93782SGreg Clayton ::fprintf(out, "%s", prompt); 40244d93782SGreg Clayton ::fflush(out); 40344d93782SGreg Clayton } 40444d93782SGreg Clayton } 40544d93782SGreg Clayton } 40644d93782SGreg Clayton char buffer[256]; 40744d93782SGreg Clayton bool done = false; 4080f86e6e7SGreg Clayton bool got_line = false; 40944d93782SGreg Clayton while (!done) 41044d93782SGreg Clayton { 41144d93782SGreg Clayton if (fgets(buffer, sizeof(buffer), in) == NULL) 412c9cf5798SGreg Clayton { 413c7797accSGreg Clayton const int saved_errno = errno; 414c9cf5798SGreg Clayton if (feof(in)) 41544d93782SGreg Clayton done = true; 416c7797accSGreg Clayton else if (ferror(in)) 417c7797accSGreg Clayton { 418c7797accSGreg Clayton if (saved_errno != EINTR) 419c7797accSGreg Clayton done = true; 420c7797accSGreg Clayton } 421c9cf5798SGreg Clayton } 42244d93782SGreg Clayton else 42344d93782SGreg Clayton { 4240f86e6e7SGreg Clayton got_line = true; 42544d93782SGreg Clayton size_t buffer_len = strlen(buffer); 42644d93782SGreg Clayton assert (buffer[buffer_len] == '\0'); 42744d93782SGreg Clayton char last_char = buffer[buffer_len-1]; 42844d93782SGreg Clayton if (last_char == '\r' || last_char == '\n') 42944d93782SGreg Clayton { 43044d93782SGreg Clayton done = true; 43144d93782SGreg Clayton // Strip trailing newlines 43244d93782SGreg Clayton while (last_char == '\r' || last_char == '\n') 43344d93782SGreg Clayton { 43444d93782SGreg Clayton --buffer_len; 43544d93782SGreg Clayton if (buffer_len == 0) 43644d93782SGreg Clayton break; 43744d93782SGreg Clayton last_char = buffer[buffer_len-1]; 43844d93782SGreg Clayton } 43944d93782SGreg Clayton } 44044d93782SGreg Clayton line.append(buffer, buffer_len); 44144d93782SGreg Clayton } 44244d93782SGreg Clayton } 4430f86e6e7SGreg Clayton // We might have gotten a newline on a line by itself 4440f86e6e7SGreg Clayton // make sure to return true in this case. 4450f86e6e7SGreg Clayton return got_line; 44644d93782SGreg Clayton } 44744d93782SGreg Clayton else 44844d93782SGreg Clayton { 44944d93782SGreg Clayton // No more input file, we are done... 45044d93782SGreg Clayton SetIsDone(true); 45144d93782SGreg Clayton } 452340b0309SGreg Clayton return false; 45344d93782SGreg Clayton } 45444d93782SGreg Clayton } 45544d93782SGreg Clayton 45644d93782SGreg Clayton 45744d93782SGreg Clayton LineStatus 45844d93782SGreg Clayton IOHandlerEditline::LineCompletedCallback (Editline *editline, 45944d93782SGreg Clayton StringList &lines, 46044d93782SGreg Clayton uint32_t line_idx, 46144d93782SGreg Clayton Error &error, 46244d93782SGreg Clayton void *baton) 46344d93782SGreg Clayton { 46444d93782SGreg Clayton IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton; 46544d93782SGreg Clayton return editline_reader->m_delegate.IOHandlerLinesUpdated(*editline_reader, lines, line_idx, error); 46644d93782SGreg Clayton } 46744d93782SGreg Clayton 46844d93782SGreg Clayton int 46944d93782SGreg Clayton IOHandlerEditline::AutoCompleteCallback (const char *current_line, 47044d93782SGreg Clayton const char *cursor, 47144d93782SGreg Clayton const char *last_char, 47244d93782SGreg Clayton int skip_first_n_matches, 47344d93782SGreg Clayton int max_matches, 47444d93782SGreg Clayton StringList &matches, 47544d93782SGreg Clayton void *baton) 47644d93782SGreg Clayton { 47744d93782SGreg Clayton IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton; 47844d93782SGreg Clayton if (editline_reader) 47944d93782SGreg Clayton return editline_reader->m_delegate.IOHandlerComplete (*editline_reader, 48044d93782SGreg Clayton current_line, 48144d93782SGreg Clayton cursor, 48244d93782SGreg Clayton last_char, 48344d93782SGreg Clayton skip_first_n_matches, 48444d93782SGreg Clayton max_matches, 48544d93782SGreg Clayton matches); 48644d93782SGreg Clayton return 0; 48744d93782SGreg Clayton } 48844d93782SGreg Clayton 48944d93782SGreg Clayton const char * 49044d93782SGreg Clayton IOHandlerEditline::GetPrompt () 49144d93782SGreg Clayton { 49244d93782SGreg Clayton if (m_editline_ap) 49344d93782SGreg Clayton return m_editline_ap->GetPrompt (); 49444d93782SGreg Clayton else if (m_prompt.empty()) 49544d93782SGreg Clayton return NULL; 49644d93782SGreg Clayton return m_prompt.c_str(); 49744d93782SGreg Clayton } 49844d93782SGreg Clayton 49944d93782SGreg Clayton bool 50044d93782SGreg Clayton IOHandlerEditline::SetPrompt (const char *p) 50144d93782SGreg Clayton { 50244d93782SGreg Clayton if (p && p[0]) 50344d93782SGreg Clayton m_prompt = p; 50444d93782SGreg Clayton else 50544d93782SGreg Clayton m_prompt.clear(); 50644d93782SGreg Clayton if (m_editline_ap) 50744d93782SGreg Clayton m_editline_ap->SetPrompt (m_prompt.empty() ? NULL : m_prompt.c_str()); 50844d93782SGreg Clayton return true; 50944d93782SGreg Clayton } 51044d93782SGreg Clayton 511f6913cd7SGreg Clayton void 512f6913cd7SGreg Clayton IOHandlerEditline::SetBaseLineNumber (uint32_t line) 513f6913cd7SGreg Clayton { 514f6913cd7SGreg Clayton m_base_line_number = line; 515f6913cd7SGreg Clayton if (m_editline_ap) 516f6913cd7SGreg Clayton m_editline_ap->ShowLineNumbers (true, line); 517f6913cd7SGreg Clayton 518f6913cd7SGreg Clayton } 51944d93782SGreg Clayton bool 52044d93782SGreg Clayton IOHandlerEditline::GetLines (StringList &lines) 52144d93782SGreg Clayton { 52244d93782SGreg Clayton bool success = false; 52344d93782SGreg Clayton if (m_editline_ap) 52444d93782SGreg Clayton { 52544d93782SGreg Clayton std::string end_token; 52644d93782SGreg Clayton success = m_editline_ap->GetLines(end_token, lines).Success(); 52744d93782SGreg Clayton } 52844d93782SGreg Clayton else 52944d93782SGreg Clayton { 53044d93782SGreg Clayton LineStatus lines_status = LineStatus::Success; 53144d93782SGreg Clayton 53244d93782SGreg Clayton while (lines_status == LineStatus::Success) 53344d93782SGreg Clayton { 534f6913cd7SGreg Clayton // Show line numbers if we are asked to 53544d93782SGreg Clayton std::string line; 536f6913cd7SGreg Clayton if (m_base_line_number > 0 && GetIsInteractive()) 537f6913cd7SGreg Clayton { 538f6913cd7SGreg Clayton FILE *out = GetOutputFILE(); 539f6913cd7SGreg Clayton if (out) 540f6913cd7SGreg Clayton ::fprintf(out, "%u", m_base_line_number + (uint32_t)lines.GetSize()); 541f6913cd7SGreg Clayton } 542f6913cd7SGreg Clayton 54344d93782SGreg Clayton if (GetLine(line)) 54444d93782SGreg Clayton { 54544d93782SGreg Clayton lines.AppendString(line); 54644d93782SGreg Clayton Error error; 54744d93782SGreg Clayton lines_status = m_delegate.IOHandlerLinesUpdated(*this, lines, lines.GetSize() - 1, error); 54844d93782SGreg Clayton } 54944d93782SGreg Clayton else 55044d93782SGreg Clayton { 55144d93782SGreg Clayton lines_status = LineStatus::Done; 55244d93782SGreg Clayton } 55344d93782SGreg Clayton } 55444d93782SGreg Clayton success = lines.GetSize() > 0; 55544d93782SGreg Clayton } 55644d93782SGreg Clayton return success; 55744d93782SGreg Clayton } 55844d93782SGreg Clayton 55944d93782SGreg Clayton // Each IOHandler gets to run until it is done. It should read data 56044d93782SGreg Clayton // from the "in" and place output into "out" and "err and return 56144d93782SGreg Clayton // when done. 56244d93782SGreg Clayton void 56344d93782SGreg Clayton IOHandlerEditline::Run () 56444d93782SGreg Clayton { 56544d93782SGreg Clayton std::string line; 56644d93782SGreg Clayton while (IsActive()) 56744d93782SGreg Clayton { 56844d93782SGreg Clayton if (m_multi_line) 56944d93782SGreg Clayton { 57044d93782SGreg Clayton StringList lines; 57144d93782SGreg Clayton if (GetLines (lines)) 57244d93782SGreg Clayton { 57344d93782SGreg Clayton line = lines.CopyList(); 57444d93782SGreg Clayton m_delegate.IOHandlerInputComplete(*this, line); 57544d93782SGreg Clayton } 57644d93782SGreg Clayton else 57744d93782SGreg Clayton { 57844d93782SGreg Clayton m_done = true; 57944d93782SGreg Clayton } 58044d93782SGreg Clayton } 58144d93782SGreg Clayton else 58244d93782SGreg Clayton { 58344d93782SGreg Clayton if (GetLine(line)) 58444d93782SGreg Clayton { 58544d93782SGreg Clayton m_delegate.IOHandlerInputComplete(*this, line); 58644d93782SGreg Clayton } 58744d93782SGreg Clayton else 58844d93782SGreg Clayton { 58944d93782SGreg Clayton m_done = true; 59044d93782SGreg Clayton } 59144d93782SGreg Clayton } 59244d93782SGreg Clayton } 59344d93782SGreg Clayton } 59444d93782SGreg Clayton 59544d93782SGreg Clayton void 59644d93782SGreg Clayton IOHandlerEditline::Hide () 59744d93782SGreg Clayton { 59844d93782SGreg Clayton if (m_editline_ap && m_editline_ap->GettingLine()) 59944d93782SGreg Clayton m_editline_ap->Hide(); 60044d93782SGreg Clayton } 60144d93782SGreg Clayton 60244d93782SGreg Clayton 60344d93782SGreg Clayton void 60444d93782SGreg Clayton IOHandlerEditline::Refresh () 60544d93782SGreg Clayton { 60644d93782SGreg Clayton if (m_editline_ap && m_editline_ap->GettingLine()) 60744d93782SGreg Clayton m_editline_ap->Refresh(); 60844d93782SGreg Clayton else 60944d93782SGreg Clayton { 61044d93782SGreg Clayton const char *prompt = GetPrompt(); 61144d93782SGreg Clayton if (prompt && prompt[0]) 61244d93782SGreg Clayton { 61344d93782SGreg Clayton FILE *out = GetOutputFILE(); 61444d93782SGreg Clayton if (out) 61544d93782SGreg Clayton { 61644d93782SGreg Clayton ::fprintf(out, "%s", prompt); 61744d93782SGreg Clayton ::fflush(out); 61844d93782SGreg Clayton } 61944d93782SGreg Clayton } 62044d93782SGreg Clayton } 62144d93782SGreg Clayton } 62244d93782SGreg Clayton 62344d93782SGreg Clayton void 624e68f5d6bSGreg Clayton IOHandlerEditline::Cancel () 625e68f5d6bSGreg Clayton { 626e68f5d6bSGreg Clayton if (m_editline_ap) 627e68f5d6bSGreg Clayton m_editline_ap->Interrupt (); 628e68f5d6bSGreg Clayton } 629e68f5d6bSGreg Clayton 630e68f5d6bSGreg Clayton void 63144d93782SGreg Clayton IOHandlerEditline::Interrupt () 63244d93782SGreg Clayton { 63344d93782SGreg Clayton if (m_editline_ap) 63444d93782SGreg Clayton m_editline_ap->Interrupt(); 63544d93782SGreg Clayton } 63644d93782SGreg Clayton 63744d93782SGreg Clayton void 63844d93782SGreg Clayton IOHandlerEditline::GotEOF() 63944d93782SGreg Clayton { 64044d93782SGreg Clayton if (m_editline_ap) 64144d93782SGreg Clayton m_editline_ap->Interrupt(); 64244d93782SGreg Clayton } 64344d93782SGreg Clayton 644914b8d98SDeepak Panickal // we may want curses to be disabled for some builds 645914b8d98SDeepak Panickal // for instance, windows 646914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES 647914b8d98SDeepak Panickal 64844d93782SGreg Clayton #include "lldb/Core/ValueObject.h" 64944d93782SGreg Clayton #include "lldb/Symbol/VariableList.h" 65044d93782SGreg Clayton #include "lldb/Target/Target.h" 65144d93782SGreg Clayton #include "lldb/Target/Process.h" 65244d93782SGreg Clayton #include "lldb/Target/Thread.h" 65344d93782SGreg Clayton #include "lldb/Target/StackFrame.h" 65444d93782SGreg Clayton 65544d93782SGreg Clayton #define KEY_RETURN 10 65644d93782SGreg Clayton #define KEY_ESCAPE 27 65744d93782SGreg Clayton 65844d93782SGreg Clayton namespace curses 65944d93782SGreg Clayton { 66044d93782SGreg Clayton class Menu; 66144d93782SGreg Clayton class MenuDelegate; 66244d93782SGreg Clayton class Window; 66344d93782SGreg Clayton class WindowDelegate; 66444d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP; 66544d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP; 66644d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP; 66744d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP; 66844d93782SGreg Clayton typedef std::vector<MenuSP> Menus; 66944d93782SGreg Clayton typedef std::vector<WindowSP> Windows; 67044d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates; 67144d93782SGreg Clayton 67244d93782SGreg Clayton #if 0 67344d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point 67444d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size 67544d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect 67644d93782SGreg Clayton #endif 67744d93782SGreg Clayton struct Point 67844d93782SGreg Clayton { 67944d93782SGreg Clayton int x; 68044d93782SGreg Clayton int y; 68144d93782SGreg Clayton 68244d93782SGreg Clayton Point (int _x = 0, int _y = 0) : 68344d93782SGreg Clayton x(_x), 68444d93782SGreg Clayton y(_y) 68544d93782SGreg Clayton { 68644d93782SGreg Clayton } 68744d93782SGreg Clayton 68844d93782SGreg Clayton void 68944d93782SGreg Clayton Clear () 69044d93782SGreg Clayton { 69144d93782SGreg Clayton x = 0; 69244d93782SGreg Clayton y = 0; 69344d93782SGreg Clayton } 69444d93782SGreg Clayton 69544d93782SGreg Clayton Point & 69644d93782SGreg Clayton operator += (const Point &rhs) 69744d93782SGreg Clayton { 69844d93782SGreg Clayton x += rhs.x; 69944d93782SGreg Clayton y += rhs.y; 70044d93782SGreg Clayton return *this; 70144d93782SGreg Clayton } 70244d93782SGreg Clayton 70344d93782SGreg Clayton void 70444d93782SGreg Clayton Dump () 70544d93782SGreg Clayton { 70644d93782SGreg Clayton printf ("(x=%i, y=%i)\n", x, y); 70744d93782SGreg Clayton } 70844d93782SGreg Clayton 70944d93782SGreg Clayton }; 71044d93782SGreg Clayton 71144d93782SGreg Clayton bool operator == (const Point &lhs, const Point &rhs) 71244d93782SGreg Clayton { 71344d93782SGreg Clayton return lhs.x == rhs.x && lhs.y == rhs.y; 71444d93782SGreg Clayton } 71544d93782SGreg Clayton bool operator != (const Point &lhs, const Point &rhs) 71644d93782SGreg Clayton { 71744d93782SGreg Clayton return lhs.x != rhs.x || lhs.y != rhs.y; 71844d93782SGreg Clayton } 71944d93782SGreg Clayton 72044d93782SGreg Clayton struct Size 72144d93782SGreg Clayton { 72244d93782SGreg Clayton int width; 72344d93782SGreg Clayton int height; 72444d93782SGreg Clayton Size (int w = 0, int h = 0) : 72544d93782SGreg Clayton width (w), 72644d93782SGreg Clayton height (h) 72744d93782SGreg Clayton { 72844d93782SGreg Clayton } 72944d93782SGreg Clayton 73044d93782SGreg Clayton void 73144d93782SGreg Clayton Clear () 73244d93782SGreg Clayton { 73344d93782SGreg Clayton width = 0; 73444d93782SGreg Clayton height = 0; 73544d93782SGreg Clayton } 73644d93782SGreg Clayton 73744d93782SGreg Clayton void 73844d93782SGreg Clayton Dump () 73944d93782SGreg Clayton { 74044d93782SGreg Clayton printf ("(w=%i, h=%i)\n", width, height); 74144d93782SGreg Clayton } 74244d93782SGreg Clayton 74344d93782SGreg Clayton }; 74444d93782SGreg Clayton 74544d93782SGreg Clayton bool operator == (const Size &lhs, const Size &rhs) 74644d93782SGreg Clayton { 74744d93782SGreg Clayton return lhs.width == rhs.width && lhs.height == rhs.height; 74844d93782SGreg Clayton } 74944d93782SGreg Clayton bool operator != (const Size &lhs, const Size &rhs) 75044d93782SGreg Clayton { 75144d93782SGreg Clayton return lhs.width != rhs.width || lhs.height != rhs.height; 75244d93782SGreg Clayton } 75344d93782SGreg Clayton 75444d93782SGreg Clayton struct Rect 75544d93782SGreg Clayton { 75644d93782SGreg Clayton Point origin; 75744d93782SGreg Clayton Size size; 75844d93782SGreg Clayton 75944d93782SGreg Clayton Rect () : 76044d93782SGreg Clayton origin(), 76144d93782SGreg Clayton size() 76244d93782SGreg Clayton { 76344d93782SGreg Clayton } 76444d93782SGreg Clayton 76544d93782SGreg Clayton Rect (const Point &p, const Size &s) : 76644d93782SGreg Clayton origin (p), 76744d93782SGreg Clayton size (s) 76844d93782SGreg Clayton { 76944d93782SGreg Clayton } 77044d93782SGreg Clayton 77144d93782SGreg Clayton void 77244d93782SGreg Clayton Clear () 77344d93782SGreg Clayton { 77444d93782SGreg Clayton origin.Clear(); 77544d93782SGreg Clayton size.Clear(); 77644d93782SGreg Clayton } 77744d93782SGreg Clayton 77844d93782SGreg Clayton void 77944d93782SGreg Clayton Dump () 78044d93782SGreg Clayton { 78144d93782SGreg Clayton printf ("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width, size.height); 78244d93782SGreg Clayton } 78344d93782SGreg Clayton 78444d93782SGreg Clayton void 78544d93782SGreg Clayton Inset (int w, int h) 78644d93782SGreg Clayton { 78744d93782SGreg Clayton if (size.width > w*2) 78844d93782SGreg Clayton size.width -= w*2; 78944d93782SGreg Clayton origin.x += w; 79044d93782SGreg Clayton 79144d93782SGreg Clayton if (size.height > h*2) 79244d93782SGreg Clayton size.height -= h*2; 79344d93782SGreg Clayton origin.y += h; 79444d93782SGreg Clayton } 79544d93782SGreg Clayton // Return a status bar rectangle which is the last line of 79644d93782SGreg Clayton // this rectangle. This rectangle will be modified to not 79744d93782SGreg Clayton // include the status bar area. 79844d93782SGreg Clayton Rect 79944d93782SGreg Clayton MakeStatusBar () 80044d93782SGreg Clayton { 80144d93782SGreg Clayton Rect status_bar; 80244d93782SGreg Clayton if (size.height > 1) 80344d93782SGreg Clayton { 80444d93782SGreg Clayton status_bar.origin.x = origin.x; 80544d93782SGreg Clayton status_bar.origin.y = size.height; 80644d93782SGreg Clayton status_bar.size.width = size.width; 80744d93782SGreg Clayton status_bar.size.height = 1; 80844d93782SGreg Clayton --size.height; 80944d93782SGreg Clayton } 81044d93782SGreg Clayton return status_bar; 81144d93782SGreg Clayton } 81244d93782SGreg Clayton 81344d93782SGreg Clayton // Return a menubar rectangle which is the first line of 81444d93782SGreg Clayton // this rectangle. This rectangle will be modified to not 81544d93782SGreg Clayton // include the menubar area. 81644d93782SGreg Clayton Rect 81744d93782SGreg Clayton MakeMenuBar () 81844d93782SGreg Clayton { 81944d93782SGreg Clayton Rect menubar; 82044d93782SGreg Clayton if (size.height > 1) 82144d93782SGreg Clayton { 82244d93782SGreg Clayton menubar.origin.x = origin.x; 82344d93782SGreg Clayton menubar.origin.y = origin.y; 82444d93782SGreg Clayton menubar.size.width = size.width; 82544d93782SGreg Clayton menubar.size.height = 1; 82644d93782SGreg Clayton ++origin.y; 82744d93782SGreg Clayton --size.height; 82844d93782SGreg Clayton } 82944d93782SGreg Clayton return menubar; 83044d93782SGreg Clayton } 83144d93782SGreg Clayton 83244d93782SGreg Clayton void 83344d93782SGreg Clayton HorizontalSplitPercentage (float top_percentage, Rect &top, Rect &bottom) const 83444d93782SGreg Clayton { 83544d93782SGreg Clayton float top_height = top_percentage * size.height; 83644d93782SGreg Clayton HorizontalSplit (top_height, top, bottom); 83744d93782SGreg Clayton } 83844d93782SGreg Clayton 83944d93782SGreg Clayton void 84044d93782SGreg Clayton HorizontalSplit (int top_height, Rect &top, Rect &bottom) const 84144d93782SGreg Clayton { 84244d93782SGreg Clayton top = *this; 84344d93782SGreg Clayton if (top_height < size.height) 84444d93782SGreg Clayton { 84544d93782SGreg Clayton top.size.height = top_height; 84644d93782SGreg Clayton bottom.origin.x = origin.x; 84744d93782SGreg Clayton bottom.origin.y = origin.y + top.size.height; 84844d93782SGreg Clayton bottom.size.width = size.width; 84944d93782SGreg Clayton bottom.size.height = size.height - top.size.height; 85044d93782SGreg Clayton } 85144d93782SGreg Clayton else 85244d93782SGreg Clayton { 85344d93782SGreg Clayton bottom.Clear(); 85444d93782SGreg Clayton } 85544d93782SGreg Clayton } 85644d93782SGreg Clayton 85744d93782SGreg Clayton void 85844d93782SGreg Clayton VerticalSplitPercentage (float left_percentage, Rect &left, Rect &right) const 85944d93782SGreg Clayton { 86044d93782SGreg Clayton float left_width = left_percentage * size.width; 86144d93782SGreg Clayton VerticalSplit (left_width, left, right); 86244d93782SGreg Clayton } 86344d93782SGreg Clayton 86444d93782SGreg Clayton 86544d93782SGreg Clayton void 86644d93782SGreg Clayton VerticalSplit (int left_width, Rect &left, Rect &right) const 86744d93782SGreg Clayton { 86844d93782SGreg Clayton left = *this; 86944d93782SGreg Clayton if (left_width < size.width) 87044d93782SGreg Clayton { 87144d93782SGreg Clayton left.size.width = left_width; 87244d93782SGreg Clayton right.origin.x = origin.x + left.size.width; 87344d93782SGreg Clayton right.origin.y = origin.y; 87444d93782SGreg Clayton right.size.width = size.width - left.size.width; 87544d93782SGreg Clayton right.size.height = size.height; 87644d93782SGreg Clayton } 87744d93782SGreg Clayton else 87844d93782SGreg Clayton { 87944d93782SGreg Clayton right.Clear(); 88044d93782SGreg Clayton } 88144d93782SGreg Clayton } 88244d93782SGreg Clayton }; 88344d93782SGreg Clayton 88444d93782SGreg Clayton bool operator == (const Rect &lhs, const Rect &rhs) 88544d93782SGreg Clayton { 88644d93782SGreg Clayton return lhs.origin == rhs.origin && lhs.size == rhs.size; 88744d93782SGreg Clayton } 88844d93782SGreg Clayton bool operator != (const Rect &lhs, const Rect &rhs) 88944d93782SGreg Clayton { 89044d93782SGreg Clayton return lhs.origin != rhs.origin || lhs.size != rhs.size; 89144d93782SGreg Clayton } 89244d93782SGreg Clayton 89344d93782SGreg Clayton enum HandleCharResult 89444d93782SGreg Clayton { 89544d93782SGreg Clayton eKeyNotHandled = 0, 89644d93782SGreg Clayton eKeyHandled = 1, 89744d93782SGreg Clayton eQuitApplication = 2 89844d93782SGreg Clayton }; 89944d93782SGreg Clayton 90044d93782SGreg Clayton enum class MenuActionResult 90144d93782SGreg Clayton { 90244d93782SGreg Clayton Handled, 90344d93782SGreg Clayton NotHandled, 90444d93782SGreg Clayton Quit // Exit all menus and quit 90544d93782SGreg Clayton }; 90644d93782SGreg Clayton 90744d93782SGreg Clayton struct KeyHelp 90844d93782SGreg Clayton { 90944d93782SGreg Clayton int ch; 91044d93782SGreg Clayton const char *description; 91144d93782SGreg Clayton }; 91244d93782SGreg Clayton 91344d93782SGreg Clayton class WindowDelegate 91444d93782SGreg Clayton { 91544d93782SGreg Clayton public: 91644d93782SGreg Clayton virtual 91744d93782SGreg Clayton ~WindowDelegate() 91844d93782SGreg Clayton { 91944d93782SGreg Clayton } 92044d93782SGreg Clayton 92144d93782SGreg Clayton virtual bool 92244d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 92344d93782SGreg Clayton { 92444d93782SGreg Clayton return false; // Drawing not handled 92544d93782SGreg Clayton } 92644d93782SGreg Clayton 92744d93782SGreg Clayton virtual HandleCharResult 92844d93782SGreg Clayton WindowDelegateHandleChar (Window &window, int key) 92944d93782SGreg Clayton { 93044d93782SGreg Clayton return eKeyNotHandled; 93144d93782SGreg Clayton } 93244d93782SGreg Clayton 93344d93782SGreg Clayton virtual const char * 93444d93782SGreg Clayton WindowDelegateGetHelpText () 93544d93782SGreg Clayton { 93644d93782SGreg Clayton return NULL; 93744d93782SGreg Clayton } 93844d93782SGreg Clayton 93944d93782SGreg Clayton virtual KeyHelp * 94044d93782SGreg Clayton WindowDelegateGetKeyHelp () 94144d93782SGreg Clayton { 94244d93782SGreg Clayton return NULL; 94344d93782SGreg Clayton } 94444d93782SGreg Clayton }; 94544d93782SGreg Clayton 94644d93782SGreg Clayton class HelpDialogDelegate : 94744d93782SGreg Clayton public WindowDelegate 94844d93782SGreg Clayton { 94944d93782SGreg Clayton public: 95044d93782SGreg Clayton HelpDialogDelegate (const char *text, KeyHelp *key_help_array); 95144d93782SGreg Clayton 95244d93782SGreg Clayton virtual 95344d93782SGreg Clayton ~HelpDialogDelegate(); 95444d93782SGreg Clayton 95544d93782SGreg Clayton virtual bool 95644d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force); 95744d93782SGreg Clayton 95844d93782SGreg Clayton virtual HandleCharResult 95944d93782SGreg Clayton WindowDelegateHandleChar (Window &window, int key); 96044d93782SGreg Clayton 96144d93782SGreg Clayton size_t 96244d93782SGreg Clayton GetNumLines() const 96344d93782SGreg Clayton { 96444d93782SGreg Clayton return m_text.GetSize(); 96544d93782SGreg Clayton } 96644d93782SGreg Clayton 96744d93782SGreg Clayton size_t 96844d93782SGreg Clayton GetMaxLineLength () const 96944d93782SGreg Clayton { 97044d93782SGreg Clayton return m_text.GetMaxStringLength(); 97144d93782SGreg Clayton } 97244d93782SGreg Clayton 97344d93782SGreg Clayton protected: 97444d93782SGreg Clayton StringList m_text; 97544d93782SGreg Clayton int m_first_visible_line; 97644d93782SGreg Clayton }; 97744d93782SGreg Clayton 97844d93782SGreg Clayton 97944d93782SGreg Clayton class Window 98044d93782SGreg Clayton { 98144d93782SGreg Clayton public: 98244d93782SGreg Clayton 98344d93782SGreg Clayton Window (const char *name) : 98444d93782SGreg Clayton m_name (name), 98544d93782SGreg Clayton m_window (NULL), 98644d93782SGreg Clayton m_panel (NULL), 98744d93782SGreg Clayton m_parent (NULL), 98844d93782SGreg Clayton m_subwindows (), 98944d93782SGreg Clayton m_delegate_sp (), 99044d93782SGreg Clayton m_curr_active_window_idx (UINT32_MAX), 99144d93782SGreg Clayton m_prev_active_window_idx (UINT32_MAX), 99244d93782SGreg Clayton m_delete (false), 99344d93782SGreg Clayton m_needs_update (true), 99444d93782SGreg Clayton m_can_activate (true), 99544d93782SGreg Clayton m_is_subwin (false) 99644d93782SGreg Clayton { 99744d93782SGreg Clayton } 99844d93782SGreg Clayton 99944d93782SGreg Clayton Window (const char *name, WINDOW *w, bool del = true) : 100044d93782SGreg Clayton m_name (name), 100144d93782SGreg Clayton m_window (NULL), 100244d93782SGreg Clayton m_panel (NULL), 100344d93782SGreg Clayton m_parent (NULL), 100444d93782SGreg Clayton m_subwindows (), 100544d93782SGreg Clayton m_delegate_sp (), 100644d93782SGreg Clayton m_curr_active_window_idx (UINT32_MAX), 100744d93782SGreg Clayton m_prev_active_window_idx (UINT32_MAX), 100844d93782SGreg Clayton m_delete (del), 100944d93782SGreg Clayton m_needs_update (true), 101044d93782SGreg Clayton m_can_activate (true), 101144d93782SGreg Clayton m_is_subwin (false) 101244d93782SGreg Clayton { 101344d93782SGreg Clayton if (w) 101444d93782SGreg Clayton Reset(w); 101544d93782SGreg Clayton } 101644d93782SGreg Clayton 101744d93782SGreg Clayton Window (const char *name, const Rect &bounds) : 101844d93782SGreg Clayton m_name (name), 101944d93782SGreg Clayton m_window (NULL), 102044d93782SGreg Clayton m_parent (NULL), 102144d93782SGreg Clayton m_subwindows (), 102244d93782SGreg Clayton m_delegate_sp (), 102344d93782SGreg Clayton m_curr_active_window_idx (UINT32_MAX), 102444d93782SGreg Clayton m_prev_active_window_idx (UINT32_MAX), 102544d93782SGreg Clayton m_delete (true), 102644d93782SGreg Clayton m_needs_update (true), 102744d93782SGreg Clayton m_can_activate (true), 102844d93782SGreg Clayton m_is_subwin (false) 102944d93782SGreg Clayton { 103044d93782SGreg Clayton Reset (::newwin (bounds.size.height, bounds.size.width, bounds.origin.y, bounds.origin.y)); 103144d93782SGreg Clayton } 103244d93782SGreg Clayton 103344d93782SGreg Clayton virtual 103444d93782SGreg Clayton ~Window () 103544d93782SGreg Clayton { 103644d93782SGreg Clayton RemoveSubWindows (); 103744d93782SGreg Clayton Reset (); 103844d93782SGreg Clayton } 103944d93782SGreg Clayton 104044d93782SGreg Clayton void 104144d93782SGreg Clayton Reset (WINDOW *w = NULL, bool del = true) 104244d93782SGreg Clayton { 104344d93782SGreg Clayton if (m_window == w) 104444d93782SGreg Clayton return; 104544d93782SGreg Clayton 104644d93782SGreg Clayton if (m_panel) 104744d93782SGreg Clayton { 104844d93782SGreg Clayton ::del_panel (m_panel); 104944d93782SGreg Clayton m_panel = NULL; 105044d93782SGreg Clayton } 105144d93782SGreg Clayton if (m_window && m_delete) 105244d93782SGreg Clayton { 105344d93782SGreg Clayton ::delwin (m_window); 105444d93782SGreg Clayton m_window = NULL; 105544d93782SGreg Clayton m_delete = false; 105644d93782SGreg Clayton } 105744d93782SGreg Clayton if (w) 105844d93782SGreg Clayton { 105944d93782SGreg Clayton m_window = w; 106044d93782SGreg Clayton m_panel = ::new_panel (m_window); 106144d93782SGreg Clayton m_delete = del; 106244d93782SGreg Clayton } 106344d93782SGreg Clayton } 106444d93782SGreg Clayton 106544d93782SGreg Clayton void AttributeOn (attr_t attr) { ::wattron (m_window, attr); } 106644d93782SGreg Clayton void AttributeOff (attr_t attr) { ::wattroff (m_window, attr); } 106744d93782SGreg Clayton void Box (chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) { ::box(m_window, v_char, h_char); } 106844d93782SGreg Clayton void Clear () { ::wclear (m_window); } 106944d93782SGreg Clayton void Erase () { ::werase (m_window); } 107044d93782SGreg Clayton Rect GetBounds () { return Rect (GetParentOrigin(), GetSize()); } // Get the rectangle in our parent window 107144d93782SGreg Clayton int GetChar () { return ::wgetch (m_window); } 107244d93782SGreg Clayton int GetCursorX () { return getcurx (m_window); } 107344d93782SGreg Clayton int GetCursorY () { return getcury (m_window); } 107444d93782SGreg Clayton Rect GetFrame () { return Rect (Point(), GetSize()); } // Get our rectangle in our own coordinate system 107544d93782SGreg Clayton Point GetParentOrigin() { return Point (GetParentX(), GetParentY()); } 107644d93782SGreg Clayton Size GetSize() { return Size (GetWidth(), GetHeight()); } 107744d93782SGreg Clayton int GetParentX () { return getparx (m_window); } 107844d93782SGreg Clayton int GetParentY () { return getpary (m_window); } 107944d93782SGreg Clayton int GetMaxX() { return getmaxx (m_window); } 108044d93782SGreg Clayton int GetMaxY() { return getmaxy (m_window); } 108144d93782SGreg Clayton int GetWidth() { return GetMaxX(); } 108244d93782SGreg Clayton int GetHeight() { return GetMaxY(); } 108344d93782SGreg Clayton void MoveCursor (int x, int y) { ::wmove (m_window, y, x); } 108444d93782SGreg Clayton void MoveWindow (int x, int y) { MoveWindow(Point(x,y)); } 108544d93782SGreg Clayton void Resize (int w, int h) { ::wresize(m_window, h, w); } 108644d93782SGreg Clayton void Resize (const Size &size) { ::wresize(m_window, size.height, size.width); } 108744d93782SGreg Clayton void PutChar (int ch) { ::waddch (m_window, ch); } 108844d93782SGreg Clayton void PutCString (const char *s, int len = -1) { ::waddnstr (m_window, s, len); } 108944d93782SGreg Clayton void Refresh () { ::wrefresh (m_window); } 109044d93782SGreg Clayton void DeferredRefresh () 109144d93782SGreg Clayton { 109244d93782SGreg Clayton // We are using panels, so we don't need to call this... 109344d93782SGreg Clayton //::wnoutrefresh(m_window); 109444d93782SGreg Clayton } 109544d93782SGreg Clayton void SetBackground (int color_pair_idx) { ::wbkgd (m_window,COLOR_PAIR(color_pair_idx)); } 109644d93782SGreg Clayton void UnderlineOn () { AttributeOn(A_UNDERLINE); } 109744d93782SGreg Clayton void UnderlineOff () { AttributeOff(A_UNDERLINE); } 109844d93782SGreg Clayton 109944d93782SGreg Clayton void PutCStringTruncated (const char *s, int right_pad) 110044d93782SGreg Clayton { 110144d93782SGreg Clayton int bytes_left = GetWidth() - GetCursorX(); 110244d93782SGreg Clayton if (bytes_left > right_pad) 110344d93782SGreg Clayton { 110444d93782SGreg Clayton bytes_left -= right_pad; 110544d93782SGreg Clayton ::waddnstr (m_window, s, bytes_left); 110644d93782SGreg Clayton } 110744d93782SGreg Clayton } 110844d93782SGreg Clayton 110944d93782SGreg Clayton void 111044d93782SGreg Clayton MoveWindow (const Point &origin) 111144d93782SGreg Clayton { 111244d93782SGreg Clayton const bool moving_window = origin != GetParentOrigin(); 111344d93782SGreg Clayton if (m_is_subwin && moving_window) 111444d93782SGreg Clayton { 111544d93782SGreg Clayton // Can't move subwindows, must delete and re-create 111644d93782SGreg Clayton Size size = GetSize(); 111744d93782SGreg Clayton Reset (::subwin (m_parent->m_window, 111844d93782SGreg Clayton size.height, 111944d93782SGreg Clayton size.width, 112044d93782SGreg Clayton origin.y, 112144d93782SGreg Clayton origin.x), true); 112244d93782SGreg Clayton } 112344d93782SGreg Clayton else 112444d93782SGreg Clayton { 112544d93782SGreg Clayton ::mvwin (m_window, origin.y, origin.x); 112644d93782SGreg Clayton } 112744d93782SGreg Clayton } 112844d93782SGreg Clayton 112944d93782SGreg Clayton void 113044d93782SGreg Clayton SetBounds (const Rect &bounds) 113144d93782SGreg Clayton { 113244d93782SGreg Clayton const bool moving_window = bounds.origin != GetParentOrigin(); 113344d93782SGreg Clayton if (m_is_subwin && moving_window) 113444d93782SGreg Clayton { 113544d93782SGreg Clayton // Can't move subwindows, must delete and re-create 113644d93782SGreg Clayton Reset (::subwin (m_parent->m_window, 113744d93782SGreg Clayton bounds.size.height, 113844d93782SGreg Clayton bounds.size.width, 113944d93782SGreg Clayton bounds.origin.y, 114044d93782SGreg Clayton bounds.origin.x), true); 114144d93782SGreg Clayton } 114244d93782SGreg Clayton else 114344d93782SGreg Clayton { 114444d93782SGreg Clayton if (moving_window) 114544d93782SGreg Clayton MoveWindow(bounds.origin); 114644d93782SGreg Clayton Resize (bounds.size); 114744d93782SGreg Clayton } 114844d93782SGreg Clayton } 114944d93782SGreg Clayton 115044d93782SGreg Clayton void 115144d93782SGreg Clayton Printf (const char *format, ...) __attribute__ ((format (printf, 2, 3))) 115244d93782SGreg Clayton { 115344d93782SGreg Clayton va_list args; 115444d93782SGreg Clayton va_start (args, format); 115544d93782SGreg Clayton vwprintw(m_window, format, args); 115644d93782SGreg Clayton va_end (args); 115744d93782SGreg Clayton } 115844d93782SGreg Clayton 115944d93782SGreg Clayton void 116044d93782SGreg Clayton Touch () 116144d93782SGreg Clayton { 116244d93782SGreg Clayton ::touchwin (m_window); 116344d93782SGreg Clayton if (m_parent) 116444d93782SGreg Clayton m_parent->Touch(); 116544d93782SGreg Clayton } 116644d93782SGreg Clayton 116744d93782SGreg Clayton WindowSP 116844d93782SGreg Clayton CreateSubWindow (const char *name, const Rect &bounds, bool make_active) 116944d93782SGreg Clayton { 117044d93782SGreg Clayton WindowSP subwindow_sp; 117144d93782SGreg Clayton if (m_window) 117244d93782SGreg Clayton { 117344d93782SGreg Clayton subwindow_sp.reset(new Window(name, ::subwin (m_window, 117444d93782SGreg Clayton bounds.size.height, 117544d93782SGreg Clayton bounds.size.width, 117644d93782SGreg Clayton bounds.origin.y, 117744d93782SGreg Clayton bounds.origin.x), true)); 117844d93782SGreg Clayton subwindow_sp->m_is_subwin = true; 117944d93782SGreg Clayton } 118044d93782SGreg Clayton else 118144d93782SGreg Clayton { 118244d93782SGreg Clayton subwindow_sp.reset(new Window(name, ::newwin (bounds.size.height, 118344d93782SGreg Clayton bounds.size.width, 118444d93782SGreg Clayton bounds.origin.y, 118544d93782SGreg Clayton bounds.origin.x), true)); 118644d93782SGreg Clayton subwindow_sp->m_is_subwin = false; 118744d93782SGreg Clayton } 118844d93782SGreg Clayton subwindow_sp->m_parent = this; 118944d93782SGreg Clayton if (make_active) 119044d93782SGreg Clayton { 119144d93782SGreg Clayton m_prev_active_window_idx = m_curr_active_window_idx; 119244d93782SGreg Clayton m_curr_active_window_idx = m_subwindows.size(); 119344d93782SGreg Clayton } 119444d93782SGreg Clayton m_subwindows.push_back(subwindow_sp); 119544d93782SGreg Clayton ::top_panel (subwindow_sp->m_panel); 119644d93782SGreg Clayton m_needs_update = true; 119744d93782SGreg Clayton return subwindow_sp; 119844d93782SGreg Clayton } 119944d93782SGreg Clayton 120044d93782SGreg Clayton bool 120144d93782SGreg Clayton RemoveSubWindow (Window *window) 120244d93782SGreg Clayton { 120344d93782SGreg Clayton Windows::iterator pos, end = m_subwindows.end(); 120444d93782SGreg Clayton size_t i = 0; 120544d93782SGreg Clayton for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) 120644d93782SGreg Clayton { 120744d93782SGreg Clayton if ((*pos).get() == window) 120844d93782SGreg Clayton { 120944d93782SGreg Clayton if (m_prev_active_window_idx == i) 121044d93782SGreg Clayton m_prev_active_window_idx = UINT32_MAX; 121144d93782SGreg Clayton else if (m_prev_active_window_idx != UINT32_MAX && m_prev_active_window_idx > i) 121244d93782SGreg Clayton --m_prev_active_window_idx; 121344d93782SGreg Clayton 121444d93782SGreg Clayton if (m_curr_active_window_idx == i) 121544d93782SGreg Clayton m_curr_active_window_idx = UINT32_MAX; 121644d93782SGreg Clayton else if (m_curr_active_window_idx != UINT32_MAX && m_curr_active_window_idx > i) 121744d93782SGreg Clayton --m_curr_active_window_idx; 121844d93782SGreg Clayton window->Erase(); 121944d93782SGreg Clayton m_subwindows.erase(pos); 122044d93782SGreg Clayton m_needs_update = true; 122144d93782SGreg Clayton if (m_parent) 122244d93782SGreg Clayton m_parent->Touch(); 122344d93782SGreg Clayton else 122444d93782SGreg Clayton ::touchwin (stdscr); 122544d93782SGreg Clayton return true; 122644d93782SGreg Clayton } 122744d93782SGreg Clayton } 122844d93782SGreg Clayton return false; 122944d93782SGreg Clayton } 123044d93782SGreg Clayton 123144d93782SGreg Clayton WindowSP 123244d93782SGreg Clayton FindSubWindow (const char *name) 123344d93782SGreg Clayton { 123444d93782SGreg Clayton Windows::iterator pos, end = m_subwindows.end(); 123544d93782SGreg Clayton size_t i = 0; 123644d93782SGreg Clayton for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) 123744d93782SGreg Clayton { 123844d93782SGreg Clayton if ((*pos)->m_name.compare(name) == 0) 123944d93782SGreg Clayton return *pos; 124044d93782SGreg Clayton } 124144d93782SGreg Clayton return WindowSP(); 124244d93782SGreg Clayton } 124344d93782SGreg Clayton 124444d93782SGreg Clayton void 124544d93782SGreg Clayton RemoveSubWindows () 124644d93782SGreg Clayton { 124744d93782SGreg Clayton m_curr_active_window_idx = UINT32_MAX; 124844d93782SGreg Clayton m_prev_active_window_idx = UINT32_MAX; 124944d93782SGreg Clayton for (Windows::iterator pos = m_subwindows.begin(); 125044d93782SGreg Clayton pos != m_subwindows.end(); 125144d93782SGreg Clayton pos = m_subwindows.erase(pos)) 125244d93782SGreg Clayton { 125344d93782SGreg Clayton (*pos)->Erase(); 125444d93782SGreg Clayton } 125544d93782SGreg Clayton if (m_parent) 125644d93782SGreg Clayton m_parent->Touch(); 125744d93782SGreg Clayton else 125844d93782SGreg Clayton ::touchwin (stdscr); 125944d93782SGreg Clayton } 126044d93782SGreg Clayton 126144d93782SGreg Clayton WINDOW * 126244d93782SGreg Clayton get() 126344d93782SGreg Clayton { 126444d93782SGreg Clayton return m_window; 126544d93782SGreg Clayton } 126644d93782SGreg Clayton 126744d93782SGreg Clayton operator WINDOW *() 126844d93782SGreg Clayton { 126944d93782SGreg Clayton return m_window; 127044d93782SGreg Clayton } 127144d93782SGreg Clayton 127244d93782SGreg Clayton //---------------------------------------------------------------------- 127344d93782SGreg Clayton // Window drawing utilities 127444d93782SGreg Clayton //---------------------------------------------------------------------- 127544d93782SGreg Clayton void 127644d93782SGreg Clayton DrawTitleBox (const char *title, const char *bottom_message = NULL) 127744d93782SGreg Clayton { 127844d93782SGreg Clayton attr_t attr = 0; 127944d93782SGreg Clayton if (IsActive()) 128044d93782SGreg Clayton attr = A_BOLD | COLOR_PAIR(2); 128144d93782SGreg Clayton else 128244d93782SGreg Clayton attr = 0; 128344d93782SGreg Clayton if (attr) 128444d93782SGreg Clayton AttributeOn(attr); 128544d93782SGreg Clayton 128644d93782SGreg Clayton Box(); 128744d93782SGreg Clayton MoveCursor(3, 0); 128844d93782SGreg Clayton 128944d93782SGreg Clayton if (title && title[0]) 129044d93782SGreg Clayton { 129144d93782SGreg Clayton PutChar ('<'); 129244d93782SGreg Clayton PutCString (title); 129344d93782SGreg Clayton PutChar ('>'); 129444d93782SGreg Clayton } 129544d93782SGreg Clayton 129644d93782SGreg Clayton if (bottom_message && bottom_message[0]) 129744d93782SGreg Clayton { 129844d93782SGreg Clayton int bottom_message_length = strlen(bottom_message); 129944d93782SGreg Clayton int x = GetWidth() - 3 - (bottom_message_length + 2); 130044d93782SGreg Clayton 130144d93782SGreg Clayton if (x > 0) 130244d93782SGreg Clayton { 130344d93782SGreg Clayton MoveCursor (x, GetHeight() - 1); 130444d93782SGreg Clayton PutChar ('['); 130544d93782SGreg Clayton PutCString(bottom_message); 130644d93782SGreg Clayton PutChar (']'); 130744d93782SGreg Clayton } 130844d93782SGreg Clayton else 130944d93782SGreg Clayton { 131044d93782SGreg Clayton MoveCursor (1, GetHeight() - 1); 131144d93782SGreg Clayton PutChar ('['); 131244d93782SGreg Clayton PutCStringTruncated (bottom_message, 1); 131344d93782SGreg Clayton } 131444d93782SGreg Clayton } 131544d93782SGreg Clayton if (attr) 131644d93782SGreg Clayton AttributeOff(attr); 131744d93782SGreg Clayton 131844d93782SGreg Clayton } 131944d93782SGreg Clayton 132044d93782SGreg Clayton virtual void 132144d93782SGreg Clayton Draw (bool force) 132244d93782SGreg Clayton { 132344d93782SGreg Clayton if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw (*this, force)) 132444d93782SGreg Clayton return; 132544d93782SGreg Clayton 132644d93782SGreg Clayton for (auto &subwindow_sp : m_subwindows) 132744d93782SGreg Clayton subwindow_sp->Draw(force); 132844d93782SGreg Clayton } 132944d93782SGreg Clayton 133044d93782SGreg Clayton bool 133144d93782SGreg Clayton CreateHelpSubwindow () 133244d93782SGreg Clayton { 133344d93782SGreg Clayton if (m_delegate_sp) 133444d93782SGreg Clayton { 133544d93782SGreg Clayton const char *text = m_delegate_sp->WindowDelegateGetHelpText (); 133644d93782SGreg Clayton KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp (); 133744d93782SGreg Clayton if ((text && text[0]) || key_help) 133844d93782SGreg Clayton { 133944d93782SGreg Clayton std::auto_ptr<HelpDialogDelegate> help_delegate_ap(new HelpDialogDelegate(text, key_help)); 134044d93782SGreg Clayton const size_t num_lines = help_delegate_ap->GetNumLines(); 134144d93782SGreg Clayton const size_t max_length = help_delegate_ap->GetMaxLineLength(); 134244d93782SGreg Clayton Rect bounds = GetBounds(); 134344d93782SGreg Clayton bounds.Inset(1, 1); 13443985c8c6SSaleem Abdulrasool if (max_length + 4 < static_cast<size_t>(bounds.size.width)) 134544d93782SGreg Clayton { 134644d93782SGreg Clayton bounds.origin.x += (bounds.size.width - max_length + 4)/2; 134744d93782SGreg Clayton bounds.size.width = max_length + 4; 134844d93782SGreg Clayton } 134944d93782SGreg Clayton else 135044d93782SGreg Clayton { 135144d93782SGreg Clayton if (bounds.size.width > 100) 135244d93782SGreg Clayton { 135344d93782SGreg Clayton const int inset_w = bounds.size.width / 4; 135444d93782SGreg Clayton bounds.origin.x += inset_w; 135544d93782SGreg Clayton bounds.size.width -= 2*inset_w; 135644d93782SGreg Clayton } 135744d93782SGreg Clayton } 135844d93782SGreg Clayton 13593985c8c6SSaleem Abdulrasool if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) 136044d93782SGreg Clayton { 136144d93782SGreg Clayton bounds.origin.y += (bounds.size.height - num_lines + 2)/2; 136244d93782SGreg Clayton bounds.size.height = num_lines + 2; 136344d93782SGreg Clayton } 136444d93782SGreg Clayton else 136544d93782SGreg Clayton { 136644d93782SGreg Clayton if (bounds.size.height > 100) 136744d93782SGreg Clayton { 136844d93782SGreg Clayton const int inset_h = bounds.size.height / 4; 136944d93782SGreg Clayton bounds.origin.y += inset_h; 137044d93782SGreg Clayton bounds.size.height -= 2*inset_h; 137144d93782SGreg Clayton } 137244d93782SGreg Clayton } 13735fdb09bbSGreg Clayton WindowSP help_window_sp; 13745fdb09bbSGreg Clayton Window *parent_window = GetParent(); 13755fdb09bbSGreg Clayton if (parent_window) 13765fdb09bbSGreg Clayton help_window_sp = parent_window->CreateSubWindow("Help", bounds, true); 13775fdb09bbSGreg Clayton else 13785fdb09bbSGreg Clayton help_window_sp = CreateSubWindow("Help", bounds, true); 137944d93782SGreg Clayton help_window_sp->SetDelegate(WindowDelegateSP(help_delegate_ap.release())); 138044d93782SGreg Clayton return true; 138144d93782SGreg Clayton } 138244d93782SGreg Clayton } 138344d93782SGreg Clayton return false; 138444d93782SGreg Clayton } 138544d93782SGreg Clayton 138644d93782SGreg Clayton virtual HandleCharResult 138744d93782SGreg Clayton HandleChar (int key) 138844d93782SGreg Clayton { 138944d93782SGreg Clayton // Always check the active window first 139044d93782SGreg Clayton HandleCharResult result = eKeyNotHandled; 139144d93782SGreg Clayton WindowSP active_window_sp = GetActiveWindow (); 139244d93782SGreg Clayton if (active_window_sp) 139344d93782SGreg Clayton { 139444d93782SGreg Clayton result = active_window_sp->HandleChar (key); 139544d93782SGreg Clayton if (result != eKeyNotHandled) 139644d93782SGreg Clayton return result; 139744d93782SGreg Clayton } 139844d93782SGreg Clayton 139944d93782SGreg Clayton if (m_delegate_sp) 140044d93782SGreg Clayton { 140144d93782SGreg Clayton result = m_delegate_sp->WindowDelegateHandleChar (*this, key); 140244d93782SGreg Clayton if (result != eKeyNotHandled) 140344d93782SGreg Clayton return result; 140444d93782SGreg Clayton } 140544d93782SGreg Clayton 140644d93782SGreg Clayton // Then check for any windows that want any keys 140744d93782SGreg Clayton // that weren't handled. This is typically only 140844d93782SGreg Clayton // for a menubar. 140944d93782SGreg Clayton // Make a copy of the subwindows in case any HandleChar() 141044d93782SGreg Clayton // functions muck with the subwindows. If we don't do this, 141144d93782SGreg Clayton // we can crash when iterating over the subwindows. 141244d93782SGreg Clayton Windows subwindows (m_subwindows); 141344d93782SGreg Clayton for (auto subwindow_sp : subwindows) 141444d93782SGreg Clayton { 141544d93782SGreg Clayton if (subwindow_sp->m_can_activate == false) 141644d93782SGreg Clayton { 141744d93782SGreg Clayton HandleCharResult result = subwindow_sp->HandleChar(key); 141844d93782SGreg Clayton if (result != eKeyNotHandled) 141944d93782SGreg Clayton return result; 142044d93782SGreg Clayton } 142144d93782SGreg Clayton } 142244d93782SGreg Clayton 142344d93782SGreg Clayton return eKeyNotHandled; 142444d93782SGreg Clayton } 142544d93782SGreg Clayton 142644d93782SGreg Clayton bool 142744d93782SGreg Clayton SetActiveWindow (Window *window) 142844d93782SGreg Clayton { 142944d93782SGreg Clayton const size_t num_subwindows = m_subwindows.size(); 143044d93782SGreg Clayton for (size_t i=0; i<num_subwindows; ++i) 143144d93782SGreg Clayton { 143244d93782SGreg Clayton if (m_subwindows[i].get() == window) 143344d93782SGreg Clayton { 143444d93782SGreg Clayton m_prev_active_window_idx = m_curr_active_window_idx; 143544d93782SGreg Clayton ::top_panel (window->m_panel); 143644d93782SGreg Clayton m_curr_active_window_idx = i; 143744d93782SGreg Clayton return true; 143844d93782SGreg Clayton } 143944d93782SGreg Clayton } 144044d93782SGreg Clayton return false; 144144d93782SGreg Clayton } 144244d93782SGreg Clayton 144344d93782SGreg Clayton WindowSP 144444d93782SGreg Clayton GetActiveWindow () 144544d93782SGreg Clayton { 144644d93782SGreg Clayton if (!m_subwindows.empty()) 144744d93782SGreg Clayton { 144844d93782SGreg Clayton if (m_curr_active_window_idx >= m_subwindows.size()) 144944d93782SGreg Clayton { 145044d93782SGreg Clayton if (m_prev_active_window_idx < m_subwindows.size()) 145144d93782SGreg Clayton { 145244d93782SGreg Clayton m_curr_active_window_idx = m_prev_active_window_idx; 145344d93782SGreg Clayton m_prev_active_window_idx = UINT32_MAX; 145444d93782SGreg Clayton } 145544d93782SGreg Clayton else if (IsActive()) 145644d93782SGreg Clayton { 145744d93782SGreg Clayton m_prev_active_window_idx = UINT32_MAX; 145844d93782SGreg Clayton m_curr_active_window_idx = UINT32_MAX; 145944d93782SGreg Clayton 146044d93782SGreg Clayton // Find first window that wants to be active if this window is active 146144d93782SGreg Clayton const size_t num_subwindows = m_subwindows.size(); 146244d93782SGreg Clayton for (size_t i=0; i<num_subwindows; ++i) 146344d93782SGreg Clayton { 146444d93782SGreg Clayton if (m_subwindows[i]->GetCanBeActive()) 146544d93782SGreg Clayton { 146644d93782SGreg Clayton m_curr_active_window_idx = i; 146744d93782SGreg Clayton break; 146844d93782SGreg Clayton } 146944d93782SGreg Clayton } 147044d93782SGreg Clayton } 147144d93782SGreg Clayton } 147244d93782SGreg Clayton 147344d93782SGreg Clayton if (m_curr_active_window_idx < m_subwindows.size()) 147444d93782SGreg Clayton return m_subwindows[m_curr_active_window_idx]; 147544d93782SGreg Clayton } 147644d93782SGreg Clayton return WindowSP(); 147744d93782SGreg Clayton } 147844d93782SGreg Clayton 147944d93782SGreg Clayton bool 148044d93782SGreg Clayton GetCanBeActive () const 148144d93782SGreg Clayton { 148244d93782SGreg Clayton return m_can_activate; 148344d93782SGreg Clayton } 148444d93782SGreg Clayton 148544d93782SGreg Clayton void 148644d93782SGreg Clayton SetCanBeActive (bool b) 148744d93782SGreg Clayton { 148844d93782SGreg Clayton m_can_activate = b; 148944d93782SGreg Clayton } 149044d93782SGreg Clayton 149144d93782SGreg Clayton const WindowDelegateSP & 149244d93782SGreg Clayton GetDelegate () const 149344d93782SGreg Clayton { 149444d93782SGreg Clayton return m_delegate_sp; 149544d93782SGreg Clayton } 149644d93782SGreg Clayton 149744d93782SGreg Clayton void 149844d93782SGreg Clayton SetDelegate (const WindowDelegateSP &delegate_sp) 149944d93782SGreg Clayton { 150044d93782SGreg Clayton m_delegate_sp = delegate_sp; 150144d93782SGreg Clayton } 150244d93782SGreg Clayton 150344d93782SGreg Clayton Window * 150444d93782SGreg Clayton GetParent () const 150544d93782SGreg Clayton { 150644d93782SGreg Clayton return m_parent; 150744d93782SGreg Clayton } 150844d93782SGreg Clayton 150944d93782SGreg Clayton bool 151044d93782SGreg Clayton IsActive () const 151144d93782SGreg Clayton { 151244d93782SGreg Clayton if (m_parent) 151344d93782SGreg Clayton return m_parent->GetActiveWindow().get() == this; 151444d93782SGreg Clayton else 151544d93782SGreg Clayton return true; // Top level window is always active 151644d93782SGreg Clayton } 151744d93782SGreg Clayton 151844d93782SGreg Clayton void 151944d93782SGreg Clayton SelectNextWindowAsActive () 152044d93782SGreg Clayton { 152144d93782SGreg Clayton // Move active focus to next window 152244d93782SGreg Clayton const size_t num_subwindows = m_subwindows.size(); 152344d93782SGreg Clayton if (m_curr_active_window_idx == UINT32_MAX) 152444d93782SGreg Clayton { 152544d93782SGreg Clayton uint32_t idx = 0; 152644d93782SGreg Clayton for (auto subwindow_sp : m_subwindows) 152744d93782SGreg Clayton { 152844d93782SGreg Clayton if (subwindow_sp->GetCanBeActive()) 152944d93782SGreg Clayton { 153044d93782SGreg Clayton m_curr_active_window_idx = idx; 153144d93782SGreg Clayton break; 153244d93782SGreg Clayton } 153344d93782SGreg Clayton ++idx; 153444d93782SGreg Clayton } 153544d93782SGreg Clayton } 153644d93782SGreg Clayton else if (m_curr_active_window_idx + 1 < num_subwindows) 153744d93782SGreg Clayton { 153844d93782SGreg Clayton bool handled = false; 153944d93782SGreg Clayton m_prev_active_window_idx = m_curr_active_window_idx; 154044d93782SGreg Clayton for (size_t idx=m_curr_active_window_idx + 1; idx<num_subwindows; ++idx) 154144d93782SGreg Clayton { 154244d93782SGreg Clayton if (m_subwindows[idx]->GetCanBeActive()) 154344d93782SGreg Clayton { 154444d93782SGreg Clayton m_curr_active_window_idx = idx; 154544d93782SGreg Clayton handled = true; 154644d93782SGreg Clayton break; 154744d93782SGreg Clayton } 154844d93782SGreg Clayton } 154944d93782SGreg Clayton if (!handled) 155044d93782SGreg Clayton { 155144d93782SGreg Clayton for (size_t idx=0; idx<=m_prev_active_window_idx; ++idx) 155244d93782SGreg Clayton { 155344d93782SGreg Clayton if (m_subwindows[idx]->GetCanBeActive()) 155444d93782SGreg Clayton { 155544d93782SGreg Clayton m_curr_active_window_idx = idx; 155644d93782SGreg Clayton break; 155744d93782SGreg Clayton } 155844d93782SGreg Clayton } 155944d93782SGreg Clayton } 156044d93782SGreg Clayton } 156144d93782SGreg Clayton else 156244d93782SGreg Clayton { 156344d93782SGreg Clayton m_prev_active_window_idx = m_curr_active_window_idx; 156444d93782SGreg Clayton for (size_t idx=0; idx<num_subwindows; ++idx) 156544d93782SGreg Clayton { 156644d93782SGreg Clayton if (m_subwindows[idx]->GetCanBeActive()) 156744d93782SGreg Clayton { 156844d93782SGreg Clayton m_curr_active_window_idx = idx; 156944d93782SGreg Clayton break; 157044d93782SGreg Clayton } 157144d93782SGreg Clayton } 157244d93782SGreg Clayton } 157344d93782SGreg Clayton } 157444d93782SGreg Clayton 157544d93782SGreg Clayton const char * 157644d93782SGreg Clayton GetName () const 157744d93782SGreg Clayton { 157844d93782SGreg Clayton return m_name.c_str(); 157944d93782SGreg Clayton } 158044d93782SGreg Clayton protected: 158144d93782SGreg Clayton std::string m_name; 158244d93782SGreg Clayton WINDOW *m_window; 158344d93782SGreg Clayton PANEL *m_panel; 158444d93782SGreg Clayton Window *m_parent; 158544d93782SGreg Clayton Windows m_subwindows; 158644d93782SGreg Clayton WindowDelegateSP m_delegate_sp; 158744d93782SGreg Clayton uint32_t m_curr_active_window_idx; 158844d93782SGreg Clayton uint32_t m_prev_active_window_idx; 158944d93782SGreg Clayton bool m_delete; 159044d93782SGreg Clayton bool m_needs_update; 159144d93782SGreg Clayton bool m_can_activate; 159244d93782SGreg Clayton bool m_is_subwin; 159344d93782SGreg Clayton 159444d93782SGreg Clayton private: 159544d93782SGreg Clayton DISALLOW_COPY_AND_ASSIGN(Window); 159644d93782SGreg Clayton }; 159744d93782SGreg Clayton 159844d93782SGreg Clayton class MenuDelegate 159944d93782SGreg Clayton { 160044d93782SGreg Clayton public: 160144d93782SGreg Clayton virtual ~MenuDelegate() {} 160244d93782SGreg Clayton 160344d93782SGreg Clayton virtual MenuActionResult 160444d93782SGreg Clayton MenuDelegateAction (Menu &menu) = 0; 160544d93782SGreg Clayton }; 160644d93782SGreg Clayton 160744d93782SGreg Clayton class Menu : public WindowDelegate 160844d93782SGreg Clayton { 160944d93782SGreg Clayton public: 161044d93782SGreg Clayton enum class Type 161144d93782SGreg Clayton { 161244d93782SGreg Clayton Invalid, 161344d93782SGreg Clayton Bar, 161444d93782SGreg Clayton Item, 161544d93782SGreg Clayton Separator 161644d93782SGreg Clayton }; 161744d93782SGreg Clayton 161844d93782SGreg Clayton // Menubar or separator constructor 161944d93782SGreg Clayton Menu (Type type); 162044d93782SGreg Clayton 162144d93782SGreg Clayton // Menuitem constructor 162244d93782SGreg Clayton Menu (const char *name, 162344d93782SGreg Clayton const char *key_name, 162444d93782SGreg Clayton int key_value, 162544d93782SGreg Clayton uint64_t identifier); 162644d93782SGreg Clayton 162744d93782SGreg Clayton virtual ~ 162844d93782SGreg Clayton Menu () 162944d93782SGreg Clayton { 163044d93782SGreg Clayton } 163144d93782SGreg Clayton 163244d93782SGreg Clayton const MenuDelegateSP & 163344d93782SGreg Clayton GetDelegate () const 163444d93782SGreg Clayton { 163544d93782SGreg Clayton return m_delegate_sp; 163644d93782SGreg Clayton } 163744d93782SGreg Clayton 163844d93782SGreg Clayton void 163944d93782SGreg Clayton SetDelegate (const MenuDelegateSP &delegate_sp) 164044d93782SGreg Clayton { 164144d93782SGreg Clayton m_delegate_sp = delegate_sp; 164244d93782SGreg Clayton } 164344d93782SGreg Clayton 164444d93782SGreg Clayton void 164544d93782SGreg Clayton RecalculateNameLengths(); 164644d93782SGreg Clayton 164744d93782SGreg Clayton void 164844d93782SGreg Clayton AddSubmenu (const MenuSP &menu_sp); 164944d93782SGreg Clayton 165044d93782SGreg Clayton int 165144d93782SGreg Clayton DrawAndRunMenu (Window &window); 165244d93782SGreg Clayton 165344d93782SGreg Clayton void 165444d93782SGreg Clayton DrawMenuTitle (Window &window, bool highlight); 165544d93782SGreg Clayton 165644d93782SGreg Clayton virtual bool 165744d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force); 165844d93782SGreg Clayton 165944d93782SGreg Clayton virtual HandleCharResult 166044d93782SGreg Clayton WindowDelegateHandleChar (Window &window, int key); 166144d93782SGreg Clayton 166244d93782SGreg Clayton MenuActionResult 166344d93782SGreg Clayton ActionPrivate (Menu &menu) 166444d93782SGreg Clayton { 166544d93782SGreg Clayton MenuActionResult result = MenuActionResult::NotHandled; 166644d93782SGreg Clayton if (m_delegate_sp) 166744d93782SGreg Clayton { 166844d93782SGreg Clayton result = m_delegate_sp->MenuDelegateAction (menu); 166944d93782SGreg Clayton if (result != MenuActionResult::NotHandled) 167044d93782SGreg Clayton return result; 167144d93782SGreg Clayton } 167244d93782SGreg Clayton else if (m_parent) 167344d93782SGreg Clayton { 167444d93782SGreg Clayton result = m_parent->ActionPrivate(menu); 167544d93782SGreg Clayton if (result != MenuActionResult::NotHandled) 167644d93782SGreg Clayton return result; 167744d93782SGreg Clayton } 167844d93782SGreg Clayton return m_canned_result; 167944d93782SGreg Clayton } 168044d93782SGreg Clayton 168144d93782SGreg Clayton MenuActionResult 168244d93782SGreg Clayton Action () 168344d93782SGreg Clayton { 168444d93782SGreg Clayton // Call the recursive action so it can try to handle it 168544d93782SGreg Clayton // with the menu delegate, and if not, try our parent menu 168644d93782SGreg Clayton return ActionPrivate (*this); 168744d93782SGreg Clayton } 168844d93782SGreg Clayton 168944d93782SGreg Clayton void 169044d93782SGreg Clayton SetCannedResult (MenuActionResult result) 169144d93782SGreg Clayton { 169244d93782SGreg Clayton m_canned_result = result; 169344d93782SGreg Clayton } 169444d93782SGreg Clayton 169544d93782SGreg Clayton Menus & 169644d93782SGreg Clayton GetSubmenus() 169744d93782SGreg Clayton { 169844d93782SGreg Clayton return m_submenus; 169944d93782SGreg Clayton } 170044d93782SGreg Clayton 170144d93782SGreg Clayton const Menus & 170244d93782SGreg Clayton GetSubmenus() const 170344d93782SGreg Clayton { 170444d93782SGreg Clayton return m_submenus; 170544d93782SGreg Clayton } 170644d93782SGreg Clayton 170744d93782SGreg Clayton int 170844d93782SGreg Clayton GetSelectedSubmenuIndex () const 170944d93782SGreg Clayton { 171044d93782SGreg Clayton return m_selected; 171144d93782SGreg Clayton } 171244d93782SGreg Clayton 171344d93782SGreg Clayton void 171444d93782SGreg Clayton SetSelectedSubmenuIndex (int idx) 171544d93782SGreg Clayton { 171644d93782SGreg Clayton m_selected = idx; 171744d93782SGreg Clayton } 171844d93782SGreg Clayton 171944d93782SGreg Clayton Type 172044d93782SGreg Clayton GetType () const 172144d93782SGreg Clayton { 172244d93782SGreg Clayton return m_type; 172344d93782SGreg Clayton } 172444d93782SGreg Clayton 172544d93782SGreg Clayton int 172644d93782SGreg Clayton GetStartingColumn() const 172744d93782SGreg Clayton { 172844d93782SGreg Clayton return m_start_col; 172944d93782SGreg Clayton } 173044d93782SGreg Clayton 173144d93782SGreg Clayton void 173244d93782SGreg Clayton SetStartingColumn(int col) 173344d93782SGreg Clayton { 173444d93782SGreg Clayton m_start_col = col; 173544d93782SGreg Clayton } 173644d93782SGreg Clayton 173744d93782SGreg Clayton int 173844d93782SGreg Clayton GetKeyValue() const 173944d93782SGreg Clayton { 174044d93782SGreg Clayton return m_key_value; 174144d93782SGreg Clayton } 174244d93782SGreg Clayton 174344d93782SGreg Clayton void 174444d93782SGreg Clayton SetKeyValue(int key_value) 174544d93782SGreg Clayton { 174644d93782SGreg Clayton m_key_value = key_value; 174744d93782SGreg Clayton } 174844d93782SGreg Clayton 174944d93782SGreg Clayton std::string & 175044d93782SGreg Clayton GetName() 175144d93782SGreg Clayton { 175244d93782SGreg Clayton return m_name; 175344d93782SGreg Clayton } 175444d93782SGreg Clayton 175544d93782SGreg Clayton std::string & 175644d93782SGreg Clayton GetKeyName() 175744d93782SGreg Clayton { 175844d93782SGreg Clayton return m_key_name; 175944d93782SGreg Clayton } 176044d93782SGreg Clayton 176144d93782SGreg Clayton int 176244d93782SGreg Clayton GetDrawWidth () const 176344d93782SGreg Clayton { 176444d93782SGreg Clayton return m_max_submenu_name_length + m_max_submenu_key_name_length + 8; 176544d93782SGreg Clayton } 176644d93782SGreg Clayton 176744d93782SGreg Clayton 176844d93782SGreg Clayton uint64_t 176944d93782SGreg Clayton GetIdentifier() const 177044d93782SGreg Clayton { 177144d93782SGreg Clayton return m_identifier; 177244d93782SGreg Clayton } 177344d93782SGreg Clayton 177444d93782SGreg Clayton void 177544d93782SGreg Clayton SetIdentifier (uint64_t identifier) 177644d93782SGreg Clayton { 177744d93782SGreg Clayton m_identifier = identifier; 177844d93782SGreg Clayton } 177944d93782SGreg Clayton 178044d93782SGreg Clayton protected: 178144d93782SGreg Clayton std::string m_name; 178244d93782SGreg Clayton std::string m_key_name; 178344d93782SGreg Clayton uint64_t m_identifier; 178444d93782SGreg Clayton Type m_type; 178544d93782SGreg Clayton int m_key_value; 178644d93782SGreg Clayton int m_start_col; 178744d93782SGreg Clayton int m_max_submenu_name_length; 178844d93782SGreg Clayton int m_max_submenu_key_name_length; 178944d93782SGreg Clayton int m_selected; 179044d93782SGreg Clayton Menu *m_parent; 179144d93782SGreg Clayton Menus m_submenus; 179244d93782SGreg Clayton WindowSP m_menu_window_sp; 179344d93782SGreg Clayton MenuActionResult m_canned_result; 179444d93782SGreg Clayton MenuDelegateSP m_delegate_sp; 179544d93782SGreg Clayton }; 179644d93782SGreg Clayton 179744d93782SGreg Clayton // Menubar or separator constructor 179844d93782SGreg Clayton Menu::Menu (Type type) : 179944d93782SGreg Clayton m_name (), 180044d93782SGreg Clayton m_key_name (), 180144d93782SGreg Clayton m_identifier (0), 180244d93782SGreg Clayton m_type (type), 180344d93782SGreg Clayton m_key_value (0), 180444d93782SGreg Clayton m_start_col (0), 180544d93782SGreg Clayton m_max_submenu_name_length (0), 180644d93782SGreg Clayton m_max_submenu_key_name_length (0), 180744d93782SGreg Clayton m_selected (0), 180844d93782SGreg Clayton m_parent (NULL), 180944d93782SGreg Clayton m_submenus (), 181044d93782SGreg Clayton m_canned_result (MenuActionResult::NotHandled), 181144d93782SGreg Clayton m_delegate_sp() 181244d93782SGreg Clayton { 181344d93782SGreg Clayton } 181444d93782SGreg Clayton 181544d93782SGreg Clayton // Menuitem constructor 181644d93782SGreg Clayton Menu::Menu (const char *name, 181744d93782SGreg Clayton const char *key_name, 181844d93782SGreg Clayton int key_value, 181944d93782SGreg Clayton uint64_t identifier) : 182044d93782SGreg Clayton m_name (), 182144d93782SGreg Clayton m_key_name (), 182244d93782SGreg Clayton m_identifier (identifier), 182344d93782SGreg Clayton m_type (Type::Invalid), 182444d93782SGreg Clayton m_key_value (key_value), 182544d93782SGreg Clayton m_start_col (0), 182644d93782SGreg Clayton m_max_submenu_name_length (0), 182744d93782SGreg Clayton m_max_submenu_key_name_length (0), 182844d93782SGreg Clayton m_selected (0), 182944d93782SGreg Clayton m_parent (NULL), 183044d93782SGreg Clayton m_submenus (), 183144d93782SGreg Clayton m_canned_result (MenuActionResult::NotHandled), 183244d93782SGreg Clayton m_delegate_sp() 183344d93782SGreg Clayton { 183444d93782SGreg Clayton if (name && name[0]) 183544d93782SGreg Clayton { 183644d93782SGreg Clayton m_name = name; 183744d93782SGreg Clayton m_type = Type::Item; 183844d93782SGreg Clayton if (key_name && key_name[0]) 183944d93782SGreg Clayton m_key_name = key_name; 184044d93782SGreg Clayton } 184144d93782SGreg Clayton else 184244d93782SGreg Clayton { 184344d93782SGreg Clayton m_type = Type::Separator; 184444d93782SGreg Clayton } 184544d93782SGreg Clayton } 184644d93782SGreg Clayton 184744d93782SGreg Clayton void 184844d93782SGreg Clayton Menu::RecalculateNameLengths() 184944d93782SGreg Clayton { 185044d93782SGreg Clayton m_max_submenu_name_length = 0; 185144d93782SGreg Clayton m_max_submenu_key_name_length = 0; 185244d93782SGreg Clayton Menus &submenus = GetSubmenus(); 185344d93782SGreg Clayton const size_t num_submenus = submenus.size(); 185444d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 185544d93782SGreg Clayton { 185644d93782SGreg Clayton Menu *submenu = submenus[i].get(); 18573985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size()) 185844d93782SGreg Clayton m_max_submenu_name_length = submenu->m_name.size(); 18593985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_max_submenu_key_name_length) < submenu->m_key_name.size()) 186044d93782SGreg Clayton m_max_submenu_key_name_length = submenu->m_key_name.size(); 186144d93782SGreg Clayton } 186244d93782SGreg Clayton } 186344d93782SGreg Clayton 186444d93782SGreg Clayton void 186544d93782SGreg Clayton Menu::AddSubmenu (const MenuSP &menu_sp) 186644d93782SGreg Clayton { 186744d93782SGreg Clayton menu_sp->m_parent = this; 18683985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size()) 186944d93782SGreg Clayton m_max_submenu_name_length = menu_sp->m_name.size(); 18703985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_max_submenu_key_name_length) < menu_sp->m_key_name.size()) 187144d93782SGreg Clayton m_max_submenu_key_name_length = menu_sp->m_key_name.size(); 187244d93782SGreg Clayton m_submenus.push_back(menu_sp); 187344d93782SGreg Clayton } 187444d93782SGreg Clayton 187544d93782SGreg Clayton void 187644d93782SGreg Clayton Menu::DrawMenuTitle (Window &window, bool highlight) 187744d93782SGreg Clayton { 187844d93782SGreg Clayton if (m_type == Type::Separator) 187944d93782SGreg Clayton { 188044d93782SGreg Clayton window.MoveCursor(0, window.GetCursorY()); 188144d93782SGreg Clayton window.PutChar(ACS_LTEE); 188244d93782SGreg Clayton int width = window.GetWidth(); 188344d93782SGreg Clayton if (width > 2) 188444d93782SGreg Clayton { 188544d93782SGreg Clayton width -= 2; 18863985c8c6SSaleem Abdulrasool for (int i=0; i< width; ++i) 188744d93782SGreg Clayton window.PutChar(ACS_HLINE); 188844d93782SGreg Clayton } 188944d93782SGreg Clayton window.PutChar(ACS_RTEE); 189044d93782SGreg Clayton } 189144d93782SGreg Clayton else 189244d93782SGreg Clayton { 189344d93782SGreg Clayton const int shortcut_key = m_key_value; 189444d93782SGreg Clayton bool underlined_shortcut = false; 189544d93782SGreg Clayton const attr_t hilgight_attr = A_REVERSE; 189644d93782SGreg Clayton if (highlight) 189744d93782SGreg Clayton window.AttributeOn(hilgight_attr); 189844d93782SGreg Clayton if (isprint(shortcut_key)) 189944d93782SGreg Clayton { 190044d93782SGreg Clayton size_t lower_pos = m_name.find(tolower(shortcut_key)); 190144d93782SGreg Clayton size_t upper_pos = m_name.find(toupper(shortcut_key)); 190244d93782SGreg Clayton const char *name = m_name.c_str(); 190344d93782SGreg Clayton size_t pos = std::min<size_t>(lower_pos, upper_pos); 190444d93782SGreg Clayton if (pos != std::string::npos) 190544d93782SGreg Clayton { 190644d93782SGreg Clayton underlined_shortcut = true; 190744d93782SGreg Clayton if (pos > 0) 190844d93782SGreg Clayton { 190944d93782SGreg Clayton window.PutCString(name, pos); 191044d93782SGreg Clayton name += pos; 191144d93782SGreg Clayton } 191244d93782SGreg Clayton const attr_t shortcut_attr = A_UNDERLINE|A_BOLD; 191344d93782SGreg Clayton window.AttributeOn (shortcut_attr); 191444d93782SGreg Clayton window.PutChar(name[0]); 191544d93782SGreg Clayton window.AttributeOff(shortcut_attr); 191644d93782SGreg Clayton name++; 191744d93782SGreg Clayton if (name[0]) 191844d93782SGreg Clayton window.PutCString(name); 191944d93782SGreg Clayton } 192044d93782SGreg Clayton } 192144d93782SGreg Clayton 192244d93782SGreg Clayton if (!underlined_shortcut) 192344d93782SGreg Clayton { 192444d93782SGreg Clayton window.PutCString(m_name.c_str()); 192544d93782SGreg Clayton } 192644d93782SGreg Clayton 192744d93782SGreg Clayton if (highlight) 192844d93782SGreg Clayton window.AttributeOff(hilgight_attr); 192944d93782SGreg Clayton 193044d93782SGreg Clayton if (m_key_name.empty()) 193144d93782SGreg Clayton { 193244d93782SGreg Clayton if (!underlined_shortcut && isprint(m_key_value)) 193344d93782SGreg Clayton { 193444d93782SGreg Clayton window.AttributeOn (COLOR_PAIR(3)); 193544d93782SGreg Clayton window.Printf (" (%c)", m_key_value); 193644d93782SGreg Clayton window.AttributeOff (COLOR_PAIR(3)); 193744d93782SGreg Clayton } 193844d93782SGreg Clayton } 193944d93782SGreg Clayton else 194044d93782SGreg Clayton { 194144d93782SGreg Clayton window.AttributeOn (COLOR_PAIR(3)); 194244d93782SGreg Clayton window.Printf (" (%s)", m_key_name.c_str()); 194344d93782SGreg Clayton window.AttributeOff (COLOR_PAIR(3)); 194444d93782SGreg Clayton } 194544d93782SGreg Clayton } 194644d93782SGreg Clayton } 194744d93782SGreg Clayton 194844d93782SGreg Clayton bool 194944d93782SGreg Clayton Menu::WindowDelegateDraw (Window &window, bool force) 195044d93782SGreg Clayton { 195144d93782SGreg Clayton Menus &submenus = GetSubmenus(); 195244d93782SGreg Clayton const size_t num_submenus = submenus.size(); 195344d93782SGreg Clayton const int selected_idx = GetSelectedSubmenuIndex(); 195444d93782SGreg Clayton Menu::Type menu_type = GetType (); 195544d93782SGreg Clayton switch (menu_type) 195644d93782SGreg Clayton { 195744d93782SGreg Clayton case Menu::Type::Bar: 195844d93782SGreg Clayton { 195944d93782SGreg Clayton window.SetBackground(2); 196044d93782SGreg Clayton window.MoveCursor(0, 0); 196144d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 196244d93782SGreg Clayton { 196344d93782SGreg Clayton Menu *menu = submenus[i].get(); 196444d93782SGreg Clayton if (i > 0) 196544d93782SGreg Clayton window.PutChar(' '); 196644d93782SGreg Clayton menu->SetStartingColumn (window.GetCursorX()); 196744d93782SGreg Clayton window.PutCString("| "); 196844d93782SGreg Clayton menu->DrawMenuTitle (window, false); 196944d93782SGreg Clayton } 197044d93782SGreg Clayton window.PutCString(" |"); 197144d93782SGreg Clayton window.DeferredRefresh(); 197244d93782SGreg Clayton } 197344d93782SGreg Clayton break; 197444d93782SGreg Clayton 197544d93782SGreg Clayton case Menu::Type::Item: 197644d93782SGreg Clayton { 197744d93782SGreg Clayton int y = 1; 197844d93782SGreg Clayton int x = 3; 197944d93782SGreg Clayton // Draw the menu 198044d93782SGreg Clayton int cursor_x = 0; 198144d93782SGreg Clayton int cursor_y = 0; 198244d93782SGreg Clayton window.Erase(); 198344d93782SGreg Clayton window.SetBackground(2); 198444d93782SGreg Clayton window.Box(); 198544d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 198644d93782SGreg Clayton { 19873985c8c6SSaleem Abdulrasool const bool is_selected = 19883985c8c6SSaleem Abdulrasool (i == static_cast<size_t>(selected_idx)); 198944d93782SGreg Clayton window.MoveCursor(x, y + i); 199044d93782SGreg Clayton if (is_selected) 199144d93782SGreg Clayton { 199244d93782SGreg Clayton // Remember where we want the cursor to be 199344d93782SGreg Clayton cursor_x = x-1; 199444d93782SGreg Clayton cursor_y = y+i; 199544d93782SGreg Clayton } 199644d93782SGreg Clayton submenus[i]->DrawMenuTitle (window, is_selected); 199744d93782SGreg Clayton } 199844d93782SGreg Clayton window.MoveCursor(cursor_x, cursor_y); 199944d93782SGreg Clayton window.DeferredRefresh(); 200044d93782SGreg Clayton } 200144d93782SGreg Clayton break; 200244d93782SGreg Clayton 200344d93782SGreg Clayton default: 200444d93782SGreg Clayton case Menu::Type::Separator: 200544d93782SGreg Clayton break; 200644d93782SGreg Clayton } 200744d93782SGreg Clayton return true; // Drawing handled... 200844d93782SGreg Clayton } 200944d93782SGreg Clayton 201044d93782SGreg Clayton HandleCharResult 201144d93782SGreg Clayton Menu::WindowDelegateHandleChar (Window &window, int key) 201244d93782SGreg Clayton { 201344d93782SGreg Clayton HandleCharResult result = eKeyNotHandled; 201444d93782SGreg Clayton 201544d93782SGreg Clayton Menus &submenus = GetSubmenus(); 201644d93782SGreg Clayton const size_t num_submenus = submenus.size(); 201744d93782SGreg Clayton const int selected_idx = GetSelectedSubmenuIndex(); 201844d93782SGreg Clayton Menu::Type menu_type = GetType (); 201944d93782SGreg Clayton if (menu_type == Menu::Type::Bar) 202044d93782SGreg Clayton { 202144d93782SGreg Clayton MenuSP run_menu_sp; 202244d93782SGreg Clayton switch (key) 202344d93782SGreg Clayton { 202444d93782SGreg Clayton case KEY_DOWN: 202544d93782SGreg Clayton case KEY_UP: 202644d93782SGreg Clayton // Show last menu or first menu 20273985c8c6SSaleem Abdulrasool if (selected_idx < static_cast<int>(num_submenus)) 202844d93782SGreg Clayton run_menu_sp = submenus[selected_idx]; 202944d93782SGreg Clayton else if (!submenus.empty()) 203044d93782SGreg Clayton run_menu_sp = submenus.front(); 203144d93782SGreg Clayton result = eKeyHandled; 203244d93782SGreg Clayton break; 203344d93782SGreg Clayton 203444d93782SGreg Clayton case KEY_RIGHT: 203544d93782SGreg Clayton { 203644d93782SGreg Clayton ++m_selected; 20373985c8c6SSaleem Abdulrasool if (m_selected >= static_cast<int>(num_submenus)) 203844d93782SGreg Clayton m_selected = 0; 20393985c8c6SSaleem Abdulrasool if (m_selected < static_cast<int>(num_submenus)) 204044d93782SGreg Clayton run_menu_sp = submenus[m_selected]; 204144d93782SGreg Clayton else if (!submenus.empty()) 204244d93782SGreg Clayton run_menu_sp = submenus.front(); 204344d93782SGreg Clayton result = eKeyHandled; 204444d93782SGreg Clayton } 204544d93782SGreg Clayton break; 204644d93782SGreg Clayton 204744d93782SGreg Clayton case KEY_LEFT: 204844d93782SGreg Clayton { 204944d93782SGreg Clayton --m_selected; 205044d93782SGreg Clayton if (m_selected < 0) 205144d93782SGreg Clayton m_selected = num_submenus - 1; 20523985c8c6SSaleem Abdulrasool if (m_selected < static_cast<int>(num_submenus)) 205344d93782SGreg Clayton run_menu_sp = submenus[m_selected]; 205444d93782SGreg Clayton else if (!submenus.empty()) 205544d93782SGreg Clayton run_menu_sp = submenus.front(); 205644d93782SGreg Clayton result = eKeyHandled; 205744d93782SGreg Clayton } 205844d93782SGreg Clayton break; 205944d93782SGreg Clayton 206044d93782SGreg Clayton default: 206144d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 206244d93782SGreg Clayton { 206344d93782SGreg Clayton if (submenus[i]->GetKeyValue() == key) 206444d93782SGreg Clayton { 206544d93782SGreg Clayton SetSelectedSubmenuIndex(i); 206644d93782SGreg Clayton run_menu_sp = submenus[i]; 206744d93782SGreg Clayton result = eKeyHandled; 206844d93782SGreg Clayton break; 206944d93782SGreg Clayton } 207044d93782SGreg Clayton } 207144d93782SGreg Clayton break; 207244d93782SGreg Clayton } 207344d93782SGreg Clayton 207444d93782SGreg Clayton if (run_menu_sp) 207544d93782SGreg Clayton { 207644d93782SGreg Clayton // Run the action on this menu in case we need to populate the 207744d93782SGreg Clayton // menu with dynamic content and also in case check marks, and 207844d93782SGreg Clayton // any other menu decorations need to be caclulated 207944d93782SGreg Clayton if (run_menu_sp->Action() == MenuActionResult::Quit) 208044d93782SGreg Clayton return eQuitApplication; 208144d93782SGreg Clayton 208244d93782SGreg Clayton Rect menu_bounds; 208344d93782SGreg Clayton menu_bounds.origin.x = run_menu_sp->GetStartingColumn(); 208444d93782SGreg Clayton menu_bounds.origin.y = 1; 208544d93782SGreg Clayton menu_bounds.size.width = run_menu_sp->GetDrawWidth(); 208644d93782SGreg Clayton menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2; 208744d93782SGreg Clayton if (m_menu_window_sp) 208844d93782SGreg Clayton window.GetParent()->RemoveSubWindow(m_menu_window_sp.get()); 208944d93782SGreg Clayton 209044d93782SGreg Clayton m_menu_window_sp = window.GetParent()->CreateSubWindow (run_menu_sp->GetName().c_str(), 209144d93782SGreg Clayton menu_bounds, 209244d93782SGreg Clayton true); 209344d93782SGreg Clayton m_menu_window_sp->SetDelegate (run_menu_sp); 209444d93782SGreg Clayton } 209544d93782SGreg Clayton } 209644d93782SGreg Clayton else if (menu_type == Menu::Type::Item) 209744d93782SGreg Clayton { 209844d93782SGreg Clayton switch (key) 209944d93782SGreg Clayton { 210044d93782SGreg Clayton case KEY_DOWN: 210144d93782SGreg Clayton if (m_submenus.size() > 1) 210244d93782SGreg Clayton { 210344d93782SGreg Clayton const int start_select = m_selected; 210444d93782SGreg Clayton while (++m_selected != start_select) 210544d93782SGreg Clayton { 21063985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_selected) >= num_submenus) 210744d93782SGreg Clayton m_selected = 0; 210844d93782SGreg Clayton if (m_submenus[m_selected]->GetType() == Type::Separator) 210944d93782SGreg Clayton continue; 211044d93782SGreg Clayton else 211144d93782SGreg Clayton break; 211244d93782SGreg Clayton } 211344d93782SGreg Clayton return eKeyHandled; 211444d93782SGreg Clayton } 211544d93782SGreg Clayton break; 211644d93782SGreg Clayton 211744d93782SGreg Clayton case KEY_UP: 211844d93782SGreg Clayton if (m_submenus.size() > 1) 211944d93782SGreg Clayton { 212044d93782SGreg Clayton const int start_select = m_selected; 212144d93782SGreg Clayton while (--m_selected != start_select) 212244d93782SGreg Clayton { 21233985c8c6SSaleem Abdulrasool if (m_selected < static_cast<int>(0)) 212444d93782SGreg Clayton m_selected = num_submenus - 1; 212544d93782SGreg Clayton if (m_submenus[m_selected]->GetType() == Type::Separator) 212644d93782SGreg Clayton continue; 212744d93782SGreg Clayton else 212844d93782SGreg Clayton break; 212944d93782SGreg Clayton } 213044d93782SGreg Clayton return eKeyHandled; 213144d93782SGreg Clayton } 213244d93782SGreg Clayton break; 213344d93782SGreg Clayton 213444d93782SGreg Clayton case KEY_RETURN: 21353985c8c6SSaleem Abdulrasool if (static_cast<size_t>(selected_idx) < num_submenus) 213644d93782SGreg Clayton { 213744d93782SGreg Clayton if (submenus[selected_idx]->Action() == MenuActionResult::Quit) 213844d93782SGreg Clayton return eQuitApplication; 213944d93782SGreg Clayton window.GetParent()->RemoveSubWindow(&window); 214044d93782SGreg Clayton return eKeyHandled; 214144d93782SGreg Clayton } 214244d93782SGreg Clayton break; 214344d93782SGreg Clayton 214444d93782SGreg Clayton case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in case other chars are entered for escaped sequences 214544d93782SGreg Clayton window.GetParent()->RemoveSubWindow(&window); 214644d93782SGreg Clayton return eKeyHandled; 214744d93782SGreg Clayton 214844d93782SGreg Clayton default: 214944d93782SGreg Clayton { 215044d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 215144d93782SGreg Clayton { 215244d93782SGreg Clayton Menu *menu = submenus[i].get(); 215344d93782SGreg Clayton if (menu->GetKeyValue() == key) 215444d93782SGreg Clayton { 215544d93782SGreg Clayton SetSelectedSubmenuIndex(i); 215644d93782SGreg Clayton window.GetParent()->RemoveSubWindow(&window); 215744d93782SGreg Clayton if (menu->Action() == MenuActionResult::Quit) 215844d93782SGreg Clayton return eQuitApplication; 215944d93782SGreg Clayton return eKeyHandled; 216044d93782SGreg Clayton } 216144d93782SGreg Clayton } 216244d93782SGreg Clayton } 216344d93782SGreg Clayton break; 216444d93782SGreg Clayton 216544d93782SGreg Clayton } 216644d93782SGreg Clayton } 216744d93782SGreg Clayton else if (menu_type == Menu::Type::Separator) 216844d93782SGreg Clayton { 216944d93782SGreg Clayton 217044d93782SGreg Clayton } 217144d93782SGreg Clayton return result; 217244d93782SGreg Clayton } 217344d93782SGreg Clayton 217444d93782SGreg Clayton 217544d93782SGreg Clayton class Application 217644d93782SGreg Clayton { 217744d93782SGreg Clayton public: 217844d93782SGreg Clayton Application (FILE *in, FILE *out) : 217944d93782SGreg Clayton m_window_sp(), 218044d93782SGreg Clayton m_screen (NULL), 218144d93782SGreg Clayton m_in (in), 218244d93782SGreg Clayton m_out (out) 218344d93782SGreg Clayton { 218444d93782SGreg Clayton 218544d93782SGreg Clayton } 218644d93782SGreg Clayton 218744d93782SGreg Clayton ~Application () 218844d93782SGreg Clayton { 218944d93782SGreg Clayton m_window_delegates.clear(); 219044d93782SGreg Clayton m_window_sp.reset(); 219144d93782SGreg Clayton if (m_screen) 219244d93782SGreg Clayton { 219344d93782SGreg Clayton ::delscreen(m_screen); 219444d93782SGreg Clayton m_screen = NULL; 219544d93782SGreg Clayton } 219644d93782SGreg Clayton } 219744d93782SGreg Clayton 219844d93782SGreg Clayton void 219944d93782SGreg Clayton Initialize () 220044d93782SGreg Clayton { 220144d93782SGreg Clayton ::setlocale(LC_ALL, ""); 220244d93782SGreg Clayton ::setlocale(LC_CTYPE, ""); 220344d93782SGreg Clayton #if 0 220444d93782SGreg Clayton ::initscr(); 220544d93782SGreg Clayton #else 220644d93782SGreg Clayton m_screen = ::newterm(NULL, m_out, m_in); 220744d93782SGreg Clayton #endif 220844d93782SGreg Clayton ::start_color(); 220944d93782SGreg Clayton ::curs_set(0); 221044d93782SGreg Clayton ::noecho(); 221144d93782SGreg Clayton ::keypad(stdscr,TRUE); 221244d93782SGreg Clayton } 221344d93782SGreg Clayton 221444d93782SGreg Clayton void 221544d93782SGreg Clayton Terminate () 221644d93782SGreg Clayton { 221744d93782SGreg Clayton ::endwin(); 221844d93782SGreg Clayton } 221944d93782SGreg Clayton 222044d93782SGreg Clayton void 222144d93782SGreg Clayton Run (Debugger &debugger) 222244d93782SGreg Clayton { 222344d93782SGreg Clayton bool done = false; 222444d93782SGreg Clayton int delay_in_tenths_of_a_second = 1; 222544d93782SGreg Clayton 222644d93782SGreg Clayton // Alas the threading model in curses is a bit lame so we need to 222744d93782SGreg Clayton // resort to polling every 0.5 seconds. We could poll for stdin 222844d93782SGreg Clayton // ourselves and then pass the keys down but then we need to 222944d93782SGreg Clayton // translate all of the escape sequences ourselves. So we resort to 223044d93782SGreg Clayton // polling for input because we need to receive async process events 223144d93782SGreg Clayton // while in this loop. 223244d93782SGreg Clayton 223344d93782SGreg Clayton halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths of seconds seconds when calling Window::GetChar() 223444d93782SGreg Clayton 223544d93782SGreg Clayton ListenerSP listener_sp (new Listener ("lldb.IOHandler.curses.Application")); 223644d93782SGreg Clayton ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass()); 223744d93782SGreg Clayton ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass()); 223844d93782SGreg Clayton ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass()); 223944d93782SGreg Clayton debugger.EnableForwardEvents (listener_sp); 224044d93782SGreg Clayton 224144d93782SGreg Clayton bool update = true; 224244d93782SGreg Clayton #if defined(__APPLE__) 224344d93782SGreg Clayton std::deque<int> escape_chars; 224444d93782SGreg Clayton #endif 224544d93782SGreg Clayton 224644d93782SGreg Clayton while (!done) 224744d93782SGreg Clayton { 224844d93782SGreg Clayton if (update) 224944d93782SGreg Clayton { 225044d93782SGreg Clayton m_window_sp->Draw(false); 225144d93782SGreg Clayton // All windows should be calling Window::DeferredRefresh() instead 225244d93782SGreg Clayton // of Window::Refresh() so we can do a single update and avoid 225344d93782SGreg Clayton // any screen blinking 225444d93782SGreg Clayton update_panels(); 225544d93782SGreg Clayton 225644d93782SGreg Clayton // Cursor hiding isn't working on MacOSX, so hide it in the top left corner 225744d93782SGreg Clayton m_window_sp->MoveCursor(0, 0); 225844d93782SGreg Clayton 225944d93782SGreg Clayton doupdate(); 226044d93782SGreg Clayton update = false; 226144d93782SGreg Clayton } 226244d93782SGreg Clayton 226344d93782SGreg Clayton #if defined(__APPLE__) 226444d93782SGreg Clayton // Terminal.app doesn't map its function keys correctly, F1-F4 default to: 226544d93782SGreg Clayton // \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if possible 226644d93782SGreg Clayton int ch; 226744d93782SGreg Clayton if (escape_chars.empty()) 226844d93782SGreg Clayton ch = m_window_sp->GetChar(); 226944d93782SGreg Clayton else 227044d93782SGreg Clayton { 227144d93782SGreg Clayton ch = escape_chars.front(); 227244d93782SGreg Clayton escape_chars.pop_front(); 227344d93782SGreg Clayton } 227444d93782SGreg Clayton if (ch == KEY_ESCAPE) 227544d93782SGreg Clayton { 227644d93782SGreg Clayton int ch2 = m_window_sp->GetChar(); 227744d93782SGreg Clayton if (ch2 == 'O') 227844d93782SGreg Clayton { 227944d93782SGreg Clayton int ch3 = m_window_sp->GetChar(); 228044d93782SGreg Clayton switch (ch3) 228144d93782SGreg Clayton { 228244d93782SGreg Clayton case 'P': ch = KEY_F(1); break; 228344d93782SGreg Clayton case 'Q': ch = KEY_F(2); break; 228444d93782SGreg Clayton case 'R': ch = KEY_F(3); break; 228544d93782SGreg Clayton case 'S': ch = KEY_F(4); break; 228644d93782SGreg Clayton default: 228744d93782SGreg Clayton escape_chars.push_back(ch2); 228844d93782SGreg Clayton if (ch3 != -1) 228944d93782SGreg Clayton escape_chars.push_back(ch3); 229044d93782SGreg Clayton break; 229144d93782SGreg Clayton } 229244d93782SGreg Clayton } 229344d93782SGreg Clayton else if (ch2 != -1) 229444d93782SGreg Clayton escape_chars.push_back(ch2); 229544d93782SGreg Clayton } 229644d93782SGreg Clayton #else 229744d93782SGreg Clayton int ch = m_window_sp->GetChar(); 229844d93782SGreg Clayton 229944d93782SGreg Clayton #endif 230044d93782SGreg Clayton if (ch == -1) 230144d93782SGreg Clayton { 230244d93782SGreg Clayton if (feof(m_in) || ferror(m_in)) 230344d93782SGreg Clayton { 230444d93782SGreg Clayton done = true; 230544d93782SGreg Clayton } 230644d93782SGreg Clayton else 230744d93782SGreg Clayton { 230844d93782SGreg Clayton // Just a timeout from using halfdelay(), check for events 230944d93782SGreg Clayton EventSP event_sp; 231044d93782SGreg Clayton while (listener_sp->PeekAtNextEvent()) 231144d93782SGreg Clayton { 231244d93782SGreg Clayton listener_sp->GetNextEvent(event_sp); 231344d93782SGreg Clayton 231444d93782SGreg Clayton if (event_sp) 231544d93782SGreg Clayton { 231644d93782SGreg Clayton Broadcaster *broadcaster = event_sp->GetBroadcaster(); 231744d93782SGreg Clayton if (broadcaster) 231844d93782SGreg Clayton { 231944d93782SGreg Clayton //uint32_t event_type = event_sp->GetType(); 232044d93782SGreg Clayton ConstString broadcaster_class (broadcaster->GetBroadcasterClass()); 232144d93782SGreg Clayton if (broadcaster_class == broadcaster_class_process) 232244d93782SGreg Clayton { 2323ec990867SGreg Clayton debugger.GetCommandInterpreter().UpdateExecutionContext(NULL); 232444d93782SGreg Clayton update = true; 232544d93782SGreg Clayton continue; // Don't get any key, just update our view 232644d93782SGreg Clayton } 232744d93782SGreg Clayton } 232844d93782SGreg Clayton } 232944d93782SGreg Clayton } 233044d93782SGreg Clayton } 233144d93782SGreg Clayton } 233244d93782SGreg Clayton else 233344d93782SGreg Clayton { 233444d93782SGreg Clayton HandleCharResult key_result = m_window_sp->HandleChar(ch); 233544d93782SGreg Clayton switch (key_result) 233644d93782SGreg Clayton { 233744d93782SGreg Clayton case eKeyHandled: 2338ec990867SGreg Clayton debugger.GetCommandInterpreter().UpdateExecutionContext(NULL); 233944d93782SGreg Clayton update = true; 234044d93782SGreg Clayton break; 234144d93782SGreg Clayton case eKeyNotHandled: 234244d93782SGreg Clayton break; 234344d93782SGreg Clayton case eQuitApplication: 234444d93782SGreg Clayton done = true; 234544d93782SGreg Clayton break; 234644d93782SGreg Clayton } 234744d93782SGreg Clayton } 234844d93782SGreg Clayton } 234944d93782SGreg Clayton 235044d93782SGreg Clayton debugger.CancelForwardEvents (listener_sp); 235144d93782SGreg Clayton 235244d93782SGreg Clayton } 235344d93782SGreg Clayton 235444d93782SGreg Clayton WindowSP & 235544d93782SGreg Clayton GetMainWindow () 235644d93782SGreg Clayton { 235744d93782SGreg Clayton if (!m_window_sp) 235844d93782SGreg Clayton m_window_sp.reset (new Window ("main", stdscr, false)); 235944d93782SGreg Clayton return m_window_sp; 236044d93782SGreg Clayton } 236144d93782SGreg Clayton 236244d93782SGreg Clayton WindowDelegates & 236344d93782SGreg Clayton GetWindowDelegates () 236444d93782SGreg Clayton { 236544d93782SGreg Clayton return m_window_delegates; 236644d93782SGreg Clayton } 236744d93782SGreg Clayton 236844d93782SGreg Clayton protected: 236944d93782SGreg Clayton WindowSP m_window_sp; 237044d93782SGreg Clayton WindowDelegates m_window_delegates; 237144d93782SGreg Clayton SCREEN *m_screen; 237244d93782SGreg Clayton FILE *m_in; 237344d93782SGreg Clayton FILE *m_out; 237444d93782SGreg Clayton }; 237544d93782SGreg Clayton 237644d93782SGreg Clayton 237744d93782SGreg Clayton } // namespace curses 237844d93782SGreg Clayton 237944d93782SGreg Clayton 238044d93782SGreg Clayton using namespace curses; 238144d93782SGreg Clayton 238244d93782SGreg Clayton struct Row 238344d93782SGreg Clayton { 238444d93782SGreg Clayton ValueObjectSP valobj; 238544d93782SGreg Clayton Row *parent; 238644d93782SGreg Clayton int row_idx; 238744d93782SGreg Clayton int x; 238844d93782SGreg Clayton int y; 238944d93782SGreg Clayton bool might_have_children; 239044d93782SGreg Clayton bool expanded; 239144d93782SGreg Clayton bool calculated_children; 239244d93782SGreg Clayton std::vector<Row> children; 239344d93782SGreg Clayton 239444d93782SGreg Clayton Row (const ValueObjectSP &v, Row *p) : 239544d93782SGreg Clayton valobj (v), 239644d93782SGreg Clayton parent (p), 239744d93782SGreg Clayton row_idx(0), 239844d93782SGreg Clayton x(1), 239944d93782SGreg Clayton y(1), 240044d93782SGreg Clayton might_have_children (v ? v->MightHaveChildren() : false), 240144d93782SGreg Clayton expanded (false), 240244d93782SGreg Clayton calculated_children (false), 240344d93782SGreg Clayton children() 240444d93782SGreg Clayton { 240544d93782SGreg Clayton } 240644d93782SGreg Clayton 240744d93782SGreg Clayton size_t 240844d93782SGreg Clayton GetDepth () const 240944d93782SGreg Clayton { 241044d93782SGreg Clayton if (parent) 241144d93782SGreg Clayton return 1 + parent->GetDepth(); 241244d93782SGreg Clayton return 0; 241344d93782SGreg Clayton } 241444d93782SGreg Clayton 241544d93782SGreg Clayton void 241644d93782SGreg Clayton Expand() 241744d93782SGreg Clayton { 241844d93782SGreg Clayton expanded = true; 241944d93782SGreg Clayton if (!calculated_children) 242044d93782SGreg Clayton { 242144d93782SGreg Clayton calculated_children = true; 242244d93782SGreg Clayton if (valobj) 242344d93782SGreg Clayton { 242444d93782SGreg Clayton const size_t num_children = valobj->GetNumChildren(); 242544d93782SGreg Clayton for (size_t i=0; i<num_children; ++i) 242644d93782SGreg Clayton { 242744d93782SGreg Clayton children.push_back(Row (valobj->GetChildAtIndex(i, true), this)); 242844d93782SGreg Clayton } 242944d93782SGreg Clayton } 243044d93782SGreg Clayton } 243144d93782SGreg Clayton } 243244d93782SGreg Clayton 243344d93782SGreg Clayton void 243444d93782SGreg Clayton Unexpand () 243544d93782SGreg Clayton { 243644d93782SGreg Clayton expanded = false; 243744d93782SGreg Clayton } 243844d93782SGreg Clayton 243944d93782SGreg Clayton void 244044d93782SGreg Clayton DrawTree (Window &window) 244144d93782SGreg Clayton { 244244d93782SGreg Clayton if (parent) 244344d93782SGreg Clayton parent->DrawTreeForChild (window, this, 0); 244444d93782SGreg Clayton 244544d93782SGreg Clayton if (might_have_children) 244644d93782SGreg Clayton { 244744d93782SGreg Clayton // It we can get UTF8 characters to work we should try to use the "symbol" 244844d93782SGreg Clayton // UTF8 string below 244944d93782SGreg Clayton // const char *symbol = ""; 245044d93782SGreg Clayton // if (row.expanded) 245144d93782SGreg Clayton // symbol = "\xe2\x96\xbd "; 245244d93782SGreg Clayton // else 245344d93782SGreg Clayton // symbol = "\xe2\x96\xb7 "; 245444d93782SGreg Clayton // window.PutCString (symbol); 245544d93782SGreg Clayton 245644d93782SGreg Clayton // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 245744d93782SGreg Clayton // 'v' or '>' character... 245844d93782SGreg Clayton // if (expanded) 245944d93782SGreg Clayton // window.PutChar (ACS_DARROW); 246044d93782SGreg Clayton // else 246144d93782SGreg Clayton // window.PutChar (ACS_RARROW); 246244d93782SGreg Clayton // Since we can't find any good looking right arrow/down arrow 246344d93782SGreg Clayton // symbols, just use a diamond... 246444d93782SGreg Clayton window.PutChar (ACS_DIAMOND); 246544d93782SGreg Clayton window.PutChar (ACS_HLINE); 246644d93782SGreg Clayton } 246744d93782SGreg Clayton } 246844d93782SGreg Clayton 246944d93782SGreg Clayton void 247044d93782SGreg Clayton DrawTreeForChild (Window &window, Row *child, uint32_t reverse_depth) 247144d93782SGreg Clayton { 247244d93782SGreg Clayton if (parent) 247344d93782SGreg Clayton parent->DrawTreeForChild (window, this, reverse_depth + 1); 247444d93782SGreg Clayton 247544d93782SGreg Clayton if (&children.back() == child) 247644d93782SGreg Clayton { 247744d93782SGreg Clayton // Last child 247844d93782SGreg Clayton if (reverse_depth == 0) 247944d93782SGreg Clayton { 248044d93782SGreg Clayton window.PutChar (ACS_LLCORNER); 248144d93782SGreg Clayton window.PutChar (ACS_HLINE); 248244d93782SGreg Clayton } 248344d93782SGreg Clayton else 248444d93782SGreg Clayton { 248544d93782SGreg Clayton window.PutChar (' '); 248644d93782SGreg Clayton window.PutChar (' '); 248744d93782SGreg Clayton } 248844d93782SGreg Clayton } 248944d93782SGreg Clayton else 249044d93782SGreg Clayton { 249144d93782SGreg Clayton if (reverse_depth == 0) 249244d93782SGreg Clayton { 249344d93782SGreg Clayton window.PutChar (ACS_LTEE); 249444d93782SGreg Clayton window.PutChar (ACS_HLINE); 249544d93782SGreg Clayton } 249644d93782SGreg Clayton else 249744d93782SGreg Clayton { 249844d93782SGreg Clayton window.PutChar (ACS_VLINE); 249944d93782SGreg Clayton window.PutChar (' '); 250044d93782SGreg Clayton } 250144d93782SGreg Clayton } 250244d93782SGreg Clayton } 250344d93782SGreg Clayton }; 250444d93782SGreg Clayton 250544d93782SGreg Clayton struct DisplayOptions 250644d93782SGreg Clayton { 250744d93782SGreg Clayton bool show_types; 250844d93782SGreg Clayton }; 250944d93782SGreg Clayton 251044d93782SGreg Clayton class TreeItem; 251144d93782SGreg Clayton 251244d93782SGreg Clayton class TreeDelegate 251344d93782SGreg Clayton { 251444d93782SGreg Clayton public: 251544d93782SGreg Clayton TreeDelegate() {} 251644d93782SGreg Clayton virtual ~TreeDelegate() {} 251744d93782SGreg Clayton virtual void TreeDelegateDrawTreeItem (TreeItem &item, Window &window) = 0; 251844d93782SGreg Clayton virtual void TreeDelegateGenerateChildren (TreeItem &item) = 0; 251944d93782SGreg Clayton virtual bool TreeDelegateItemSelected (TreeItem &item) = 0; // Return true if we need to update views 252044d93782SGreg Clayton }; 252144d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP; 252244d93782SGreg Clayton 252344d93782SGreg Clayton class TreeItem 252444d93782SGreg Clayton { 252544d93782SGreg Clayton public: 252644d93782SGreg Clayton 252744d93782SGreg Clayton TreeItem (TreeItem *parent, TreeDelegate &delegate, bool might_have_children) : 252844d93782SGreg Clayton m_parent (parent), 252944d93782SGreg Clayton m_delegate (delegate), 2530ec990867SGreg Clayton m_user_data (NULL), 253144d93782SGreg Clayton m_identifier (0), 253244d93782SGreg Clayton m_row_idx (-1), 253344d93782SGreg Clayton m_children (), 253444d93782SGreg Clayton m_might_have_children (might_have_children), 253544d93782SGreg Clayton m_is_expanded (false) 253644d93782SGreg Clayton { 253744d93782SGreg Clayton } 253844d93782SGreg Clayton 253944d93782SGreg Clayton TreeItem & 254044d93782SGreg Clayton operator=(const TreeItem &rhs) 254144d93782SGreg Clayton { 254244d93782SGreg Clayton if (this != &rhs) 254344d93782SGreg Clayton { 254444d93782SGreg Clayton m_parent = rhs.m_parent; 254544d93782SGreg Clayton m_delegate = rhs.m_delegate; 2546ec990867SGreg Clayton m_user_data = rhs.m_user_data; 254744d93782SGreg Clayton m_identifier = rhs.m_identifier; 254844d93782SGreg Clayton m_row_idx = rhs.m_row_idx; 254944d93782SGreg Clayton m_children = rhs.m_children; 255044d93782SGreg Clayton m_might_have_children = rhs.m_might_have_children; 255144d93782SGreg Clayton m_is_expanded = rhs.m_is_expanded; 255244d93782SGreg Clayton } 255344d93782SGreg Clayton return *this; 255444d93782SGreg Clayton } 255544d93782SGreg Clayton 255644d93782SGreg Clayton size_t 255744d93782SGreg Clayton GetDepth () const 255844d93782SGreg Clayton { 255944d93782SGreg Clayton if (m_parent) 256044d93782SGreg Clayton return 1 + m_parent->GetDepth(); 256144d93782SGreg Clayton return 0; 256244d93782SGreg Clayton } 256344d93782SGreg Clayton 256444d93782SGreg Clayton int 256544d93782SGreg Clayton GetRowIndex () const 256644d93782SGreg Clayton { 256744d93782SGreg Clayton return m_row_idx; 256844d93782SGreg Clayton } 256944d93782SGreg Clayton 257044d93782SGreg Clayton void 257144d93782SGreg Clayton ClearChildren () 257244d93782SGreg Clayton { 257344d93782SGreg Clayton m_children.clear(); 257444d93782SGreg Clayton } 257544d93782SGreg Clayton 257644d93782SGreg Clayton void 257744d93782SGreg Clayton Resize (size_t n, const TreeItem &t) 257844d93782SGreg Clayton { 257944d93782SGreg Clayton m_children.resize(n, t); 258044d93782SGreg Clayton } 258144d93782SGreg Clayton 258244d93782SGreg Clayton TreeItem & 258344d93782SGreg Clayton operator [](size_t i) 258444d93782SGreg Clayton { 258544d93782SGreg Clayton return m_children[i]; 258644d93782SGreg Clayton } 258744d93782SGreg Clayton 258844d93782SGreg Clayton void 258944d93782SGreg Clayton SetRowIndex (int row_idx) 259044d93782SGreg Clayton { 259144d93782SGreg Clayton m_row_idx = row_idx; 259244d93782SGreg Clayton } 259344d93782SGreg Clayton 259444d93782SGreg Clayton size_t 259544d93782SGreg Clayton GetNumChildren () 259644d93782SGreg Clayton { 259744d93782SGreg Clayton m_delegate.TreeDelegateGenerateChildren (*this); 259844d93782SGreg Clayton return m_children.size(); 259944d93782SGreg Clayton } 260044d93782SGreg Clayton 260144d93782SGreg Clayton void 260244d93782SGreg Clayton ItemWasSelected () 260344d93782SGreg Clayton { 260444d93782SGreg Clayton m_delegate.TreeDelegateItemSelected(*this); 260544d93782SGreg Clayton } 260644d93782SGreg Clayton void 260744d93782SGreg Clayton CalculateRowIndexes (int &row_idx) 260844d93782SGreg Clayton { 260944d93782SGreg Clayton SetRowIndex(row_idx); 261044d93782SGreg Clayton ++row_idx; 261144d93782SGreg Clayton 2612ec990867SGreg Clayton const bool expanded = IsExpanded(); 2613ec990867SGreg Clayton 2614ec990867SGreg Clayton // The root item must calculate its children, 2615ec990867SGreg Clayton // or we must calculate the number of children 2616ec990867SGreg Clayton // if the item is expanded 2617ec990867SGreg Clayton if (m_parent == NULL || expanded) 261844d93782SGreg Clayton GetNumChildren(); 261944d93782SGreg Clayton 262044d93782SGreg Clayton for (auto &item : m_children) 262144d93782SGreg Clayton { 262244d93782SGreg Clayton if (expanded) 262344d93782SGreg Clayton item.CalculateRowIndexes(row_idx); 262444d93782SGreg Clayton else 262544d93782SGreg Clayton item.SetRowIndex(-1); 262644d93782SGreg Clayton } 262744d93782SGreg Clayton } 262844d93782SGreg Clayton 262944d93782SGreg Clayton TreeItem * 263044d93782SGreg Clayton GetParent () 263144d93782SGreg Clayton { 263244d93782SGreg Clayton return m_parent; 263344d93782SGreg Clayton } 263444d93782SGreg Clayton 263544d93782SGreg Clayton bool 263644d93782SGreg Clayton IsExpanded () const 263744d93782SGreg Clayton { 263844d93782SGreg Clayton return m_is_expanded; 263944d93782SGreg Clayton } 264044d93782SGreg Clayton 264144d93782SGreg Clayton void 264244d93782SGreg Clayton Expand() 264344d93782SGreg Clayton { 264444d93782SGreg Clayton m_is_expanded = true; 264544d93782SGreg Clayton } 264644d93782SGreg Clayton 264744d93782SGreg Clayton void 264844d93782SGreg Clayton Unexpand () 264944d93782SGreg Clayton { 265044d93782SGreg Clayton m_is_expanded = false; 265144d93782SGreg Clayton } 265244d93782SGreg Clayton 265344d93782SGreg Clayton bool 265444d93782SGreg Clayton Draw (Window &window, 265544d93782SGreg Clayton const int first_visible_row, 265644d93782SGreg Clayton const uint32_t selected_row_idx, 265744d93782SGreg Clayton int &row_idx, 265844d93782SGreg Clayton int &num_rows_left) 265944d93782SGreg Clayton { 266044d93782SGreg Clayton if (num_rows_left <= 0) 266144d93782SGreg Clayton return false; 266244d93782SGreg Clayton 266344d93782SGreg Clayton if (m_row_idx >= first_visible_row) 266444d93782SGreg Clayton { 266544d93782SGreg Clayton window.MoveCursor(2, row_idx + 1); 266644d93782SGreg Clayton 266744d93782SGreg Clayton if (m_parent) 266844d93782SGreg Clayton m_parent->DrawTreeForChild (window, this, 0); 266944d93782SGreg Clayton 267044d93782SGreg Clayton if (m_might_have_children) 267144d93782SGreg Clayton { 267244d93782SGreg Clayton // It we can get UTF8 characters to work we should try to use the "symbol" 267344d93782SGreg Clayton // UTF8 string below 267444d93782SGreg Clayton // const char *symbol = ""; 267544d93782SGreg Clayton // if (row.expanded) 267644d93782SGreg Clayton // symbol = "\xe2\x96\xbd "; 267744d93782SGreg Clayton // else 267844d93782SGreg Clayton // symbol = "\xe2\x96\xb7 "; 267944d93782SGreg Clayton // window.PutCString (symbol); 268044d93782SGreg Clayton 268144d93782SGreg Clayton // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 268244d93782SGreg Clayton // 'v' or '>' character... 268344d93782SGreg Clayton // if (expanded) 268444d93782SGreg Clayton // window.PutChar (ACS_DARROW); 268544d93782SGreg Clayton // else 268644d93782SGreg Clayton // window.PutChar (ACS_RARROW); 268744d93782SGreg Clayton // Since we can't find any good looking right arrow/down arrow 268844d93782SGreg Clayton // symbols, just use a diamond... 268944d93782SGreg Clayton window.PutChar (ACS_DIAMOND); 269044d93782SGreg Clayton window.PutChar (ACS_HLINE); 269144d93782SGreg Clayton } 26923985c8c6SSaleem Abdulrasool bool highlight = 26933985c8c6SSaleem Abdulrasool (selected_row_idx == static_cast<size_t>(m_row_idx)) && window.IsActive(); 269444d93782SGreg Clayton 269544d93782SGreg Clayton if (highlight) 269644d93782SGreg Clayton window.AttributeOn(A_REVERSE); 269744d93782SGreg Clayton 269844d93782SGreg Clayton m_delegate.TreeDelegateDrawTreeItem(*this, window); 269944d93782SGreg Clayton 270044d93782SGreg Clayton if (highlight) 270144d93782SGreg Clayton window.AttributeOff(A_REVERSE); 270244d93782SGreg Clayton ++row_idx; 270344d93782SGreg Clayton --num_rows_left; 270444d93782SGreg Clayton } 270544d93782SGreg Clayton 270644d93782SGreg Clayton if (num_rows_left <= 0) 270744d93782SGreg Clayton return false; // We are done drawing... 270844d93782SGreg Clayton 270944d93782SGreg Clayton if (IsExpanded()) 271044d93782SGreg Clayton { 271144d93782SGreg Clayton for (auto &item : m_children) 271244d93782SGreg Clayton { 271344d93782SGreg Clayton // If we displayed all the rows and item.Draw() returns 271444d93782SGreg Clayton // false we are done drawing and can exit this for loop 271544d93782SGreg Clayton if (item.Draw(window, first_visible_row, selected_row_idx, row_idx, num_rows_left) == false) 271644d93782SGreg Clayton break; 271744d93782SGreg Clayton } 271844d93782SGreg Clayton } 271944d93782SGreg Clayton return num_rows_left >= 0; // Return true if not done drawing yet 272044d93782SGreg Clayton } 272144d93782SGreg Clayton 272244d93782SGreg Clayton void 272344d93782SGreg Clayton DrawTreeForChild (Window &window, TreeItem *child, uint32_t reverse_depth) 272444d93782SGreg Clayton { 272544d93782SGreg Clayton if (m_parent) 272644d93782SGreg Clayton m_parent->DrawTreeForChild (window, this, reverse_depth + 1); 272744d93782SGreg Clayton 272844d93782SGreg Clayton if (&m_children.back() == child) 272944d93782SGreg Clayton { 273044d93782SGreg Clayton // Last child 273144d93782SGreg Clayton if (reverse_depth == 0) 273244d93782SGreg Clayton { 273344d93782SGreg Clayton window.PutChar (ACS_LLCORNER); 273444d93782SGreg Clayton window.PutChar (ACS_HLINE); 273544d93782SGreg Clayton } 273644d93782SGreg Clayton else 273744d93782SGreg Clayton { 273844d93782SGreg Clayton window.PutChar (' '); 273944d93782SGreg Clayton window.PutChar (' '); 274044d93782SGreg Clayton } 274144d93782SGreg Clayton } 274244d93782SGreg Clayton else 274344d93782SGreg Clayton { 274444d93782SGreg Clayton if (reverse_depth == 0) 274544d93782SGreg Clayton { 274644d93782SGreg Clayton window.PutChar (ACS_LTEE); 274744d93782SGreg Clayton window.PutChar (ACS_HLINE); 274844d93782SGreg Clayton } 274944d93782SGreg Clayton else 275044d93782SGreg Clayton { 275144d93782SGreg Clayton window.PutChar (ACS_VLINE); 275244d93782SGreg Clayton window.PutChar (' '); 275344d93782SGreg Clayton } 275444d93782SGreg Clayton } 275544d93782SGreg Clayton } 275644d93782SGreg Clayton 275744d93782SGreg Clayton TreeItem * 275844d93782SGreg Clayton GetItemForRowIndex (uint32_t row_idx) 275944d93782SGreg Clayton { 27603985c8c6SSaleem Abdulrasool if (static_cast<uint32_t>(m_row_idx) == row_idx) 276144d93782SGreg Clayton return this; 276244d93782SGreg Clayton if (m_children.empty()) 276344d93782SGreg Clayton return NULL; 27643985c8c6SSaleem Abdulrasool if (static_cast<uint32_t>(m_children.back().m_row_idx) < row_idx) 276544d93782SGreg Clayton return NULL; 276644d93782SGreg Clayton if (IsExpanded()) 276744d93782SGreg Clayton { 276844d93782SGreg Clayton for (auto &item : m_children) 276944d93782SGreg Clayton { 277044d93782SGreg Clayton TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx); 277144d93782SGreg Clayton if (selected_item_ptr) 277244d93782SGreg Clayton return selected_item_ptr; 277344d93782SGreg Clayton } 277444d93782SGreg Clayton } 277544d93782SGreg Clayton return NULL; 277644d93782SGreg Clayton } 277744d93782SGreg Clayton 2778ec990867SGreg Clayton void * 2779ec990867SGreg Clayton GetUserData() const 2780ec990867SGreg Clayton { 2781ec990867SGreg Clayton return m_user_data; 2782ec990867SGreg Clayton } 2783ec990867SGreg Clayton 2784ec990867SGreg Clayton void 2785ec990867SGreg Clayton SetUserData (void *user_data) 2786ec990867SGreg Clayton { 2787ec990867SGreg Clayton m_user_data = user_data; 2788ec990867SGreg Clayton } 2789ec990867SGreg Clayton 279044d93782SGreg Clayton uint64_t 279144d93782SGreg Clayton GetIdentifier() const 279244d93782SGreg Clayton { 279344d93782SGreg Clayton return m_identifier; 279444d93782SGreg Clayton } 279544d93782SGreg Clayton 279644d93782SGreg Clayton void 279744d93782SGreg Clayton SetIdentifier (uint64_t identifier) 279844d93782SGreg Clayton { 279944d93782SGreg Clayton m_identifier = identifier; 280044d93782SGreg Clayton } 280144d93782SGreg Clayton 280244d93782SGreg Clayton 2803ec990867SGreg Clayton void 2804ec990867SGreg Clayton SetMightHaveChildren (bool b) 2805ec990867SGreg Clayton { 2806ec990867SGreg Clayton m_might_have_children = b; 2807ec990867SGreg Clayton } 2808ec990867SGreg Clayton 280944d93782SGreg Clayton protected: 281044d93782SGreg Clayton TreeItem *m_parent; 281144d93782SGreg Clayton TreeDelegate &m_delegate; 2812ec990867SGreg Clayton void *m_user_data; 281344d93782SGreg Clayton uint64_t m_identifier; 281444d93782SGreg Clayton int m_row_idx; // Zero based visible row index, -1 if not visible or for the root item 281544d93782SGreg Clayton std::vector<TreeItem> m_children; 281644d93782SGreg Clayton bool m_might_have_children; 281744d93782SGreg Clayton bool m_is_expanded; 281844d93782SGreg Clayton 281944d93782SGreg Clayton }; 282044d93782SGreg Clayton 282144d93782SGreg Clayton class TreeWindowDelegate : public WindowDelegate 282244d93782SGreg Clayton { 282344d93782SGreg Clayton public: 282444d93782SGreg Clayton TreeWindowDelegate (Debugger &debugger, const TreeDelegateSP &delegate_sp) : 282544d93782SGreg Clayton m_debugger (debugger), 282644d93782SGreg Clayton m_delegate_sp (delegate_sp), 282744d93782SGreg Clayton m_root (NULL, *delegate_sp, true), 282844d93782SGreg Clayton m_selected_item (NULL), 282944d93782SGreg Clayton m_num_rows (0), 283044d93782SGreg Clayton m_selected_row_idx (0), 283144d93782SGreg Clayton m_first_visible_row (0), 283244d93782SGreg Clayton m_min_x (0), 283344d93782SGreg Clayton m_min_y (0), 283444d93782SGreg Clayton m_max_x (0), 283544d93782SGreg Clayton m_max_y (0) 283644d93782SGreg Clayton { 283744d93782SGreg Clayton } 283844d93782SGreg Clayton 283944d93782SGreg Clayton int 284044d93782SGreg Clayton NumVisibleRows () const 284144d93782SGreg Clayton { 284244d93782SGreg Clayton return m_max_y - m_min_y; 284344d93782SGreg Clayton } 284444d93782SGreg Clayton 284544d93782SGreg Clayton virtual bool 284644d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 284744d93782SGreg Clayton { 284844d93782SGreg Clayton ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext()); 284944d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 285044d93782SGreg Clayton 285144d93782SGreg Clayton bool display_content = false; 285244d93782SGreg Clayton if (process) 285344d93782SGreg Clayton { 285444d93782SGreg Clayton StateType state = process->GetState(); 285544d93782SGreg Clayton if (StateIsStoppedState(state, true)) 285644d93782SGreg Clayton { 285744d93782SGreg Clayton // We are stopped, so it is ok to 285844d93782SGreg Clayton display_content = true; 285944d93782SGreg Clayton } 286044d93782SGreg Clayton else if (StateIsRunningState(state)) 286144d93782SGreg Clayton { 286244d93782SGreg Clayton return true; // Don't do any updating when we are running 286344d93782SGreg Clayton } 286444d93782SGreg Clayton } 286544d93782SGreg Clayton 286644d93782SGreg Clayton m_min_x = 2; 286744d93782SGreg Clayton m_min_y = 1; 286844d93782SGreg Clayton m_max_x = window.GetWidth() - 1; 286944d93782SGreg Clayton m_max_y = window.GetHeight() - 1; 287044d93782SGreg Clayton 287144d93782SGreg Clayton window.Erase(); 287244d93782SGreg Clayton window.DrawTitleBox (window.GetName()); 287344d93782SGreg Clayton 287444d93782SGreg Clayton if (display_content) 287544d93782SGreg Clayton { 287644d93782SGreg Clayton const int num_visible_rows = NumVisibleRows(); 287744d93782SGreg Clayton m_num_rows = 0; 287844d93782SGreg Clayton m_root.CalculateRowIndexes(m_num_rows); 287944d93782SGreg Clayton 288044d93782SGreg Clayton // If we unexpanded while having something selected our 288144d93782SGreg Clayton // total number of rows is less than the num visible rows, 288244d93782SGreg Clayton // then make sure we show all the rows by setting the first 288344d93782SGreg Clayton // visible row accordingly. 288444d93782SGreg Clayton if (m_first_visible_row > 0 && m_num_rows < num_visible_rows) 288544d93782SGreg Clayton m_first_visible_row = 0; 288644d93782SGreg Clayton 288744d93782SGreg Clayton // Make sure the selected row is always visible 288844d93782SGreg Clayton if (m_selected_row_idx < m_first_visible_row) 288944d93782SGreg Clayton m_first_visible_row = m_selected_row_idx; 289044d93782SGreg Clayton else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx) 289144d93782SGreg Clayton m_first_visible_row = m_selected_row_idx - num_visible_rows + 1; 289244d93782SGreg Clayton 289344d93782SGreg Clayton int row_idx = 0; 289444d93782SGreg Clayton int num_rows_left = num_visible_rows; 289544d93782SGreg Clayton m_root.Draw (window, m_first_visible_row, m_selected_row_idx, row_idx, num_rows_left); 289644d93782SGreg Clayton // Get the selected row 289744d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex (m_selected_row_idx); 289844d93782SGreg Clayton } 289944d93782SGreg Clayton else 290044d93782SGreg Clayton { 290144d93782SGreg Clayton m_selected_item = NULL; 290244d93782SGreg Clayton } 290344d93782SGreg Clayton 290444d93782SGreg Clayton window.DeferredRefresh(); 290544d93782SGreg Clayton 290644d93782SGreg Clayton 290744d93782SGreg Clayton return true; // Drawing handled 290844d93782SGreg Clayton } 290944d93782SGreg Clayton 291044d93782SGreg Clayton 291144d93782SGreg Clayton virtual const char * 291244d93782SGreg Clayton WindowDelegateGetHelpText () 291344d93782SGreg Clayton { 291444d93782SGreg Clayton return "Thread window keyboard shortcuts:"; 291544d93782SGreg Clayton } 291644d93782SGreg Clayton 291744d93782SGreg Clayton virtual KeyHelp * 291844d93782SGreg Clayton WindowDelegateGetKeyHelp () 291944d93782SGreg Clayton { 292044d93782SGreg Clayton static curses::KeyHelp g_source_view_key_help[] = { 292144d93782SGreg Clayton { KEY_UP, "Select previous item" }, 292244d93782SGreg Clayton { KEY_DOWN, "Select next item" }, 292344d93782SGreg Clayton { KEY_RIGHT, "Expand the selected item" }, 292444d93782SGreg Clayton { KEY_LEFT, "Unexpand the selected item or select parent if not expanded" }, 292544d93782SGreg Clayton { KEY_PPAGE, "Page up" }, 292644d93782SGreg Clayton { KEY_NPAGE, "Page down" }, 292744d93782SGreg Clayton { 'h', "Show help dialog" }, 292844d93782SGreg Clayton { ' ', "Toggle item expansion" }, 292944d93782SGreg Clayton { ',', "Page up" }, 293044d93782SGreg Clayton { '.', "Page down" }, 293144d93782SGreg Clayton { '\0', NULL } 293244d93782SGreg Clayton }; 293344d93782SGreg Clayton return g_source_view_key_help; 293444d93782SGreg Clayton } 293544d93782SGreg Clayton 293644d93782SGreg Clayton virtual HandleCharResult 293744d93782SGreg Clayton WindowDelegateHandleChar (Window &window, int c) 293844d93782SGreg Clayton { 293944d93782SGreg Clayton switch(c) 294044d93782SGreg Clayton { 294144d93782SGreg Clayton case ',': 294244d93782SGreg Clayton case KEY_PPAGE: 294344d93782SGreg Clayton // Page up key 294444d93782SGreg Clayton if (m_first_visible_row > 0) 294544d93782SGreg Clayton { 294644d93782SGreg Clayton if (m_first_visible_row > m_max_y) 294744d93782SGreg Clayton m_first_visible_row -= m_max_y; 294844d93782SGreg Clayton else 294944d93782SGreg Clayton m_first_visible_row = 0; 295044d93782SGreg Clayton m_selected_row_idx = m_first_visible_row; 295144d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 295244d93782SGreg Clayton if (m_selected_item) 295344d93782SGreg Clayton m_selected_item->ItemWasSelected (); 295444d93782SGreg Clayton } 295544d93782SGreg Clayton return eKeyHandled; 295644d93782SGreg Clayton 295744d93782SGreg Clayton case '.': 295844d93782SGreg Clayton case KEY_NPAGE: 295944d93782SGreg Clayton // Page down key 296044d93782SGreg Clayton if (m_num_rows > m_max_y) 296144d93782SGreg Clayton { 296244d93782SGreg Clayton if (m_first_visible_row + m_max_y < m_num_rows) 296344d93782SGreg Clayton { 296444d93782SGreg Clayton m_first_visible_row += m_max_y; 296544d93782SGreg Clayton m_selected_row_idx = m_first_visible_row; 296644d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 296744d93782SGreg Clayton if (m_selected_item) 296844d93782SGreg Clayton m_selected_item->ItemWasSelected (); 296944d93782SGreg Clayton } 297044d93782SGreg Clayton } 297144d93782SGreg Clayton return eKeyHandled; 297244d93782SGreg Clayton 297344d93782SGreg Clayton case KEY_UP: 297444d93782SGreg Clayton if (m_selected_row_idx > 0) 297544d93782SGreg Clayton { 297644d93782SGreg Clayton --m_selected_row_idx; 297744d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 297844d93782SGreg Clayton if (m_selected_item) 297944d93782SGreg Clayton m_selected_item->ItemWasSelected (); 298044d93782SGreg Clayton } 298144d93782SGreg Clayton return eKeyHandled; 298244d93782SGreg Clayton case KEY_DOWN: 298344d93782SGreg Clayton if (m_selected_row_idx + 1 < m_num_rows) 298444d93782SGreg Clayton { 298544d93782SGreg Clayton ++m_selected_row_idx; 298644d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 298744d93782SGreg Clayton if (m_selected_item) 298844d93782SGreg Clayton m_selected_item->ItemWasSelected (); 298944d93782SGreg Clayton } 299044d93782SGreg Clayton return eKeyHandled; 299144d93782SGreg Clayton 299244d93782SGreg Clayton case KEY_RIGHT: 299344d93782SGreg Clayton if (m_selected_item) 299444d93782SGreg Clayton { 299544d93782SGreg Clayton if (!m_selected_item->IsExpanded()) 299644d93782SGreg Clayton m_selected_item->Expand(); 299744d93782SGreg Clayton } 299844d93782SGreg Clayton return eKeyHandled; 299944d93782SGreg Clayton 300044d93782SGreg Clayton case KEY_LEFT: 300144d93782SGreg Clayton if (m_selected_item) 300244d93782SGreg Clayton { 300344d93782SGreg Clayton if (m_selected_item->IsExpanded()) 300444d93782SGreg Clayton m_selected_item->Unexpand(); 300544d93782SGreg Clayton else if (m_selected_item->GetParent()) 300644d93782SGreg Clayton { 300744d93782SGreg Clayton m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex(); 300844d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 300944d93782SGreg Clayton if (m_selected_item) 301044d93782SGreg Clayton m_selected_item->ItemWasSelected (); 301144d93782SGreg Clayton } 301244d93782SGreg Clayton } 301344d93782SGreg Clayton return eKeyHandled; 301444d93782SGreg Clayton 301544d93782SGreg Clayton case ' ': 301644d93782SGreg Clayton // Toggle expansion state when SPACE is pressed 301744d93782SGreg Clayton if (m_selected_item) 301844d93782SGreg Clayton { 301944d93782SGreg Clayton if (m_selected_item->IsExpanded()) 302044d93782SGreg Clayton m_selected_item->Unexpand(); 302144d93782SGreg Clayton else 302244d93782SGreg Clayton m_selected_item->Expand(); 302344d93782SGreg Clayton } 302444d93782SGreg Clayton return eKeyHandled; 302544d93782SGreg Clayton 302644d93782SGreg Clayton case 'h': 302744d93782SGreg Clayton window.CreateHelpSubwindow (); 302844d93782SGreg Clayton return eKeyHandled; 302944d93782SGreg Clayton 303044d93782SGreg Clayton default: 303144d93782SGreg Clayton break; 303244d93782SGreg Clayton } 303344d93782SGreg Clayton return eKeyNotHandled; 303444d93782SGreg Clayton } 303544d93782SGreg Clayton 303644d93782SGreg Clayton protected: 303744d93782SGreg Clayton Debugger &m_debugger; 303844d93782SGreg Clayton TreeDelegateSP m_delegate_sp; 303944d93782SGreg Clayton TreeItem m_root; 304044d93782SGreg Clayton TreeItem *m_selected_item; 304144d93782SGreg Clayton int m_num_rows; 304244d93782SGreg Clayton int m_selected_row_idx; 304344d93782SGreg Clayton int m_first_visible_row; 304444d93782SGreg Clayton int m_min_x; 304544d93782SGreg Clayton int m_min_y; 304644d93782SGreg Clayton int m_max_x; 304744d93782SGreg Clayton int m_max_y; 304844d93782SGreg Clayton 304944d93782SGreg Clayton }; 305044d93782SGreg Clayton 305144d93782SGreg Clayton class FrameTreeDelegate : public TreeDelegate 305244d93782SGreg Clayton { 305344d93782SGreg Clayton public: 3054ec990867SGreg Clayton FrameTreeDelegate () : 3055ec990867SGreg Clayton TreeDelegate() 305644d93782SGreg Clayton { 305744d93782SGreg Clayton } 305844d93782SGreg Clayton 305944d93782SGreg Clayton virtual ~FrameTreeDelegate() 306044d93782SGreg Clayton { 306144d93782SGreg Clayton } 306244d93782SGreg Clayton 306344d93782SGreg Clayton virtual void 306444d93782SGreg Clayton TreeDelegateDrawTreeItem (TreeItem &item, Window &window) 306544d93782SGreg Clayton { 3066ec990867SGreg Clayton Thread* thread = (Thread*)item.GetUserData(); 3067ec990867SGreg Clayton if (thread) 306844d93782SGreg Clayton { 306944d93782SGreg Clayton const uint64_t frame_idx = item.GetIdentifier(); 3070ec990867SGreg Clayton StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx); 307144d93782SGreg Clayton if (frame_sp) 307244d93782SGreg Clayton { 307344d93782SGreg Clayton StreamString strm; 307444d93782SGreg Clayton const SymbolContext &sc = frame_sp->GetSymbolContext(eSymbolContextEverything); 307544d93782SGreg Clayton ExecutionContext exe_ctx (frame_sp); 307644d93782SGreg Clayton //const char *frame_format = "frame #${frame.index}: ${module.file.basename}{`${function.name}${function.pc-offset}}}"; 307744d93782SGreg Clayton const char *frame_format = "frame #${frame.index}: {${function.name}${function.pc-offset}}}"; 307844d93782SGreg Clayton if (Debugger::FormatPrompt (frame_format, &sc, &exe_ctx, NULL, strm)) 307944d93782SGreg Clayton { 308044d93782SGreg Clayton int right_pad = 1; 308144d93782SGreg Clayton window.PutCStringTruncated(strm.GetString().c_str(), right_pad); 308244d93782SGreg Clayton } 308344d93782SGreg Clayton } 308444d93782SGreg Clayton } 308544d93782SGreg Clayton } 308644d93782SGreg Clayton virtual void 308744d93782SGreg Clayton TreeDelegateGenerateChildren (TreeItem &item) 308844d93782SGreg Clayton { 308944d93782SGreg Clayton // No children for frames yet... 309044d93782SGreg Clayton } 309144d93782SGreg Clayton 309244d93782SGreg Clayton virtual bool 309344d93782SGreg Clayton TreeDelegateItemSelected (TreeItem &item) 309444d93782SGreg Clayton { 3095ec990867SGreg Clayton Thread* thread = (Thread*)item.GetUserData(); 3096ec990867SGreg Clayton if (thread) 309744d93782SGreg Clayton { 3098ec990867SGreg Clayton thread->GetProcess()->GetThreadList().SetSelectedThreadByID(thread->GetID()); 309944d93782SGreg Clayton const uint64_t frame_idx = item.GetIdentifier(); 3100ec990867SGreg Clayton thread->SetSelectedFrameByIndex(frame_idx); 310144d93782SGreg Clayton return true; 310244d93782SGreg Clayton } 310344d93782SGreg Clayton return false; 310444d93782SGreg Clayton } 310544d93782SGreg Clayton }; 310644d93782SGreg Clayton 310744d93782SGreg Clayton class ThreadTreeDelegate : public TreeDelegate 310844d93782SGreg Clayton { 310944d93782SGreg Clayton public: 311044d93782SGreg Clayton ThreadTreeDelegate (Debugger &debugger) : 311144d93782SGreg Clayton TreeDelegate(), 311244d93782SGreg Clayton m_debugger (debugger), 311344d93782SGreg Clayton m_tid (LLDB_INVALID_THREAD_ID), 311444d93782SGreg Clayton m_stop_id (UINT32_MAX) 311544d93782SGreg Clayton { 311644d93782SGreg Clayton } 311744d93782SGreg Clayton 311844d93782SGreg Clayton virtual 311944d93782SGreg Clayton ~ThreadTreeDelegate() 312044d93782SGreg Clayton { 312144d93782SGreg Clayton } 312244d93782SGreg Clayton 3123ec990867SGreg Clayton ProcessSP 3124ec990867SGreg Clayton GetProcess () 3125ec990867SGreg Clayton { 3126ec990867SGreg Clayton return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP(); 3127ec990867SGreg Clayton } 3128ec990867SGreg Clayton 3129ec990867SGreg Clayton ThreadSP 3130ec990867SGreg Clayton GetThread (const TreeItem &item) 3131ec990867SGreg Clayton { 3132ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 3133ec990867SGreg Clayton if (process_sp) 3134ec990867SGreg Clayton return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier()); 3135ec990867SGreg Clayton return ThreadSP(); 3136ec990867SGreg Clayton } 3137ec990867SGreg Clayton 313844d93782SGreg Clayton virtual void 313944d93782SGreg Clayton TreeDelegateDrawTreeItem (TreeItem &item, Window &window) 314044d93782SGreg Clayton { 3141ec990867SGreg Clayton ThreadSP thread_sp = GetThread (item); 314244d93782SGreg Clayton if (thread_sp) 314344d93782SGreg Clayton { 314444d93782SGreg Clayton StreamString strm; 314544d93782SGreg Clayton ExecutionContext exe_ctx (thread_sp); 314644d93782SGreg Clayton const char *format = "thread #${thread.index}: tid = ${thread.id}{, stop reason = ${thread.stop-reason}}"; 314744d93782SGreg Clayton if (Debugger::FormatPrompt (format, NULL, &exe_ctx, NULL, strm)) 314844d93782SGreg Clayton { 314944d93782SGreg Clayton int right_pad = 1; 315044d93782SGreg Clayton window.PutCStringTruncated(strm.GetString().c_str(), right_pad); 315144d93782SGreg Clayton } 315244d93782SGreg Clayton } 315344d93782SGreg Clayton } 315444d93782SGreg Clayton virtual void 315544d93782SGreg Clayton TreeDelegateGenerateChildren (TreeItem &item) 315644d93782SGreg Clayton { 3157ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 315844d93782SGreg Clayton if (process_sp && process_sp->IsAlive()) 315944d93782SGreg Clayton { 316044d93782SGreg Clayton StateType state = process_sp->GetState(); 316144d93782SGreg Clayton if (StateIsStoppedState(state, true)) 316244d93782SGreg Clayton { 3163ec990867SGreg Clayton ThreadSP thread_sp = GetThread (item); 316444d93782SGreg Clayton if (thread_sp) 316544d93782SGreg Clayton { 316644d93782SGreg Clayton if (m_stop_id == process_sp->GetStopID() && thread_sp->GetID() == m_tid) 316744d93782SGreg Clayton return; // Children are already up to date 3168ec990867SGreg Clayton if (!m_frame_delegate_sp) 316944d93782SGreg Clayton { 317044d93782SGreg Clayton // Always expand the thread item the first time we show it 3171ec990867SGreg Clayton m_frame_delegate_sp.reset (new FrameTreeDelegate()); 317244d93782SGreg Clayton } 317344d93782SGreg Clayton 317444d93782SGreg Clayton m_stop_id = process_sp->GetStopID(); 317544d93782SGreg Clayton m_tid = thread_sp->GetID(); 317644d93782SGreg Clayton 317744d93782SGreg Clayton TreeItem t (&item, *m_frame_delegate_sp, false); 317844d93782SGreg Clayton size_t num_frames = thread_sp->GetStackFrameCount(); 317944d93782SGreg Clayton item.Resize (num_frames, t); 318044d93782SGreg Clayton for (size_t i=0; i<num_frames; ++i) 318144d93782SGreg Clayton { 3182ec990867SGreg Clayton item[i].SetUserData(thread_sp.get()); 318344d93782SGreg Clayton item[i].SetIdentifier(i); 318444d93782SGreg Clayton } 318544d93782SGreg Clayton } 318644d93782SGreg Clayton return; 318744d93782SGreg Clayton } 318844d93782SGreg Clayton } 318944d93782SGreg Clayton item.ClearChildren(); 319044d93782SGreg Clayton } 319144d93782SGreg Clayton 319244d93782SGreg Clayton virtual bool 319344d93782SGreg Clayton TreeDelegateItemSelected (TreeItem &item) 319444d93782SGreg Clayton { 3195ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 3196ec990867SGreg Clayton if (process_sp && process_sp->IsAlive()) 3197ec990867SGreg Clayton { 3198ec990867SGreg Clayton StateType state = process_sp->GetState(); 3199ec990867SGreg Clayton if (StateIsStoppedState(state, true)) 3200ec990867SGreg Clayton { 3201ec990867SGreg Clayton ThreadSP thread_sp = GetThread (item); 320244d93782SGreg Clayton if (thread_sp) 320344d93782SGreg Clayton { 320444d93782SGreg Clayton ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList(); 320544d93782SGreg Clayton Mutex::Locker locker (thread_list.GetMutex()); 320644d93782SGreg Clayton ThreadSP selected_thread_sp = thread_list.GetSelectedThread(); 320744d93782SGreg Clayton if (selected_thread_sp->GetID() != thread_sp->GetID()) 320844d93782SGreg Clayton { 320944d93782SGreg Clayton thread_list.SetSelectedThreadByID(thread_sp->GetID()); 321044d93782SGreg Clayton return true; 321144d93782SGreg Clayton } 321244d93782SGreg Clayton } 3213ec990867SGreg Clayton } 3214ec990867SGreg Clayton } 321544d93782SGreg Clayton return false; 321644d93782SGreg Clayton } 321744d93782SGreg Clayton 321844d93782SGreg Clayton protected: 321944d93782SGreg Clayton Debugger &m_debugger; 322044d93782SGreg Clayton std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp; 322144d93782SGreg Clayton lldb::user_id_t m_tid; 322244d93782SGreg Clayton uint32_t m_stop_id; 322344d93782SGreg Clayton }; 322444d93782SGreg Clayton 3225ec990867SGreg Clayton class ThreadsTreeDelegate : public TreeDelegate 3226ec990867SGreg Clayton { 3227ec990867SGreg Clayton public: 3228ec990867SGreg Clayton ThreadsTreeDelegate (Debugger &debugger) : 3229ec990867SGreg Clayton TreeDelegate(), 3230ec990867SGreg Clayton m_thread_delegate_sp (), 3231ec990867SGreg Clayton m_debugger (debugger), 3232ec990867SGreg Clayton m_stop_id (UINT32_MAX) 3233ec990867SGreg Clayton { 3234ec990867SGreg Clayton } 3235ec990867SGreg Clayton 3236ec990867SGreg Clayton virtual 3237ec990867SGreg Clayton ~ThreadsTreeDelegate() 3238ec990867SGreg Clayton { 3239ec990867SGreg Clayton } 3240ec990867SGreg Clayton 3241ec990867SGreg Clayton ProcessSP 3242ec990867SGreg Clayton GetProcess () 3243ec990867SGreg Clayton { 3244ec990867SGreg Clayton return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP(); 3245ec990867SGreg Clayton } 3246ec990867SGreg Clayton 3247ec990867SGreg Clayton virtual void 3248ec990867SGreg Clayton TreeDelegateDrawTreeItem (TreeItem &item, Window &window) 3249ec990867SGreg Clayton { 3250ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 3251ec990867SGreg Clayton if (process_sp && process_sp->IsAlive()) 3252ec990867SGreg Clayton { 3253ec990867SGreg Clayton StreamString strm; 3254ec990867SGreg Clayton ExecutionContext exe_ctx (process_sp); 3255ec990867SGreg Clayton const char *format = "process ${process.id}{, name = ${process.name}}"; 3256ec990867SGreg Clayton if (Debugger::FormatPrompt (format, NULL, &exe_ctx, NULL, strm)) 3257ec990867SGreg Clayton { 3258ec990867SGreg Clayton int right_pad = 1; 3259ec990867SGreg Clayton window.PutCStringTruncated(strm.GetString().c_str(), right_pad); 3260ec990867SGreg Clayton } 3261ec990867SGreg Clayton } 3262ec990867SGreg Clayton } 3263ec990867SGreg Clayton 3264ec990867SGreg Clayton virtual void 3265ec990867SGreg Clayton TreeDelegateGenerateChildren (TreeItem &item) 3266ec990867SGreg Clayton { 3267ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 3268ec990867SGreg Clayton if (process_sp && process_sp->IsAlive()) 3269ec990867SGreg Clayton { 3270ec990867SGreg Clayton StateType state = process_sp->GetState(); 3271ec990867SGreg Clayton if (StateIsStoppedState(state, true)) 3272ec990867SGreg Clayton { 3273ec990867SGreg Clayton const uint32_t stop_id = process_sp->GetStopID(); 3274ec990867SGreg Clayton if (m_stop_id == stop_id) 3275ec990867SGreg Clayton return; // Children are already up to date 3276ec990867SGreg Clayton 3277ec990867SGreg Clayton m_stop_id = stop_id; 3278ec990867SGreg Clayton 3279ec990867SGreg Clayton if (!m_thread_delegate_sp) 3280ec990867SGreg Clayton { 3281ec990867SGreg Clayton // Always expand the thread item the first time we show it 3282ec990867SGreg Clayton //item.Expand(); 3283ec990867SGreg Clayton m_thread_delegate_sp.reset (new ThreadTreeDelegate(m_debugger)); 3284ec990867SGreg Clayton } 3285ec990867SGreg Clayton 3286ec990867SGreg Clayton TreeItem t (&item, *m_thread_delegate_sp, false); 3287ec990867SGreg Clayton ThreadList &threads = process_sp->GetThreadList(); 3288ec990867SGreg Clayton Mutex::Locker locker (threads.GetMutex()); 3289ec990867SGreg Clayton size_t num_threads = threads.GetSize(); 3290ec990867SGreg Clayton item.Resize (num_threads, t); 3291ec990867SGreg Clayton for (size_t i=0; i<num_threads; ++i) 3292ec990867SGreg Clayton { 3293ec990867SGreg Clayton item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID()); 3294ec990867SGreg Clayton item[i].SetMightHaveChildren(true); 3295ec990867SGreg Clayton } 3296ec990867SGreg Clayton return; 3297ec990867SGreg Clayton } 3298ec990867SGreg Clayton } 3299ec990867SGreg Clayton item.ClearChildren(); 3300ec990867SGreg Clayton } 3301ec990867SGreg Clayton 3302ec990867SGreg Clayton virtual bool 3303ec990867SGreg Clayton TreeDelegateItemSelected (TreeItem &item) 3304ec990867SGreg Clayton { 3305ec990867SGreg Clayton return false; 3306ec990867SGreg Clayton } 3307ec990867SGreg Clayton 3308ec990867SGreg Clayton protected: 3309ec990867SGreg Clayton std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp; 3310ec990867SGreg Clayton Debugger &m_debugger; 3311ec990867SGreg Clayton uint32_t m_stop_id; 3312ec990867SGreg Clayton }; 3313ec990867SGreg Clayton 331444d93782SGreg Clayton class ValueObjectListDelegate : public WindowDelegate 331544d93782SGreg Clayton { 331644d93782SGreg Clayton public: 331744d93782SGreg Clayton ValueObjectListDelegate () : 331844d93782SGreg Clayton m_valobj_list (), 331944d93782SGreg Clayton m_rows (), 332044d93782SGreg Clayton m_selected_row (NULL), 332144d93782SGreg Clayton m_selected_row_idx (0), 332244d93782SGreg Clayton m_first_visible_row (0), 332344d93782SGreg Clayton m_num_rows (0), 332444d93782SGreg Clayton m_max_x (0), 332544d93782SGreg Clayton m_max_y (0) 332644d93782SGreg Clayton { 332744d93782SGreg Clayton } 332844d93782SGreg Clayton 332944d93782SGreg Clayton ValueObjectListDelegate (ValueObjectList &valobj_list) : 333044d93782SGreg Clayton m_valobj_list (valobj_list), 333144d93782SGreg Clayton m_rows (), 333244d93782SGreg Clayton m_selected_row (NULL), 333344d93782SGreg Clayton m_selected_row_idx (0), 333444d93782SGreg Clayton m_first_visible_row (0), 333544d93782SGreg Clayton m_num_rows (0), 333644d93782SGreg Clayton m_max_x (0), 333744d93782SGreg Clayton m_max_y (0) 333844d93782SGreg Clayton { 333944d93782SGreg Clayton SetValues (valobj_list); 334044d93782SGreg Clayton } 334144d93782SGreg Clayton 334244d93782SGreg Clayton virtual 334344d93782SGreg Clayton ~ValueObjectListDelegate() 334444d93782SGreg Clayton { 334544d93782SGreg Clayton } 334644d93782SGreg Clayton 334744d93782SGreg Clayton void 334844d93782SGreg Clayton SetValues (ValueObjectList &valobj_list) 334944d93782SGreg Clayton { 335044d93782SGreg Clayton m_selected_row = NULL; 335144d93782SGreg Clayton m_selected_row_idx = 0; 335244d93782SGreg Clayton m_first_visible_row = 0; 335344d93782SGreg Clayton m_num_rows = 0; 335444d93782SGreg Clayton m_rows.clear(); 335544d93782SGreg Clayton m_valobj_list = valobj_list; 335644d93782SGreg Clayton const size_t num_values = m_valobj_list.GetSize(); 335744d93782SGreg Clayton for (size_t i=0; i<num_values; ++i) 335844d93782SGreg Clayton m_rows.push_back(Row(m_valobj_list.GetValueObjectAtIndex(i), NULL)); 335944d93782SGreg Clayton } 336044d93782SGreg Clayton 336144d93782SGreg Clayton virtual bool 336244d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 336344d93782SGreg Clayton { 336444d93782SGreg Clayton m_num_rows = 0; 336544d93782SGreg Clayton m_min_x = 2; 336644d93782SGreg Clayton m_min_y = 1; 336744d93782SGreg Clayton m_max_x = window.GetWidth() - 1; 336844d93782SGreg Clayton m_max_y = window.GetHeight() - 1; 336944d93782SGreg Clayton 337044d93782SGreg Clayton window.Erase(); 337144d93782SGreg Clayton window.DrawTitleBox (window.GetName()); 337244d93782SGreg Clayton 337344d93782SGreg Clayton const int num_visible_rows = NumVisibleRows(); 337444d93782SGreg Clayton const int num_rows = CalculateTotalNumberRows (m_rows); 337544d93782SGreg Clayton 337644d93782SGreg Clayton // If we unexpanded while having something selected our 337744d93782SGreg Clayton // total number of rows is less than the num visible rows, 337844d93782SGreg Clayton // then make sure we show all the rows by setting the first 337944d93782SGreg Clayton // visible row accordingly. 338044d93782SGreg Clayton if (m_first_visible_row > 0 && num_rows < num_visible_rows) 338144d93782SGreg Clayton m_first_visible_row = 0; 338244d93782SGreg Clayton 338344d93782SGreg Clayton // Make sure the selected row is always visible 338444d93782SGreg Clayton if (m_selected_row_idx < m_first_visible_row) 338544d93782SGreg Clayton m_first_visible_row = m_selected_row_idx; 338644d93782SGreg Clayton else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx) 338744d93782SGreg Clayton m_first_visible_row = m_selected_row_idx - num_visible_rows + 1; 338844d93782SGreg Clayton 338944d93782SGreg Clayton DisplayRows (window, m_rows, g_options); 339044d93782SGreg Clayton 339144d93782SGreg Clayton window.DeferredRefresh(); 339244d93782SGreg Clayton 339344d93782SGreg Clayton // Get the selected row 339444d93782SGreg Clayton m_selected_row = GetRowForRowIndex (m_selected_row_idx); 339544d93782SGreg Clayton // Keep the cursor on the selected row so the highlight and the cursor 339644d93782SGreg Clayton // are always on the same line 339744d93782SGreg Clayton if (m_selected_row) 339844d93782SGreg Clayton window.MoveCursor (m_selected_row->x, 339944d93782SGreg Clayton m_selected_row->y); 340044d93782SGreg Clayton 340144d93782SGreg Clayton return true; // Drawing handled 340244d93782SGreg Clayton } 340344d93782SGreg Clayton 340444d93782SGreg Clayton virtual KeyHelp * 340544d93782SGreg Clayton WindowDelegateGetKeyHelp () 340644d93782SGreg Clayton { 340744d93782SGreg Clayton static curses::KeyHelp g_source_view_key_help[] = { 340844d93782SGreg Clayton { KEY_UP, "Select previous item" }, 340944d93782SGreg Clayton { KEY_DOWN, "Select next item" }, 341044d93782SGreg Clayton { KEY_RIGHT, "Expand selected item" }, 341144d93782SGreg Clayton { KEY_LEFT, "Unexpand selected item or select parent if not expanded" }, 341244d93782SGreg Clayton { KEY_PPAGE, "Page up" }, 341344d93782SGreg Clayton { KEY_NPAGE, "Page down" }, 341444d93782SGreg Clayton { 'A', "Format as annotated address" }, 341544d93782SGreg Clayton { 'b', "Format as binary" }, 341644d93782SGreg Clayton { 'B', "Format as hex bytes with ASCII" }, 341744d93782SGreg Clayton { 'c', "Format as character" }, 341844d93782SGreg Clayton { 'd', "Format as a signed integer" }, 341944d93782SGreg Clayton { 'D', "Format selected value using the default format for the type" }, 342044d93782SGreg Clayton { 'f', "Format as float" }, 342144d93782SGreg Clayton { 'h', "Show help dialog" }, 342244d93782SGreg Clayton { 'i', "Format as instructions" }, 342344d93782SGreg Clayton { 'o', "Format as octal" }, 342444d93782SGreg Clayton { 'p', "Format as pointer" }, 342544d93782SGreg Clayton { 's', "Format as C string" }, 342644d93782SGreg Clayton { 't', "Toggle showing/hiding type names" }, 342744d93782SGreg Clayton { 'u', "Format as an unsigned integer" }, 342844d93782SGreg Clayton { 'x', "Format as hex" }, 342944d93782SGreg Clayton { 'X', "Format as uppercase hex" }, 343044d93782SGreg Clayton { ' ', "Toggle item expansion" }, 343144d93782SGreg Clayton { ',', "Page up" }, 343244d93782SGreg Clayton { '.', "Page down" }, 343344d93782SGreg Clayton { '\0', NULL } 343444d93782SGreg Clayton }; 343544d93782SGreg Clayton return g_source_view_key_help; 343644d93782SGreg Clayton } 343744d93782SGreg Clayton 343844d93782SGreg Clayton 343944d93782SGreg Clayton virtual HandleCharResult 344044d93782SGreg Clayton WindowDelegateHandleChar (Window &window, int c) 344144d93782SGreg Clayton { 344244d93782SGreg Clayton switch(c) 344344d93782SGreg Clayton { 344444d93782SGreg Clayton case 'x': 344544d93782SGreg Clayton case 'X': 344644d93782SGreg Clayton case 'o': 344744d93782SGreg Clayton case 's': 344844d93782SGreg Clayton case 'u': 344944d93782SGreg Clayton case 'd': 345044d93782SGreg Clayton case 'D': 345144d93782SGreg Clayton case 'i': 345244d93782SGreg Clayton case 'A': 345344d93782SGreg Clayton case 'p': 345444d93782SGreg Clayton case 'c': 345544d93782SGreg Clayton case 'b': 345644d93782SGreg Clayton case 'B': 345744d93782SGreg Clayton case 'f': 345844d93782SGreg Clayton // Change the format for the currently selected item 345944d93782SGreg Clayton if (m_selected_row) 346044d93782SGreg Clayton m_selected_row->valobj->SetFormat (FormatForChar (c)); 346144d93782SGreg Clayton return eKeyHandled; 346244d93782SGreg Clayton 346344d93782SGreg Clayton case 't': 346444d93782SGreg Clayton // Toggle showing type names 346544d93782SGreg Clayton g_options.show_types = !g_options.show_types; 346644d93782SGreg Clayton return eKeyHandled; 346744d93782SGreg Clayton 346844d93782SGreg Clayton case ',': 346944d93782SGreg Clayton case KEY_PPAGE: 347044d93782SGreg Clayton // Page up key 347144d93782SGreg Clayton if (m_first_visible_row > 0) 347244d93782SGreg Clayton { 34733985c8c6SSaleem Abdulrasool if (static_cast<int>(m_first_visible_row) > m_max_y) 347444d93782SGreg Clayton m_first_visible_row -= m_max_y; 347544d93782SGreg Clayton else 347644d93782SGreg Clayton m_first_visible_row = 0; 347744d93782SGreg Clayton m_selected_row_idx = m_first_visible_row; 347844d93782SGreg Clayton } 347944d93782SGreg Clayton return eKeyHandled; 348044d93782SGreg Clayton 348144d93782SGreg Clayton case '.': 348244d93782SGreg Clayton case KEY_NPAGE: 348344d93782SGreg Clayton // Page down key 34843985c8c6SSaleem Abdulrasool if (m_num_rows > static_cast<size_t>(m_max_y)) 348544d93782SGreg Clayton { 348644d93782SGreg Clayton if (m_first_visible_row + m_max_y < m_num_rows) 348744d93782SGreg Clayton { 348844d93782SGreg Clayton m_first_visible_row += m_max_y; 348944d93782SGreg Clayton m_selected_row_idx = m_first_visible_row; 349044d93782SGreg Clayton } 349144d93782SGreg Clayton } 349244d93782SGreg Clayton return eKeyHandled; 349344d93782SGreg Clayton 349444d93782SGreg Clayton case KEY_UP: 349544d93782SGreg Clayton if (m_selected_row_idx > 0) 349644d93782SGreg Clayton --m_selected_row_idx; 349744d93782SGreg Clayton return eKeyHandled; 349844d93782SGreg Clayton case KEY_DOWN: 349944d93782SGreg Clayton if (m_selected_row_idx + 1 < m_num_rows) 350044d93782SGreg Clayton ++m_selected_row_idx; 350144d93782SGreg Clayton return eKeyHandled; 350244d93782SGreg Clayton 350344d93782SGreg Clayton case KEY_RIGHT: 350444d93782SGreg Clayton if (m_selected_row) 350544d93782SGreg Clayton { 350644d93782SGreg Clayton if (!m_selected_row->expanded) 350744d93782SGreg Clayton m_selected_row->Expand(); 350844d93782SGreg Clayton } 350944d93782SGreg Clayton return eKeyHandled; 351044d93782SGreg Clayton 351144d93782SGreg Clayton case KEY_LEFT: 351244d93782SGreg Clayton if (m_selected_row) 351344d93782SGreg Clayton { 351444d93782SGreg Clayton if (m_selected_row->expanded) 351544d93782SGreg Clayton m_selected_row->Unexpand(); 351644d93782SGreg Clayton else if (m_selected_row->parent) 351744d93782SGreg Clayton m_selected_row_idx = m_selected_row->parent->row_idx; 351844d93782SGreg Clayton } 351944d93782SGreg Clayton return eKeyHandled; 352044d93782SGreg Clayton 352144d93782SGreg Clayton case ' ': 352244d93782SGreg Clayton // Toggle expansion state when SPACE is pressed 352344d93782SGreg Clayton if (m_selected_row) 352444d93782SGreg Clayton { 352544d93782SGreg Clayton if (m_selected_row->expanded) 352644d93782SGreg Clayton m_selected_row->Unexpand(); 352744d93782SGreg Clayton else 352844d93782SGreg Clayton m_selected_row->Expand(); 352944d93782SGreg Clayton } 353044d93782SGreg Clayton return eKeyHandled; 353144d93782SGreg Clayton 353244d93782SGreg Clayton case 'h': 353344d93782SGreg Clayton window.CreateHelpSubwindow (); 353444d93782SGreg Clayton return eKeyHandled; 353544d93782SGreg Clayton 353644d93782SGreg Clayton default: 353744d93782SGreg Clayton break; 353844d93782SGreg Clayton } 353944d93782SGreg Clayton return eKeyNotHandled; 354044d93782SGreg Clayton } 354144d93782SGreg Clayton 354244d93782SGreg Clayton protected: 354344d93782SGreg Clayton ValueObjectList m_valobj_list; 354444d93782SGreg Clayton std::vector<Row> m_rows; 354544d93782SGreg Clayton Row *m_selected_row; 354644d93782SGreg Clayton uint32_t m_selected_row_idx; 354744d93782SGreg Clayton uint32_t m_first_visible_row; 354844d93782SGreg Clayton uint32_t m_num_rows; 354944d93782SGreg Clayton int m_min_x; 355044d93782SGreg Clayton int m_min_y; 355144d93782SGreg Clayton int m_max_x; 355244d93782SGreg Clayton int m_max_y; 355344d93782SGreg Clayton 355444d93782SGreg Clayton static Format 355544d93782SGreg Clayton FormatForChar (int c) 355644d93782SGreg Clayton { 355744d93782SGreg Clayton switch (c) 355844d93782SGreg Clayton { 355944d93782SGreg Clayton case 'x': return eFormatHex; 356044d93782SGreg Clayton case 'X': return eFormatHexUppercase; 356144d93782SGreg Clayton case 'o': return eFormatOctal; 356244d93782SGreg Clayton case 's': return eFormatCString; 356344d93782SGreg Clayton case 'u': return eFormatUnsigned; 356444d93782SGreg Clayton case 'd': return eFormatDecimal; 356544d93782SGreg Clayton case 'D': return eFormatDefault; 356644d93782SGreg Clayton case 'i': return eFormatInstruction; 356744d93782SGreg Clayton case 'A': return eFormatAddressInfo; 356844d93782SGreg Clayton case 'p': return eFormatPointer; 356944d93782SGreg Clayton case 'c': return eFormatChar; 357044d93782SGreg Clayton case 'b': return eFormatBinary; 357144d93782SGreg Clayton case 'B': return eFormatBytesWithASCII; 357244d93782SGreg Clayton case 'f': return eFormatFloat; 357344d93782SGreg Clayton } 357444d93782SGreg Clayton return eFormatDefault; 357544d93782SGreg Clayton } 357644d93782SGreg Clayton 357744d93782SGreg Clayton bool 357844d93782SGreg Clayton DisplayRowObject (Window &window, 357944d93782SGreg Clayton Row &row, 358044d93782SGreg Clayton DisplayOptions &options, 358144d93782SGreg Clayton bool highlight, 358244d93782SGreg Clayton bool last_child) 358344d93782SGreg Clayton { 358444d93782SGreg Clayton ValueObject *valobj = row.valobj.get(); 358544d93782SGreg Clayton 358644d93782SGreg Clayton if (valobj == NULL) 358744d93782SGreg Clayton return false; 358844d93782SGreg Clayton 358944d93782SGreg Clayton const char *type_name = options.show_types ? valobj->GetTypeName().GetCString() : NULL; 359044d93782SGreg Clayton const char *name = valobj->GetName().GetCString(); 359144d93782SGreg Clayton const char *value = valobj->GetValueAsCString (); 359244d93782SGreg Clayton const char *summary = valobj->GetSummaryAsCString (); 359344d93782SGreg Clayton 359444d93782SGreg Clayton window.MoveCursor (row.x, row.y); 359544d93782SGreg Clayton 359644d93782SGreg Clayton row.DrawTree (window); 359744d93782SGreg Clayton 359844d93782SGreg Clayton if (highlight) 359944d93782SGreg Clayton window.AttributeOn(A_REVERSE); 360044d93782SGreg Clayton 360144d93782SGreg Clayton if (type_name && type_name[0]) 360244d93782SGreg Clayton window.Printf ("(%s) ", type_name); 360344d93782SGreg Clayton 360444d93782SGreg Clayton if (name && name[0]) 360544d93782SGreg Clayton window.PutCString(name); 360644d93782SGreg Clayton 360744d93782SGreg Clayton attr_t changd_attr = 0; 360844d93782SGreg Clayton if (valobj->GetValueDidChange()) 360944d93782SGreg Clayton changd_attr = COLOR_PAIR(5) | A_BOLD; 361044d93782SGreg Clayton 361144d93782SGreg Clayton if (value && value[0]) 361244d93782SGreg Clayton { 361344d93782SGreg Clayton window.PutCString(" = "); 361444d93782SGreg Clayton if (changd_attr) 361544d93782SGreg Clayton window.AttributeOn(changd_attr); 361644d93782SGreg Clayton window.PutCString (value); 361744d93782SGreg Clayton if (changd_attr) 361844d93782SGreg Clayton window.AttributeOff(changd_attr); 361944d93782SGreg Clayton } 362044d93782SGreg Clayton 362144d93782SGreg Clayton if (summary && summary[0]) 362244d93782SGreg Clayton { 362344d93782SGreg Clayton window.PutChar(' '); 362444d93782SGreg Clayton if (changd_attr) 362544d93782SGreg Clayton window.AttributeOn(changd_attr); 362644d93782SGreg Clayton window.PutCString(summary); 362744d93782SGreg Clayton if (changd_attr) 362844d93782SGreg Clayton window.AttributeOff(changd_attr); 362944d93782SGreg Clayton } 363044d93782SGreg Clayton 363144d93782SGreg Clayton if (highlight) 363244d93782SGreg Clayton window.AttributeOff (A_REVERSE); 363344d93782SGreg Clayton 363444d93782SGreg Clayton return true; 363544d93782SGreg Clayton } 363644d93782SGreg Clayton void 363744d93782SGreg Clayton DisplayRows (Window &window, 363844d93782SGreg Clayton std::vector<Row> &rows, 363944d93782SGreg Clayton DisplayOptions &options) 364044d93782SGreg Clayton { 364144d93782SGreg Clayton // > 0x25B7 364244d93782SGreg Clayton // \/ 0x25BD 364344d93782SGreg Clayton 364444d93782SGreg Clayton bool window_is_active = window.IsActive(); 364544d93782SGreg Clayton for (auto &row : rows) 364644d93782SGreg Clayton { 364744d93782SGreg Clayton const bool last_child = row.parent && &rows[rows.size()-1] == &row; 364844d93782SGreg Clayton // Save the row index in each Row structure 364944d93782SGreg Clayton row.row_idx = m_num_rows; 365044d93782SGreg Clayton if ((m_num_rows >= m_first_visible_row) && 36513985c8c6SSaleem Abdulrasool ((m_num_rows - m_first_visible_row) < static_cast<size_t>(NumVisibleRows()))) 365244d93782SGreg Clayton { 365344d93782SGreg Clayton row.x = m_min_x; 365444d93782SGreg Clayton row.y = m_num_rows - m_first_visible_row + 1; 365544d93782SGreg Clayton if (DisplayRowObject (window, 365644d93782SGreg Clayton row, 365744d93782SGreg Clayton options, 365844d93782SGreg Clayton window_is_active && m_num_rows == m_selected_row_idx, 365944d93782SGreg Clayton last_child)) 366044d93782SGreg Clayton { 366144d93782SGreg Clayton ++m_num_rows; 366244d93782SGreg Clayton } 366344d93782SGreg Clayton else 366444d93782SGreg Clayton { 366544d93782SGreg Clayton row.x = 0; 366644d93782SGreg Clayton row.y = 0; 366744d93782SGreg Clayton } 366844d93782SGreg Clayton } 366944d93782SGreg Clayton else 367044d93782SGreg Clayton { 367144d93782SGreg Clayton row.x = 0; 367244d93782SGreg Clayton row.y = 0; 367344d93782SGreg Clayton ++m_num_rows; 367444d93782SGreg Clayton } 367544d93782SGreg Clayton 367644d93782SGreg Clayton if (row.expanded && !row.children.empty()) 367744d93782SGreg Clayton { 367844d93782SGreg Clayton DisplayRows (window, 367944d93782SGreg Clayton row.children, 368044d93782SGreg Clayton options); 368144d93782SGreg Clayton } 368244d93782SGreg Clayton } 368344d93782SGreg Clayton } 368444d93782SGreg Clayton 368544d93782SGreg Clayton int 368644d93782SGreg Clayton CalculateTotalNumberRows (const std::vector<Row> &rows) 368744d93782SGreg Clayton { 368844d93782SGreg Clayton int row_count = 0; 368944d93782SGreg Clayton for (const auto &row : rows) 369044d93782SGreg Clayton { 369144d93782SGreg Clayton ++row_count; 369244d93782SGreg Clayton if (row.expanded) 369344d93782SGreg Clayton row_count += CalculateTotalNumberRows(row.children); 369444d93782SGreg Clayton } 369544d93782SGreg Clayton return row_count; 369644d93782SGreg Clayton } 369744d93782SGreg Clayton static Row * 369844d93782SGreg Clayton GetRowForRowIndexImpl (std::vector<Row> &rows, size_t &row_index) 369944d93782SGreg Clayton { 370044d93782SGreg Clayton for (auto &row : rows) 370144d93782SGreg Clayton { 370244d93782SGreg Clayton if (row_index == 0) 370344d93782SGreg Clayton return &row; 370444d93782SGreg Clayton else 370544d93782SGreg Clayton { 370644d93782SGreg Clayton --row_index; 370744d93782SGreg Clayton if (row.expanded && !row.children.empty()) 370844d93782SGreg Clayton { 370944d93782SGreg Clayton Row *result = GetRowForRowIndexImpl (row.children, row_index); 371044d93782SGreg Clayton if (result) 371144d93782SGreg Clayton return result; 371244d93782SGreg Clayton } 371344d93782SGreg Clayton } 371444d93782SGreg Clayton } 371544d93782SGreg Clayton return NULL; 371644d93782SGreg Clayton } 371744d93782SGreg Clayton 371844d93782SGreg Clayton Row * 371944d93782SGreg Clayton GetRowForRowIndex (size_t row_index) 372044d93782SGreg Clayton { 372144d93782SGreg Clayton return GetRowForRowIndexImpl (m_rows, row_index); 372244d93782SGreg Clayton } 372344d93782SGreg Clayton 372444d93782SGreg Clayton int 372544d93782SGreg Clayton NumVisibleRows () const 372644d93782SGreg Clayton { 372744d93782SGreg Clayton return m_max_y - m_min_y; 372844d93782SGreg Clayton } 372944d93782SGreg Clayton 373044d93782SGreg Clayton static DisplayOptions g_options; 373144d93782SGreg Clayton }; 373244d93782SGreg Clayton 373344d93782SGreg Clayton class FrameVariablesWindowDelegate : public ValueObjectListDelegate 373444d93782SGreg Clayton { 373544d93782SGreg Clayton public: 373644d93782SGreg Clayton FrameVariablesWindowDelegate (Debugger &debugger) : 373744d93782SGreg Clayton ValueObjectListDelegate (), 373844d93782SGreg Clayton m_debugger (debugger), 373944d93782SGreg Clayton m_frame_block (NULL) 374044d93782SGreg Clayton { 374144d93782SGreg Clayton } 374244d93782SGreg Clayton 374344d93782SGreg Clayton virtual 374444d93782SGreg Clayton ~FrameVariablesWindowDelegate() 374544d93782SGreg Clayton { 374644d93782SGreg Clayton } 374744d93782SGreg Clayton 374844d93782SGreg Clayton virtual const char * 374944d93782SGreg Clayton WindowDelegateGetHelpText () 375044d93782SGreg Clayton { 375144d93782SGreg Clayton return "Frame variable window keyboard shortcuts:"; 375244d93782SGreg Clayton } 375344d93782SGreg Clayton 375444d93782SGreg Clayton virtual bool 375544d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 375644d93782SGreg Clayton { 375744d93782SGreg Clayton ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext()); 375844d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 375944d93782SGreg Clayton Block *frame_block = NULL; 376044d93782SGreg Clayton StackFrame *frame = NULL; 376144d93782SGreg Clayton 376244d93782SGreg Clayton if (process) 376344d93782SGreg Clayton { 376444d93782SGreg Clayton StateType state = process->GetState(); 376544d93782SGreg Clayton if (StateIsStoppedState(state, true)) 376644d93782SGreg Clayton { 376744d93782SGreg Clayton frame = exe_ctx.GetFramePtr(); 376844d93782SGreg Clayton if (frame) 376944d93782SGreg Clayton frame_block = frame->GetFrameBlock (); 377044d93782SGreg Clayton } 377144d93782SGreg Clayton else if (StateIsRunningState(state)) 377244d93782SGreg Clayton { 377344d93782SGreg Clayton return true; // Don't do any updating when we are running 377444d93782SGreg Clayton } 377544d93782SGreg Clayton } 377644d93782SGreg Clayton 377744d93782SGreg Clayton ValueObjectList local_values; 377844d93782SGreg Clayton if (frame_block) 377944d93782SGreg Clayton { 378044d93782SGreg Clayton // Only update the variables if they have changed 378144d93782SGreg Clayton if (m_frame_block != frame_block) 378244d93782SGreg Clayton { 378344d93782SGreg Clayton m_frame_block = frame_block; 378444d93782SGreg Clayton 378544d93782SGreg Clayton VariableList *locals = frame->GetVariableList(true); 378644d93782SGreg Clayton if (locals) 378744d93782SGreg Clayton { 378844d93782SGreg Clayton const DynamicValueType use_dynamic = eDynamicDontRunTarget; 378944d93782SGreg Clayton const size_t num_locals = locals->GetSize(); 379044d93782SGreg Clayton for (size_t i=0; i<num_locals; ++i) 379144d93782SGreg Clayton local_values.Append(frame->GetValueObjectForFrameVariable (locals->GetVariableAtIndex(i), use_dynamic)); 379244d93782SGreg Clayton // Update the values 379344d93782SGreg Clayton SetValues(local_values); 379444d93782SGreg Clayton } 379544d93782SGreg Clayton } 379644d93782SGreg Clayton } 379744d93782SGreg Clayton else 379844d93782SGreg Clayton { 379944d93782SGreg Clayton m_frame_block = NULL; 380044d93782SGreg Clayton // Update the values with an empty list if there is no frame 380144d93782SGreg Clayton SetValues(local_values); 380244d93782SGreg Clayton } 380344d93782SGreg Clayton 380444d93782SGreg Clayton return ValueObjectListDelegate::WindowDelegateDraw (window, force); 380544d93782SGreg Clayton 380644d93782SGreg Clayton } 380744d93782SGreg Clayton 380844d93782SGreg Clayton protected: 380944d93782SGreg Clayton Debugger &m_debugger; 381044d93782SGreg Clayton Block *m_frame_block; 381144d93782SGreg Clayton }; 381244d93782SGreg Clayton 381344d93782SGreg Clayton 381444d93782SGreg Clayton class RegistersWindowDelegate : public ValueObjectListDelegate 381544d93782SGreg Clayton { 381644d93782SGreg Clayton public: 381744d93782SGreg Clayton RegistersWindowDelegate (Debugger &debugger) : 381844d93782SGreg Clayton ValueObjectListDelegate (), 381944d93782SGreg Clayton m_debugger (debugger) 382044d93782SGreg Clayton { 382144d93782SGreg Clayton } 382244d93782SGreg Clayton 382344d93782SGreg Clayton virtual 382444d93782SGreg Clayton ~RegistersWindowDelegate() 382544d93782SGreg Clayton { 382644d93782SGreg Clayton } 382744d93782SGreg Clayton 382844d93782SGreg Clayton virtual const char * 382944d93782SGreg Clayton WindowDelegateGetHelpText () 383044d93782SGreg Clayton { 383144d93782SGreg Clayton return "Register window keyboard shortcuts:"; 383244d93782SGreg Clayton } 383344d93782SGreg Clayton 383444d93782SGreg Clayton virtual bool 383544d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 383644d93782SGreg Clayton { 383744d93782SGreg Clayton ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext()); 383844d93782SGreg Clayton StackFrame *frame = exe_ctx.GetFramePtr(); 383944d93782SGreg Clayton 384044d93782SGreg Clayton ValueObjectList value_list; 384144d93782SGreg Clayton if (frame) 384244d93782SGreg Clayton { 384344d93782SGreg Clayton if (frame->GetStackID() != m_stack_id) 384444d93782SGreg Clayton { 384544d93782SGreg Clayton m_stack_id = frame->GetStackID(); 384644d93782SGreg Clayton RegisterContextSP reg_ctx (frame->GetRegisterContext()); 384744d93782SGreg Clayton if (reg_ctx) 384844d93782SGreg Clayton { 384944d93782SGreg Clayton const uint32_t num_sets = reg_ctx->GetRegisterSetCount(); 385044d93782SGreg Clayton for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) 385144d93782SGreg Clayton { 385244d93782SGreg Clayton value_list.Append(ValueObjectRegisterSet::Create (frame, reg_ctx, set_idx)); 385344d93782SGreg Clayton } 385444d93782SGreg Clayton } 385544d93782SGreg Clayton SetValues(value_list); 385644d93782SGreg Clayton } 385744d93782SGreg Clayton } 385844d93782SGreg Clayton else 385944d93782SGreg Clayton { 386044d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 386144d93782SGreg Clayton if (process && process->IsAlive()) 386244d93782SGreg Clayton return true; // Don't do any updating if we are running 386344d93782SGreg Clayton else 386444d93782SGreg Clayton { 386544d93782SGreg Clayton // Update the values with an empty list if there 386644d93782SGreg Clayton // is no process or the process isn't alive anymore 386744d93782SGreg Clayton SetValues(value_list); 386844d93782SGreg Clayton } 386944d93782SGreg Clayton } 387044d93782SGreg Clayton return ValueObjectListDelegate::WindowDelegateDraw (window, force); 387144d93782SGreg Clayton } 387244d93782SGreg Clayton 387344d93782SGreg Clayton protected: 387444d93782SGreg Clayton Debugger &m_debugger; 387544d93782SGreg Clayton StackID m_stack_id; 387644d93782SGreg Clayton }; 387744d93782SGreg Clayton 387844d93782SGreg Clayton static const char * 387944d93782SGreg Clayton CursesKeyToCString (int ch) 388044d93782SGreg Clayton { 388144d93782SGreg Clayton static char g_desc[32]; 388244d93782SGreg Clayton if (ch >= KEY_F0 && ch < KEY_F0 + 64) 388344d93782SGreg Clayton { 388444d93782SGreg Clayton snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0); 388544d93782SGreg Clayton return g_desc; 388644d93782SGreg Clayton } 388744d93782SGreg Clayton switch (ch) 388844d93782SGreg Clayton { 388944d93782SGreg Clayton case KEY_DOWN: return "down"; 389044d93782SGreg Clayton case KEY_UP: return "up"; 389144d93782SGreg Clayton case KEY_LEFT: return "left"; 389244d93782SGreg Clayton case KEY_RIGHT: return "right"; 389344d93782SGreg Clayton case KEY_HOME: return "home"; 389444d93782SGreg Clayton case KEY_BACKSPACE: return "backspace"; 389544d93782SGreg Clayton case KEY_DL: return "delete-line"; 389644d93782SGreg Clayton case KEY_IL: return "insert-line"; 389744d93782SGreg Clayton case KEY_DC: return "delete-char"; 389844d93782SGreg Clayton case KEY_IC: return "insert-char"; 389944d93782SGreg Clayton case KEY_CLEAR: return "clear"; 390044d93782SGreg Clayton case KEY_EOS: return "clear-to-eos"; 390144d93782SGreg Clayton case KEY_EOL: return "clear-to-eol"; 390244d93782SGreg Clayton case KEY_SF: return "scroll-forward"; 390344d93782SGreg Clayton case KEY_SR: return "scroll-backward"; 390444d93782SGreg Clayton case KEY_NPAGE: return "page-down"; 390544d93782SGreg Clayton case KEY_PPAGE: return "page-up"; 390644d93782SGreg Clayton case KEY_STAB: return "set-tab"; 390744d93782SGreg Clayton case KEY_CTAB: return "clear-tab"; 390844d93782SGreg Clayton case KEY_CATAB: return "clear-all-tabs"; 390944d93782SGreg Clayton case KEY_ENTER: return "enter"; 391044d93782SGreg Clayton case KEY_PRINT: return "print"; 391144d93782SGreg Clayton case KEY_LL: return "lower-left key"; 391244d93782SGreg Clayton case KEY_A1: return "upper left of keypad"; 391344d93782SGreg Clayton case KEY_A3: return "upper right of keypad"; 391444d93782SGreg Clayton case KEY_B2: return "center of keypad"; 391544d93782SGreg Clayton case KEY_C1: return "lower left of keypad"; 391644d93782SGreg Clayton case KEY_C3: return "lower right of keypad"; 391744d93782SGreg Clayton case KEY_BTAB: return "back-tab key"; 391844d93782SGreg Clayton case KEY_BEG: return "begin key"; 391944d93782SGreg Clayton case KEY_CANCEL: return "cancel key"; 392044d93782SGreg Clayton case KEY_CLOSE: return "close key"; 392144d93782SGreg Clayton case KEY_COMMAND: return "command key"; 392244d93782SGreg Clayton case KEY_COPY: return "copy key"; 392344d93782SGreg Clayton case KEY_CREATE: return "create key"; 392444d93782SGreg Clayton case KEY_END: return "end key"; 392544d93782SGreg Clayton case KEY_EXIT: return "exit key"; 392644d93782SGreg Clayton case KEY_FIND: return "find key"; 392744d93782SGreg Clayton case KEY_HELP: return "help key"; 392844d93782SGreg Clayton case KEY_MARK: return "mark key"; 392944d93782SGreg Clayton case KEY_MESSAGE: return "message key"; 393044d93782SGreg Clayton case KEY_MOVE: return "move key"; 393144d93782SGreg Clayton case KEY_NEXT: return "next key"; 393244d93782SGreg Clayton case KEY_OPEN: return "open key"; 393344d93782SGreg Clayton case KEY_OPTIONS: return "options key"; 393444d93782SGreg Clayton case KEY_PREVIOUS: return "previous key"; 393544d93782SGreg Clayton case KEY_REDO: return "redo key"; 393644d93782SGreg Clayton case KEY_REFERENCE: return "reference key"; 393744d93782SGreg Clayton case KEY_REFRESH: return "refresh key"; 393844d93782SGreg Clayton case KEY_REPLACE: return "replace key"; 393944d93782SGreg Clayton case KEY_RESTART: return "restart key"; 394044d93782SGreg Clayton case KEY_RESUME: return "resume key"; 394144d93782SGreg Clayton case KEY_SAVE: return "save key"; 394244d93782SGreg Clayton case KEY_SBEG: return "shifted begin key"; 394344d93782SGreg Clayton case KEY_SCANCEL: return "shifted cancel key"; 394444d93782SGreg Clayton case KEY_SCOMMAND: return "shifted command key"; 394544d93782SGreg Clayton case KEY_SCOPY: return "shifted copy key"; 394644d93782SGreg Clayton case KEY_SCREATE: return "shifted create key"; 394744d93782SGreg Clayton case KEY_SDC: return "shifted delete-character key"; 394844d93782SGreg Clayton case KEY_SDL: return "shifted delete-line key"; 394944d93782SGreg Clayton case KEY_SELECT: return "select key"; 395044d93782SGreg Clayton case KEY_SEND: return "shifted end key"; 395144d93782SGreg Clayton case KEY_SEOL: return "shifted clear-to-end-of-line key"; 395244d93782SGreg Clayton case KEY_SEXIT: return "shifted exit key"; 395344d93782SGreg Clayton case KEY_SFIND: return "shifted find key"; 395444d93782SGreg Clayton case KEY_SHELP: return "shifted help key"; 395544d93782SGreg Clayton case KEY_SHOME: return "shifted home key"; 395644d93782SGreg Clayton case KEY_SIC: return "shifted insert-character key"; 395744d93782SGreg Clayton case KEY_SLEFT: return "shifted left-arrow key"; 395844d93782SGreg Clayton case KEY_SMESSAGE: return "shifted message key"; 395944d93782SGreg Clayton case KEY_SMOVE: return "shifted move key"; 396044d93782SGreg Clayton case KEY_SNEXT: return "shifted next key"; 396144d93782SGreg Clayton case KEY_SOPTIONS: return "shifted options key"; 396244d93782SGreg Clayton case KEY_SPREVIOUS: return "shifted previous key"; 396344d93782SGreg Clayton case KEY_SPRINT: return "shifted print key"; 396444d93782SGreg Clayton case KEY_SREDO: return "shifted redo key"; 396544d93782SGreg Clayton case KEY_SREPLACE: return "shifted replace key"; 396644d93782SGreg Clayton case KEY_SRIGHT: return "shifted right-arrow key"; 396744d93782SGreg Clayton case KEY_SRSUME: return "shifted resume key"; 396844d93782SGreg Clayton case KEY_SSAVE: return "shifted save key"; 396944d93782SGreg Clayton case KEY_SSUSPEND: return "shifted suspend key"; 397044d93782SGreg Clayton case KEY_SUNDO: return "shifted undo key"; 397144d93782SGreg Clayton case KEY_SUSPEND: return "suspend key"; 397244d93782SGreg Clayton case KEY_UNDO: return "undo key"; 397344d93782SGreg Clayton case KEY_MOUSE: return "Mouse event has occurred"; 397444d93782SGreg Clayton case KEY_RESIZE: return "Terminal resize event"; 397544d93782SGreg Clayton case KEY_EVENT: return "We were interrupted by an event"; 397644d93782SGreg Clayton case KEY_RETURN: return "return"; 397744d93782SGreg Clayton case ' ': return "space"; 39785fdb09bbSGreg Clayton case '\t': return "tab"; 397944d93782SGreg Clayton case KEY_ESCAPE: return "escape"; 398044d93782SGreg Clayton default: 398144d93782SGreg Clayton if (isprint(ch)) 398244d93782SGreg Clayton snprintf(g_desc, sizeof(g_desc), "%c", ch); 398344d93782SGreg Clayton else 398444d93782SGreg Clayton snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch); 398544d93782SGreg Clayton return g_desc; 398644d93782SGreg Clayton } 398744d93782SGreg Clayton return NULL; 398844d93782SGreg Clayton } 398944d93782SGreg Clayton 399044d93782SGreg Clayton HelpDialogDelegate::HelpDialogDelegate (const char *text, KeyHelp *key_help_array) : 399144d93782SGreg Clayton m_text (), 399244d93782SGreg Clayton m_first_visible_line (0) 399344d93782SGreg Clayton { 399444d93782SGreg Clayton if (text && text[0]) 399544d93782SGreg Clayton { 399644d93782SGreg Clayton m_text.SplitIntoLines(text); 399744d93782SGreg Clayton m_text.AppendString(""); 399844d93782SGreg Clayton } 399944d93782SGreg Clayton if (key_help_array) 400044d93782SGreg Clayton { 400144d93782SGreg Clayton for (KeyHelp *key = key_help_array; key->ch; ++key) 400244d93782SGreg Clayton { 400344d93782SGreg Clayton StreamString key_description; 400444d93782SGreg Clayton key_description.Printf("%10s - %s", CursesKeyToCString(key->ch), key->description); 400544d93782SGreg Clayton m_text.AppendString(std::move(key_description.GetString())); 400644d93782SGreg Clayton } 400744d93782SGreg Clayton } 400844d93782SGreg Clayton } 400944d93782SGreg Clayton 401044d93782SGreg Clayton HelpDialogDelegate::~HelpDialogDelegate() 401144d93782SGreg Clayton { 401244d93782SGreg Clayton } 401344d93782SGreg Clayton 401444d93782SGreg Clayton bool 401544d93782SGreg Clayton HelpDialogDelegate::WindowDelegateDraw (Window &window, bool force) 401644d93782SGreg Clayton { 401744d93782SGreg Clayton window.Erase(); 401844d93782SGreg Clayton const int window_height = window.GetHeight(); 401944d93782SGreg Clayton int x = 2; 402044d93782SGreg Clayton int y = 1; 402144d93782SGreg Clayton const int min_y = y; 402244d93782SGreg Clayton const int max_y = window_height - 1 - y; 40233985c8c6SSaleem Abdulrasool const size_t num_visible_lines = max_y - min_y + 1; 402444d93782SGreg Clayton const size_t num_lines = m_text.GetSize(); 402544d93782SGreg Clayton const char *bottom_message; 402644d93782SGreg Clayton if (num_lines <= num_visible_lines) 402744d93782SGreg Clayton bottom_message = "Press any key to exit"; 402844d93782SGreg Clayton else 402944d93782SGreg Clayton bottom_message = "Use arrows to scroll, any other key to exit"; 403044d93782SGreg Clayton window.DrawTitleBox(window.GetName(), bottom_message); 403144d93782SGreg Clayton while (y <= max_y) 403244d93782SGreg Clayton { 403344d93782SGreg Clayton window.MoveCursor(x, y); 403444d93782SGreg Clayton window.PutCStringTruncated(m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1); 403544d93782SGreg Clayton ++y; 403644d93782SGreg Clayton } 403744d93782SGreg Clayton return true; 403844d93782SGreg Clayton } 403944d93782SGreg Clayton 404044d93782SGreg Clayton HandleCharResult 404144d93782SGreg Clayton HelpDialogDelegate::WindowDelegateHandleChar (Window &window, int key) 404244d93782SGreg Clayton { 404344d93782SGreg Clayton bool done = false; 404444d93782SGreg Clayton const size_t num_lines = m_text.GetSize(); 404544d93782SGreg Clayton const size_t num_visible_lines = window.GetHeight() - 2; 404644d93782SGreg Clayton 404744d93782SGreg Clayton if (num_lines <= num_visible_lines) 404844d93782SGreg Clayton { 404944d93782SGreg Clayton done = true; 405044d93782SGreg Clayton // If we have all lines visible and don't need scrolling, then any 405144d93782SGreg Clayton // key press will cause us to exit 405244d93782SGreg Clayton } 405344d93782SGreg Clayton else 405444d93782SGreg Clayton { 405544d93782SGreg Clayton switch (key) 405644d93782SGreg Clayton { 405744d93782SGreg Clayton case KEY_UP: 405844d93782SGreg Clayton if (m_first_visible_line > 0) 405944d93782SGreg Clayton --m_first_visible_line; 406044d93782SGreg Clayton break; 406144d93782SGreg Clayton 406244d93782SGreg Clayton case KEY_DOWN: 406344d93782SGreg Clayton if (m_first_visible_line + num_visible_lines < num_lines) 406444d93782SGreg Clayton ++m_first_visible_line; 406544d93782SGreg Clayton break; 406644d93782SGreg Clayton 406744d93782SGreg Clayton case KEY_PPAGE: 406844d93782SGreg Clayton case ',': 406944d93782SGreg Clayton if (m_first_visible_line > 0) 407044d93782SGreg Clayton { 40713985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines) 407244d93782SGreg Clayton m_first_visible_line -= num_visible_lines; 407344d93782SGreg Clayton else 407444d93782SGreg Clayton m_first_visible_line = 0; 407544d93782SGreg Clayton } 407644d93782SGreg Clayton break; 407744d93782SGreg Clayton case KEY_NPAGE: 407844d93782SGreg Clayton case '.': 407944d93782SGreg Clayton if (m_first_visible_line + num_visible_lines < num_lines) 408044d93782SGreg Clayton { 408144d93782SGreg Clayton m_first_visible_line += num_visible_lines; 40823985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_first_visible_line) > num_lines) 408344d93782SGreg Clayton m_first_visible_line = num_lines - num_visible_lines; 408444d93782SGreg Clayton } 408544d93782SGreg Clayton break; 408644d93782SGreg Clayton default: 408744d93782SGreg Clayton done = true; 408844d93782SGreg Clayton break; 408944d93782SGreg Clayton } 409044d93782SGreg Clayton } 409144d93782SGreg Clayton if (done) 409244d93782SGreg Clayton window.GetParent()->RemoveSubWindow(&window); 409344d93782SGreg Clayton return eKeyHandled; 409444d93782SGreg Clayton } 409544d93782SGreg Clayton 409644d93782SGreg Clayton class ApplicationDelegate : 409744d93782SGreg Clayton public WindowDelegate, 409844d93782SGreg Clayton public MenuDelegate 409944d93782SGreg Clayton { 410044d93782SGreg Clayton public: 410144d93782SGreg Clayton enum { 410244d93782SGreg Clayton eMenuID_LLDB = 1, 410344d93782SGreg Clayton eMenuID_LLDBAbout, 410444d93782SGreg Clayton eMenuID_LLDBExit, 410544d93782SGreg Clayton 410644d93782SGreg Clayton eMenuID_Target, 410744d93782SGreg Clayton eMenuID_TargetCreate, 410844d93782SGreg Clayton eMenuID_TargetDelete, 410944d93782SGreg Clayton 411044d93782SGreg Clayton eMenuID_Process, 411144d93782SGreg Clayton eMenuID_ProcessAttach, 411244d93782SGreg Clayton eMenuID_ProcessDetach, 411344d93782SGreg Clayton eMenuID_ProcessLaunch, 411444d93782SGreg Clayton eMenuID_ProcessContinue, 411544d93782SGreg Clayton eMenuID_ProcessHalt, 411644d93782SGreg Clayton eMenuID_ProcessKill, 411744d93782SGreg Clayton 411844d93782SGreg Clayton eMenuID_Thread, 411944d93782SGreg Clayton eMenuID_ThreadStepIn, 412044d93782SGreg Clayton eMenuID_ThreadStepOver, 412144d93782SGreg Clayton eMenuID_ThreadStepOut, 412244d93782SGreg Clayton 412344d93782SGreg Clayton eMenuID_View, 412444d93782SGreg Clayton eMenuID_ViewBacktrace, 412544d93782SGreg Clayton eMenuID_ViewRegisters, 412644d93782SGreg Clayton eMenuID_ViewSource, 412744d93782SGreg Clayton eMenuID_ViewVariables, 412844d93782SGreg Clayton 412944d93782SGreg Clayton eMenuID_Help, 413044d93782SGreg Clayton eMenuID_HelpGUIHelp 413144d93782SGreg Clayton }; 413244d93782SGreg Clayton 413344d93782SGreg Clayton ApplicationDelegate (Application &app, Debugger &debugger) : 413444d93782SGreg Clayton WindowDelegate (), 413544d93782SGreg Clayton MenuDelegate (), 413644d93782SGreg Clayton m_app (app), 413744d93782SGreg Clayton m_debugger (debugger) 413844d93782SGreg Clayton { 413944d93782SGreg Clayton } 414044d93782SGreg Clayton 414144d93782SGreg Clayton virtual 414244d93782SGreg Clayton ~ApplicationDelegate () 414344d93782SGreg Clayton { 414444d93782SGreg Clayton } 414544d93782SGreg Clayton virtual bool 414644d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 414744d93782SGreg Clayton { 414844d93782SGreg Clayton return false; // Drawing not handled, let standard window drawing happen 414944d93782SGreg Clayton } 415044d93782SGreg Clayton 415144d93782SGreg Clayton virtual HandleCharResult 415244d93782SGreg Clayton WindowDelegateHandleChar (Window &window, int key) 415344d93782SGreg Clayton { 41545fdb09bbSGreg Clayton switch (key) 415544d93782SGreg Clayton { 41565fdb09bbSGreg Clayton case '\t': 415744d93782SGreg Clayton window.SelectNextWindowAsActive(); 415844d93782SGreg Clayton return eKeyHandled; 41595fdb09bbSGreg Clayton 41605fdb09bbSGreg Clayton case 'h': 41615fdb09bbSGreg Clayton window.CreateHelpSubwindow(); 41625fdb09bbSGreg Clayton return eKeyHandled; 41635fdb09bbSGreg Clayton 41645fdb09bbSGreg Clayton case KEY_ESCAPE: 41655fdb09bbSGreg Clayton return eQuitApplication; 41665fdb09bbSGreg Clayton 41675fdb09bbSGreg Clayton default: 41685fdb09bbSGreg Clayton break; 416944d93782SGreg Clayton } 417044d93782SGreg Clayton return eKeyNotHandled; 417144d93782SGreg Clayton } 417244d93782SGreg Clayton 41735fdb09bbSGreg Clayton 41745fdb09bbSGreg Clayton virtual const char * 41755fdb09bbSGreg Clayton WindowDelegateGetHelpText () 41765fdb09bbSGreg Clayton { 41775fdb09bbSGreg Clayton return "Welcome to the LLDB curses GUI.\n\n" 41785fdb09bbSGreg Clayton "Press the TAB key to change the selected view.\n" 41795fdb09bbSGreg Clayton "Each view has its own keyboard shortcuts, press 'h' to open a dialog to display them.\n\n" 41805fdb09bbSGreg Clayton "Common key bindings for all views:"; 41815fdb09bbSGreg Clayton } 41825fdb09bbSGreg Clayton 41835fdb09bbSGreg Clayton virtual KeyHelp * 41845fdb09bbSGreg Clayton WindowDelegateGetKeyHelp () 41855fdb09bbSGreg Clayton { 41865fdb09bbSGreg Clayton static curses::KeyHelp g_source_view_key_help[] = { 41875fdb09bbSGreg Clayton { '\t', "Select next view" }, 41885fdb09bbSGreg Clayton { 'h', "Show help dialog with view specific key bindings" }, 41895fdb09bbSGreg Clayton { ',', "Page up" }, 41905fdb09bbSGreg Clayton { '.', "Page down" }, 41915fdb09bbSGreg Clayton { KEY_UP, "Select previous" }, 41925fdb09bbSGreg Clayton { KEY_DOWN, "Select next" }, 41935fdb09bbSGreg Clayton { KEY_LEFT, "Unexpand or select parent" }, 41945fdb09bbSGreg Clayton { KEY_RIGHT, "Expand" }, 41955fdb09bbSGreg Clayton { KEY_PPAGE, "Page up" }, 41965fdb09bbSGreg Clayton { KEY_NPAGE, "Page down" }, 41975fdb09bbSGreg Clayton { '\0', NULL } 41985fdb09bbSGreg Clayton }; 41995fdb09bbSGreg Clayton return g_source_view_key_help; 42005fdb09bbSGreg Clayton } 42015fdb09bbSGreg Clayton 420244d93782SGreg Clayton virtual MenuActionResult 420344d93782SGreg Clayton MenuDelegateAction (Menu &menu) 420444d93782SGreg Clayton { 420544d93782SGreg Clayton switch (menu.GetIdentifier()) 420644d93782SGreg Clayton { 420744d93782SGreg Clayton case eMenuID_ThreadStepIn: 420844d93782SGreg Clayton { 420944d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 421044d93782SGreg Clayton if (exe_ctx.HasThreadScope()) 421144d93782SGreg Clayton { 421244d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 421344d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 42144b4b2478SJim Ingham exe_ctx.GetThreadRef().StepIn(true); 421544d93782SGreg Clayton } 421644d93782SGreg Clayton } 421744d93782SGreg Clayton return MenuActionResult::Handled; 421844d93782SGreg Clayton 421944d93782SGreg Clayton case eMenuID_ThreadStepOut: 422044d93782SGreg Clayton { 422144d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 422244d93782SGreg Clayton if (exe_ctx.HasThreadScope()) 422344d93782SGreg Clayton { 422444d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 422544d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 422644d93782SGreg Clayton exe_ctx.GetThreadRef().StepOut(); 422744d93782SGreg Clayton } 422844d93782SGreg Clayton } 422944d93782SGreg Clayton return MenuActionResult::Handled; 423044d93782SGreg Clayton 423144d93782SGreg Clayton case eMenuID_ThreadStepOver: 423244d93782SGreg Clayton { 423344d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 423444d93782SGreg Clayton if (exe_ctx.HasThreadScope()) 423544d93782SGreg Clayton { 423644d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 423744d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 423844d93782SGreg Clayton exe_ctx.GetThreadRef().StepOver(true); 423944d93782SGreg Clayton } 424044d93782SGreg Clayton } 424144d93782SGreg Clayton return MenuActionResult::Handled; 424244d93782SGreg Clayton 424344d93782SGreg Clayton case eMenuID_ProcessContinue: 424444d93782SGreg Clayton { 424544d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 424644d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 424744d93782SGreg Clayton { 424844d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 424944d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 425044d93782SGreg Clayton process->Resume(); 425144d93782SGreg Clayton } 425244d93782SGreg Clayton } 425344d93782SGreg Clayton return MenuActionResult::Handled; 425444d93782SGreg Clayton 425544d93782SGreg Clayton case eMenuID_ProcessKill: 425644d93782SGreg Clayton { 425744d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 425844d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 425944d93782SGreg Clayton { 426044d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 426144d93782SGreg Clayton if (process && process->IsAlive()) 426244d93782SGreg Clayton process->Destroy(); 426344d93782SGreg Clayton } 426444d93782SGreg Clayton } 426544d93782SGreg Clayton return MenuActionResult::Handled; 426644d93782SGreg Clayton 426744d93782SGreg Clayton case eMenuID_ProcessHalt: 426844d93782SGreg Clayton { 426944d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 427044d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 427144d93782SGreg Clayton { 427244d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 427344d93782SGreg Clayton if (process && process->IsAlive()) 427444d93782SGreg Clayton process->Halt(); 427544d93782SGreg Clayton } 427644d93782SGreg Clayton } 427744d93782SGreg Clayton return MenuActionResult::Handled; 427844d93782SGreg Clayton 427944d93782SGreg Clayton case eMenuID_ProcessDetach: 428044d93782SGreg Clayton { 428144d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 428244d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 428344d93782SGreg Clayton { 428444d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 428544d93782SGreg Clayton if (process && process->IsAlive()) 428644d93782SGreg Clayton process->Detach(false); 428744d93782SGreg Clayton } 428844d93782SGreg Clayton } 428944d93782SGreg Clayton return MenuActionResult::Handled; 429044d93782SGreg Clayton 429144d93782SGreg Clayton case eMenuID_Process: 429244d93782SGreg Clayton { 429344d93782SGreg Clayton // Populate the menu with all of the threads if the process is stopped when 429444d93782SGreg Clayton // the Process menu gets selected and is about to display its submenu. 429544d93782SGreg Clayton Menus &submenus = menu.GetSubmenus(); 429644d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 429744d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 429844d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 429944d93782SGreg Clayton { 430044d93782SGreg Clayton if (submenus.size() == 7) 430144d93782SGreg Clayton menu.AddSubmenu (MenuSP (new Menu(Menu::Type::Separator))); 430244d93782SGreg Clayton else if (submenus.size() > 8) 430344d93782SGreg Clayton submenus.erase (submenus.begin() + 8, submenus.end()); 430444d93782SGreg Clayton 430544d93782SGreg Clayton ThreadList &threads = process->GetThreadList(); 430644d93782SGreg Clayton Mutex::Locker locker (threads.GetMutex()); 430744d93782SGreg Clayton size_t num_threads = threads.GetSize(); 430844d93782SGreg Clayton for (size_t i=0; i<num_threads; ++i) 430944d93782SGreg Clayton { 431044d93782SGreg Clayton ThreadSP thread_sp = threads.GetThreadAtIndex(i); 431144d93782SGreg Clayton char menu_char = '\0'; 431244d93782SGreg Clayton if (i < 9) 431344d93782SGreg Clayton menu_char = '1' + i; 431444d93782SGreg Clayton StreamString thread_menu_title; 431544d93782SGreg Clayton thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID()); 431644d93782SGreg Clayton const char *thread_name = thread_sp->GetName(); 431744d93782SGreg Clayton if (thread_name && thread_name[0]) 431844d93782SGreg Clayton thread_menu_title.Printf (" %s", thread_name); 431944d93782SGreg Clayton else 432044d93782SGreg Clayton { 432144d93782SGreg Clayton const char *queue_name = thread_sp->GetQueueName(); 432244d93782SGreg Clayton if (queue_name && queue_name[0]) 432344d93782SGreg Clayton thread_menu_title.Printf (" %s", queue_name); 432444d93782SGreg Clayton } 432544d93782SGreg Clayton menu.AddSubmenu (MenuSP (new Menu(thread_menu_title.GetString().c_str(), NULL, menu_char, thread_sp->GetID()))); 432644d93782SGreg Clayton } 432744d93782SGreg Clayton } 432844d93782SGreg Clayton else if (submenus.size() > 7) 432944d93782SGreg Clayton { 433044d93782SGreg Clayton // Remove the separator and any other thread submenu items 433144d93782SGreg Clayton // that were previously added 433244d93782SGreg Clayton submenus.erase (submenus.begin() + 7, submenus.end()); 433344d93782SGreg Clayton } 433444d93782SGreg Clayton // Since we are adding and removing items we need to recalculate the name lengths 433544d93782SGreg Clayton menu.RecalculateNameLengths(); 433644d93782SGreg Clayton } 433744d93782SGreg Clayton return MenuActionResult::Handled; 433844d93782SGreg Clayton 433944d93782SGreg Clayton case eMenuID_ViewVariables: 434044d93782SGreg Clayton { 434144d93782SGreg Clayton WindowSP main_window_sp = m_app.GetMainWindow(); 434244d93782SGreg Clayton WindowSP source_window_sp = main_window_sp->FindSubWindow("Source"); 434344d93782SGreg Clayton WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables"); 434444d93782SGreg Clayton WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers"); 434544d93782SGreg Clayton const Rect source_bounds = source_window_sp->GetBounds(); 434644d93782SGreg Clayton 434744d93782SGreg Clayton if (variables_window_sp) 434844d93782SGreg Clayton { 434944d93782SGreg Clayton const Rect variables_bounds = variables_window_sp->GetBounds(); 435044d93782SGreg Clayton 435144d93782SGreg Clayton main_window_sp->RemoveSubWindow(variables_window_sp.get()); 435244d93782SGreg Clayton 435344d93782SGreg Clayton if (registers_window_sp) 435444d93782SGreg Clayton { 435544d93782SGreg Clayton // We have a registers window, so give all the area back to the registers window 435644d93782SGreg Clayton Rect registers_bounds = variables_bounds; 435744d93782SGreg Clayton registers_bounds.size.width = source_bounds.size.width; 435844d93782SGreg Clayton registers_window_sp->SetBounds(registers_bounds); 435944d93782SGreg Clayton } 436044d93782SGreg Clayton else 436144d93782SGreg Clayton { 436244d93782SGreg Clayton // We have no registers window showing so give the bottom 436344d93782SGreg Clayton // area back to the source view 436444d93782SGreg Clayton source_window_sp->Resize (source_bounds.size.width, 436544d93782SGreg Clayton source_bounds.size.height + variables_bounds.size.height); 436644d93782SGreg Clayton } 436744d93782SGreg Clayton } 436844d93782SGreg Clayton else 436944d93782SGreg Clayton { 437044d93782SGreg Clayton Rect new_variables_rect; 437144d93782SGreg Clayton if (registers_window_sp) 437244d93782SGreg Clayton { 437344d93782SGreg Clayton // We have a registers window so split the area of the registers 437444d93782SGreg Clayton // window into two columns where the left hand side will be the 437544d93782SGreg Clayton // variables and the right hand side will be the registers 437644d93782SGreg Clayton const Rect variables_bounds = registers_window_sp->GetBounds(); 437744d93782SGreg Clayton Rect new_registers_rect; 437844d93782SGreg Clayton variables_bounds.VerticalSplitPercentage (0.50, new_variables_rect, new_registers_rect); 437944d93782SGreg Clayton registers_window_sp->SetBounds (new_registers_rect); 438044d93782SGreg Clayton } 438144d93782SGreg Clayton else 438244d93782SGreg Clayton { 438344d93782SGreg Clayton // No variables window, grab the bottom part of the source window 438444d93782SGreg Clayton Rect new_source_rect; 438544d93782SGreg Clayton source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_variables_rect); 438644d93782SGreg Clayton source_window_sp->SetBounds (new_source_rect); 438744d93782SGreg Clayton } 438844d93782SGreg Clayton WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Variables", 438944d93782SGreg Clayton new_variables_rect, 439044d93782SGreg Clayton false); 439144d93782SGreg Clayton new_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger))); 439244d93782SGreg Clayton } 439344d93782SGreg Clayton touchwin(stdscr); 439444d93782SGreg Clayton } 439544d93782SGreg Clayton return MenuActionResult::Handled; 439644d93782SGreg Clayton 439744d93782SGreg Clayton case eMenuID_ViewRegisters: 439844d93782SGreg Clayton { 439944d93782SGreg Clayton WindowSP main_window_sp = m_app.GetMainWindow(); 440044d93782SGreg Clayton WindowSP source_window_sp = main_window_sp->FindSubWindow("Source"); 440144d93782SGreg Clayton WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables"); 440244d93782SGreg Clayton WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers"); 440344d93782SGreg Clayton const Rect source_bounds = source_window_sp->GetBounds(); 440444d93782SGreg Clayton 440544d93782SGreg Clayton if (registers_window_sp) 440644d93782SGreg Clayton { 440744d93782SGreg Clayton if (variables_window_sp) 440844d93782SGreg Clayton { 440944d93782SGreg Clayton const Rect variables_bounds = variables_window_sp->GetBounds(); 441044d93782SGreg Clayton 441144d93782SGreg Clayton // We have a variables window, so give all the area back to the variables window 441244d93782SGreg Clayton variables_window_sp->Resize (variables_bounds.size.width + registers_window_sp->GetWidth(), 441344d93782SGreg Clayton variables_bounds.size.height); 441444d93782SGreg Clayton } 441544d93782SGreg Clayton else 441644d93782SGreg Clayton { 441744d93782SGreg Clayton // We have no variables window showing so give the bottom 441844d93782SGreg Clayton // area back to the source view 441944d93782SGreg Clayton source_window_sp->Resize (source_bounds.size.width, 442044d93782SGreg Clayton source_bounds.size.height + registers_window_sp->GetHeight()); 442144d93782SGreg Clayton } 442244d93782SGreg Clayton main_window_sp->RemoveSubWindow(registers_window_sp.get()); 442344d93782SGreg Clayton } 442444d93782SGreg Clayton else 442544d93782SGreg Clayton { 442644d93782SGreg Clayton Rect new_regs_rect; 442744d93782SGreg Clayton if (variables_window_sp) 442844d93782SGreg Clayton { 442944d93782SGreg Clayton // We have a variables window, split it into two columns 443044d93782SGreg Clayton // where the left hand side will be the variables and the 443144d93782SGreg Clayton // right hand side will be the registers 443244d93782SGreg Clayton const Rect variables_bounds = variables_window_sp->GetBounds(); 443344d93782SGreg Clayton Rect new_vars_rect; 443444d93782SGreg Clayton variables_bounds.VerticalSplitPercentage (0.50, new_vars_rect, new_regs_rect); 443544d93782SGreg Clayton variables_window_sp->SetBounds (new_vars_rect); 443644d93782SGreg Clayton } 443744d93782SGreg Clayton else 443844d93782SGreg Clayton { 443944d93782SGreg Clayton // No registers window, grab the bottom part of the source window 444044d93782SGreg Clayton Rect new_source_rect; 444144d93782SGreg Clayton source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_regs_rect); 444244d93782SGreg Clayton source_window_sp->SetBounds (new_source_rect); 444344d93782SGreg Clayton } 444444d93782SGreg Clayton WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Registers", 444544d93782SGreg Clayton new_regs_rect, 444644d93782SGreg Clayton false); 444744d93782SGreg Clayton new_window_sp->SetDelegate (WindowDelegateSP(new RegistersWindowDelegate(m_debugger))); 444844d93782SGreg Clayton } 444944d93782SGreg Clayton touchwin(stdscr); 445044d93782SGreg Clayton } 445144d93782SGreg Clayton return MenuActionResult::Handled; 445244d93782SGreg Clayton 445344d93782SGreg Clayton case eMenuID_HelpGUIHelp: 44545fdb09bbSGreg Clayton m_app.GetMainWindow ()->CreateHelpSubwindow(); 445544d93782SGreg Clayton return MenuActionResult::Handled; 445644d93782SGreg Clayton 445744d93782SGreg Clayton default: 445844d93782SGreg Clayton break; 445944d93782SGreg Clayton } 446044d93782SGreg Clayton 446144d93782SGreg Clayton return MenuActionResult::NotHandled; 446244d93782SGreg Clayton } 446344d93782SGreg Clayton protected: 446444d93782SGreg Clayton Application &m_app; 446544d93782SGreg Clayton Debugger &m_debugger; 446644d93782SGreg Clayton }; 446744d93782SGreg Clayton 446844d93782SGreg Clayton 446944d93782SGreg Clayton class StatusBarWindowDelegate : public WindowDelegate 447044d93782SGreg Clayton { 447144d93782SGreg Clayton public: 447244d93782SGreg Clayton StatusBarWindowDelegate (Debugger &debugger) : 447344d93782SGreg Clayton m_debugger (debugger) 447444d93782SGreg Clayton { 447544d93782SGreg Clayton } 447644d93782SGreg Clayton 447744d93782SGreg Clayton virtual 447844d93782SGreg Clayton ~StatusBarWindowDelegate () 447944d93782SGreg Clayton { 448044d93782SGreg Clayton } 448144d93782SGreg Clayton virtual bool 448244d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 448344d93782SGreg Clayton { 448444d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 448544d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 448644d93782SGreg Clayton Thread *thread = exe_ctx.GetThreadPtr(); 448744d93782SGreg Clayton StackFrame *frame = exe_ctx.GetFramePtr(); 448844d93782SGreg Clayton window.Erase(); 448944d93782SGreg Clayton window.SetBackground(2); 449044d93782SGreg Clayton window.MoveCursor (0, 0); 449144d93782SGreg Clayton if (process) 449244d93782SGreg Clayton { 449344d93782SGreg Clayton const StateType state = process->GetState(); 449444d93782SGreg Clayton window.Printf ("Process: %5" PRIu64 " %10s", process->GetID(), StateAsCString(state)); 449544d93782SGreg Clayton 449644d93782SGreg Clayton if (StateIsStoppedState(state, true)) 449744d93782SGreg Clayton { 4498*5b031ebcSEd Maste StreamString strm; 4499*5b031ebcSEd Maste const char *format = "Thread: ${thread.id%tid}"; 4500*5b031ebcSEd Maste if (thread && Debugger::FormatPrompt (format, NULL, &exe_ctx, NULL, strm)) 4501*5b031ebcSEd Maste { 450244d93782SGreg Clayton window.MoveCursor (40, 0); 4503*5b031ebcSEd Maste window.PutCStringTruncated(strm.GetString().c_str(), 1); 4504*5b031ebcSEd Maste } 450544d93782SGreg Clayton 450644d93782SGreg Clayton window.MoveCursor (60, 0); 450744d93782SGreg Clayton if (frame) 450844d93782SGreg Clayton window.Printf ("Frame: %3u PC = 0x%16.16" PRIx64, frame->GetFrameIndex(), frame->GetFrameCodeAddress().GetOpcodeLoadAddress (exe_ctx.GetTargetPtr())); 450944d93782SGreg Clayton } 451044d93782SGreg Clayton else if (state == eStateExited) 451144d93782SGreg Clayton { 451244d93782SGreg Clayton const char *exit_desc = process->GetExitDescription(); 451344d93782SGreg Clayton const int exit_status = process->GetExitStatus(); 451444d93782SGreg Clayton if (exit_desc && exit_desc[0]) 451544d93782SGreg Clayton window.Printf (" with status = %i (%s)", exit_status, exit_desc); 451644d93782SGreg Clayton else 451744d93782SGreg Clayton window.Printf (" with status = %i", exit_status); 451844d93782SGreg Clayton } 451944d93782SGreg Clayton } 452044d93782SGreg Clayton window.DeferredRefresh(); 452144d93782SGreg Clayton return true; 452244d93782SGreg Clayton } 452344d93782SGreg Clayton 452444d93782SGreg Clayton protected: 452544d93782SGreg Clayton Debugger &m_debugger; 452644d93782SGreg Clayton }; 452744d93782SGreg Clayton 452844d93782SGreg Clayton class SourceFileWindowDelegate : public WindowDelegate 452944d93782SGreg Clayton { 453044d93782SGreg Clayton public: 453144d93782SGreg Clayton SourceFileWindowDelegate (Debugger &debugger) : 453244d93782SGreg Clayton WindowDelegate (), 453344d93782SGreg Clayton m_debugger (debugger), 453444d93782SGreg Clayton m_sc (), 453544d93782SGreg Clayton m_file_sp (), 453644d93782SGreg Clayton m_disassembly_scope (NULL), 453744d93782SGreg Clayton m_disassembly_sp (), 453844d93782SGreg Clayton m_disassembly_range (), 4539ec990867SGreg Clayton m_title (), 454044d93782SGreg Clayton m_line_width (4), 454144d93782SGreg Clayton m_selected_line (0), 454244d93782SGreg Clayton m_pc_line (0), 454344d93782SGreg Clayton m_stop_id (0), 454444d93782SGreg Clayton m_frame_idx (UINT32_MAX), 454544d93782SGreg Clayton m_first_visible_line (0), 454644d93782SGreg Clayton m_min_x (0), 454744d93782SGreg Clayton m_min_y (0), 454844d93782SGreg Clayton m_max_x (0), 454944d93782SGreg Clayton m_max_y (0) 455044d93782SGreg Clayton { 455144d93782SGreg Clayton } 455244d93782SGreg Clayton 455344d93782SGreg Clayton virtual 455444d93782SGreg Clayton ~SourceFileWindowDelegate() 455544d93782SGreg Clayton { 455644d93782SGreg Clayton } 455744d93782SGreg Clayton 455844d93782SGreg Clayton void 455944d93782SGreg Clayton Update (const SymbolContext &sc) 456044d93782SGreg Clayton { 456144d93782SGreg Clayton m_sc = sc; 456244d93782SGreg Clayton } 456344d93782SGreg Clayton 456444d93782SGreg Clayton uint32_t 456544d93782SGreg Clayton NumVisibleLines () const 456644d93782SGreg Clayton { 456744d93782SGreg Clayton return m_max_y - m_min_y; 456844d93782SGreg Clayton } 456944d93782SGreg Clayton 457044d93782SGreg Clayton virtual const char * 457144d93782SGreg Clayton WindowDelegateGetHelpText () 457244d93782SGreg Clayton { 457344d93782SGreg Clayton return "Source/Disassembly window keyboard shortcuts:"; 457444d93782SGreg Clayton } 457544d93782SGreg Clayton 457644d93782SGreg Clayton virtual KeyHelp * 457744d93782SGreg Clayton WindowDelegateGetKeyHelp () 457844d93782SGreg Clayton { 457944d93782SGreg Clayton static curses::KeyHelp g_source_view_key_help[] = { 458044d93782SGreg Clayton { KEY_RETURN, "Run to selected line with one shot breakpoint" }, 458144d93782SGreg Clayton { KEY_UP, "Select previous source line" }, 458244d93782SGreg Clayton { KEY_DOWN, "Select next source line" }, 458344d93782SGreg Clayton { KEY_PPAGE, "Page up" }, 458444d93782SGreg Clayton { KEY_NPAGE, "Page down" }, 458544d93782SGreg Clayton { 'b', "Set breakpoint on selected source/disassembly line" }, 458644d93782SGreg Clayton { 'c', "Continue process" }, 458744d93782SGreg Clayton { 'd', "Detach and resume process" }, 458844d93782SGreg Clayton { 'D', "Detach with process suspended" }, 458944d93782SGreg Clayton { 'h', "Show help dialog" }, 459044d93782SGreg Clayton { 'k', "Kill process" }, 459144d93782SGreg Clayton { 'n', "Step over (source line)" }, 459244d93782SGreg Clayton { 'N', "Step over (single instruction)" }, 459344d93782SGreg Clayton { 'o', "Step out" }, 459444d93782SGreg Clayton { 's', "Step in (source line)" }, 459544d93782SGreg Clayton { 'S', "Step in (single instruction)" }, 459644d93782SGreg Clayton { ',', "Page up" }, 459744d93782SGreg Clayton { '.', "Page down" }, 459844d93782SGreg Clayton { '\0', NULL } 459944d93782SGreg Clayton }; 460044d93782SGreg Clayton return g_source_view_key_help; 460144d93782SGreg Clayton } 460244d93782SGreg Clayton 460344d93782SGreg Clayton virtual bool 460444d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 460544d93782SGreg Clayton { 460644d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 460744d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 460844d93782SGreg Clayton Thread *thread = NULL; 460944d93782SGreg Clayton 461044d93782SGreg Clayton bool update_location = false; 461144d93782SGreg Clayton if (process) 461244d93782SGreg Clayton { 461344d93782SGreg Clayton StateType state = process->GetState(); 461444d93782SGreg Clayton if (StateIsStoppedState(state, true)) 461544d93782SGreg Clayton { 461644d93782SGreg Clayton // We are stopped, so it is ok to 461744d93782SGreg Clayton update_location = true; 461844d93782SGreg Clayton } 461944d93782SGreg Clayton } 462044d93782SGreg Clayton 462144d93782SGreg Clayton m_min_x = 1; 4622ec990867SGreg Clayton m_min_y = 2; 462344d93782SGreg Clayton m_max_x = window.GetMaxX()-1; 462444d93782SGreg Clayton m_max_y = window.GetMaxY()-1; 462544d93782SGreg Clayton 462644d93782SGreg Clayton const uint32_t num_visible_lines = NumVisibleLines(); 462744d93782SGreg Clayton StackFrameSP frame_sp; 462844d93782SGreg Clayton bool set_selected_line_to_pc = false; 462944d93782SGreg Clayton 463044d93782SGreg Clayton if (update_location) 463144d93782SGreg Clayton { 463244d93782SGreg Clayton const bool process_alive = process ? process->IsAlive() : false; 463344d93782SGreg Clayton bool thread_changed = false; 463444d93782SGreg Clayton if (process_alive) 463544d93782SGreg Clayton { 463644d93782SGreg Clayton thread = exe_ctx.GetThreadPtr(); 463744d93782SGreg Clayton if (thread) 463844d93782SGreg Clayton { 463944d93782SGreg Clayton frame_sp = thread->GetSelectedFrame(); 464044d93782SGreg Clayton auto tid = thread->GetID(); 464144d93782SGreg Clayton thread_changed = tid != m_tid; 464244d93782SGreg Clayton m_tid = tid; 464344d93782SGreg Clayton } 464444d93782SGreg Clayton else 464544d93782SGreg Clayton { 464644d93782SGreg Clayton if (m_tid != LLDB_INVALID_THREAD_ID) 464744d93782SGreg Clayton { 464844d93782SGreg Clayton thread_changed = true; 464944d93782SGreg Clayton m_tid = LLDB_INVALID_THREAD_ID; 465044d93782SGreg Clayton } 465144d93782SGreg Clayton } 465244d93782SGreg Clayton } 465344d93782SGreg Clayton const uint32_t stop_id = process ? process->GetStopID() : 0; 465444d93782SGreg Clayton const bool stop_id_changed = stop_id != m_stop_id; 465544d93782SGreg Clayton bool frame_changed = false; 465644d93782SGreg Clayton m_stop_id = stop_id; 4657ec990867SGreg Clayton m_title.Clear(); 465844d93782SGreg Clayton if (frame_sp) 465944d93782SGreg Clayton { 466044d93782SGreg Clayton m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything); 4661ec990867SGreg Clayton if (m_sc.module_sp) 4662ec990867SGreg Clayton { 4663ec990867SGreg Clayton m_title.Printf("%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString()); 4664ec990867SGreg Clayton ConstString func_name = m_sc.GetFunctionName(); 4665ec990867SGreg Clayton if (func_name) 4666ec990867SGreg Clayton m_title.Printf("`%s", func_name.GetCString()); 4667ec990867SGreg Clayton } 466844d93782SGreg Clayton const uint32_t frame_idx = frame_sp->GetFrameIndex(); 466944d93782SGreg Clayton frame_changed = frame_idx != m_frame_idx; 467044d93782SGreg Clayton m_frame_idx = frame_idx; 467144d93782SGreg Clayton } 467244d93782SGreg Clayton else 467344d93782SGreg Clayton { 467444d93782SGreg Clayton m_sc.Clear(true); 467544d93782SGreg Clayton frame_changed = m_frame_idx != UINT32_MAX; 467644d93782SGreg Clayton m_frame_idx = UINT32_MAX; 467744d93782SGreg Clayton } 467844d93782SGreg Clayton 467944d93782SGreg Clayton const bool context_changed = thread_changed || frame_changed || stop_id_changed; 468044d93782SGreg Clayton 468144d93782SGreg Clayton if (process_alive) 468244d93782SGreg Clayton { 468344d93782SGreg Clayton if (m_sc.line_entry.IsValid()) 468444d93782SGreg Clayton { 468544d93782SGreg Clayton m_pc_line = m_sc.line_entry.line; 468644d93782SGreg Clayton if (m_pc_line != UINT32_MAX) 468744d93782SGreg Clayton --m_pc_line; // Convert to zero based line number... 468844d93782SGreg Clayton // Update the selected line if the stop ID changed... 468944d93782SGreg Clayton if (context_changed) 469044d93782SGreg Clayton m_selected_line = m_pc_line; 469144d93782SGreg Clayton 469244d93782SGreg Clayton if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) 469344d93782SGreg Clayton { 469444d93782SGreg Clayton // Same file, nothing to do, we should either have the 469544d93782SGreg Clayton // lines or not (source file missing) 46963985c8c6SSaleem Abdulrasool if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) 469744d93782SGreg Clayton { 469844d93782SGreg Clayton if (m_selected_line >= m_first_visible_line + num_visible_lines) 469944d93782SGreg Clayton m_first_visible_line = m_selected_line - 10; 470044d93782SGreg Clayton } 470144d93782SGreg Clayton else 470244d93782SGreg Clayton { 470344d93782SGreg Clayton if (m_selected_line > 10) 470444d93782SGreg Clayton m_first_visible_line = m_selected_line - 10; 470544d93782SGreg Clayton else 470644d93782SGreg Clayton m_first_visible_line = 0; 470744d93782SGreg Clayton } 470844d93782SGreg Clayton } 470944d93782SGreg Clayton else 471044d93782SGreg Clayton { 471144d93782SGreg Clayton // File changed, set selected line to the line with the PC 471244d93782SGreg Clayton m_selected_line = m_pc_line; 471344d93782SGreg Clayton m_file_sp = m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file); 471444d93782SGreg Clayton if (m_file_sp) 471544d93782SGreg Clayton { 471644d93782SGreg Clayton const size_t num_lines = m_file_sp->GetNumLines(); 471744d93782SGreg Clayton int m_line_width = 1; 471844d93782SGreg Clayton for (size_t n = num_lines; n >= 10; n = n / 10) 471944d93782SGreg Clayton ++m_line_width; 472044d93782SGreg Clayton 472144d93782SGreg Clayton snprintf (m_line_format, sizeof(m_line_format), " %%%iu ", m_line_width); 472244d93782SGreg Clayton if (num_lines < num_visible_lines || m_selected_line < num_visible_lines) 472344d93782SGreg Clayton m_first_visible_line = 0; 472444d93782SGreg Clayton else 472544d93782SGreg Clayton m_first_visible_line = m_selected_line - 10; 472644d93782SGreg Clayton } 472744d93782SGreg Clayton } 472844d93782SGreg Clayton } 472944d93782SGreg Clayton else 473044d93782SGreg Clayton { 473144d93782SGreg Clayton m_file_sp.reset(); 473244d93782SGreg Clayton } 473344d93782SGreg Clayton 473444d93782SGreg Clayton if (!m_file_sp || m_file_sp->GetNumLines() == 0) 473544d93782SGreg Clayton { 473644d93782SGreg Clayton // Show disassembly 473744d93782SGreg Clayton bool prefer_file_cache = false; 473844d93782SGreg Clayton if (m_sc.function) 473944d93782SGreg Clayton { 474044d93782SGreg Clayton if (m_disassembly_scope != m_sc.function) 474144d93782SGreg Clayton { 474244d93782SGreg Clayton m_disassembly_scope = m_sc.function; 474344d93782SGreg Clayton m_disassembly_sp = m_sc.function->GetInstructions (exe_ctx, NULL, prefer_file_cache); 474444d93782SGreg Clayton if (m_disassembly_sp) 474544d93782SGreg Clayton { 474644d93782SGreg Clayton set_selected_line_to_pc = true; 474744d93782SGreg Clayton m_disassembly_range = m_sc.function->GetAddressRange(); 474844d93782SGreg Clayton } 474944d93782SGreg Clayton else 475044d93782SGreg Clayton { 475144d93782SGreg Clayton m_disassembly_range.Clear(); 475244d93782SGreg Clayton } 475344d93782SGreg Clayton } 475444d93782SGreg Clayton else 475544d93782SGreg Clayton { 475644d93782SGreg Clayton set_selected_line_to_pc = context_changed; 475744d93782SGreg Clayton } 475844d93782SGreg Clayton } 475944d93782SGreg Clayton else if (m_sc.symbol) 476044d93782SGreg Clayton { 476144d93782SGreg Clayton if (m_disassembly_scope != m_sc.symbol) 476244d93782SGreg Clayton { 476344d93782SGreg Clayton m_disassembly_scope = m_sc.symbol; 476444d93782SGreg Clayton m_disassembly_sp = m_sc.symbol->GetInstructions (exe_ctx, NULL, prefer_file_cache); 476544d93782SGreg Clayton if (m_disassembly_sp) 476644d93782SGreg Clayton { 476744d93782SGreg Clayton set_selected_line_to_pc = true; 476844d93782SGreg Clayton m_disassembly_range.GetBaseAddress() = m_sc.symbol->GetAddress(); 476944d93782SGreg Clayton m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize()); 477044d93782SGreg Clayton } 477144d93782SGreg Clayton else 477244d93782SGreg Clayton { 477344d93782SGreg Clayton m_disassembly_range.Clear(); 477444d93782SGreg Clayton } 477544d93782SGreg Clayton } 477644d93782SGreg Clayton else 477744d93782SGreg Clayton { 477844d93782SGreg Clayton set_selected_line_to_pc = context_changed; 477944d93782SGreg Clayton } 478044d93782SGreg Clayton } 478144d93782SGreg Clayton } 478244d93782SGreg Clayton } 478344d93782SGreg Clayton else 478444d93782SGreg Clayton { 478544d93782SGreg Clayton m_pc_line = UINT32_MAX; 478644d93782SGreg Clayton } 478744d93782SGreg Clayton } 478844d93782SGreg Clayton 4789ec990867SGreg Clayton const int window_width = window.GetWidth(); 479044d93782SGreg Clayton window.Erase(); 479144d93782SGreg Clayton window.DrawTitleBox ("Sources"); 4792ec990867SGreg Clayton if (!m_title.GetString().empty()) 4793ec990867SGreg Clayton { 4794ec990867SGreg Clayton window.AttributeOn(A_REVERSE); 4795ec990867SGreg Clayton window.MoveCursor(1, 1); 4796ec990867SGreg Clayton window.PutChar(' '); 4797ec990867SGreg Clayton window.PutCStringTruncated(m_title.GetString().c_str(), 1); 4798ec990867SGreg Clayton int x = window.GetCursorX(); 4799ec990867SGreg Clayton if (x < window_width - 1) 4800ec990867SGreg Clayton { 4801ec990867SGreg Clayton window.Printf ("%*s", window_width - x - 1, ""); 4802ec990867SGreg Clayton } 4803ec990867SGreg Clayton window.AttributeOff(A_REVERSE); 4804ec990867SGreg Clayton } 480544d93782SGreg Clayton 480644d93782SGreg Clayton Target *target = exe_ctx.GetTargetPtr(); 480744d93782SGreg Clayton const size_t num_source_lines = GetNumSourceLines(); 480844d93782SGreg Clayton if (num_source_lines > 0) 480944d93782SGreg Clayton { 481044d93782SGreg Clayton // Display source 481144d93782SGreg Clayton BreakpointLines bp_lines; 481244d93782SGreg Clayton if (target) 481344d93782SGreg Clayton { 481444d93782SGreg Clayton BreakpointList &bp_list = target->GetBreakpointList(); 481544d93782SGreg Clayton const size_t num_bps = bp_list.GetSize(); 481644d93782SGreg Clayton for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx) 481744d93782SGreg Clayton { 481844d93782SGreg Clayton BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx); 481944d93782SGreg Clayton const size_t num_bps_locs = bp_sp->GetNumLocations(); 482044d93782SGreg Clayton for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx) 482144d93782SGreg Clayton { 482244d93782SGreg Clayton BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx); 482344d93782SGreg Clayton LineEntry bp_loc_line_entry; 482444d93782SGreg Clayton if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry (bp_loc_line_entry)) 482544d93782SGreg Clayton { 482644d93782SGreg Clayton if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) 482744d93782SGreg Clayton { 482844d93782SGreg Clayton bp_lines.insert(bp_loc_line_entry.line); 482944d93782SGreg Clayton } 483044d93782SGreg Clayton } 483144d93782SGreg Clayton } 483244d93782SGreg Clayton } 483344d93782SGreg Clayton } 483444d93782SGreg Clayton 483544d93782SGreg Clayton const attr_t selected_highlight_attr = A_REVERSE; 483644d93782SGreg Clayton const attr_t pc_highlight_attr = COLOR_PAIR(1); 483744d93782SGreg Clayton 48383985c8c6SSaleem Abdulrasool for (size_t i=0; i<num_visible_lines; ++i) 483944d93782SGreg Clayton { 484044d93782SGreg Clayton const uint32_t curr_line = m_first_visible_line + i; 484144d93782SGreg Clayton if (curr_line < num_source_lines) 484244d93782SGreg Clayton { 4843ec990867SGreg Clayton const int line_y = m_min_y+i; 484444d93782SGreg Clayton window.MoveCursor(1, line_y); 484544d93782SGreg Clayton const bool is_pc_line = curr_line == m_pc_line; 484644d93782SGreg Clayton const bool line_is_selected = m_selected_line == curr_line; 484744d93782SGreg Clayton // Highlight the line as the PC line first, then if the selected line 484844d93782SGreg Clayton // isn't the same as the PC line, highlight it differently 484944d93782SGreg Clayton attr_t highlight_attr = 0; 485044d93782SGreg Clayton attr_t bp_attr = 0; 485144d93782SGreg Clayton if (is_pc_line) 485244d93782SGreg Clayton highlight_attr = pc_highlight_attr; 485344d93782SGreg Clayton else if (line_is_selected) 485444d93782SGreg Clayton highlight_attr = selected_highlight_attr; 485544d93782SGreg Clayton 485644d93782SGreg Clayton if (bp_lines.find(curr_line+1) != bp_lines.end()) 485744d93782SGreg Clayton bp_attr = COLOR_PAIR(2); 485844d93782SGreg Clayton 485944d93782SGreg Clayton if (bp_attr) 486044d93782SGreg Clayton window.AttributeOn(bp_attr); 486144d93782SGreg Clayton 486244d93782SGreg Clayton window.Printf (m_line_format, curr_line + 1); 486344d93782SGreg Clayton 486444d93782SGreg Clayton if (bp_attr) 486544d93782SGreg Clayton window.AttributeOff(bp_attr); 486644d93782SGreg Clayton 486744d93782SGreg Clayton window.PutChar(ACS_VLINE); 486844d93782SGreg Clayton // Mark the line with the PC with a diamond 486944d93782SGreg Clayton if (is_pc_line) 487044d93782SGreg Clayton window.PutChar(ACS_DIAMOND); 487144d93782SGreg Clayton else 487244d93782SGreg Clayton window.PutChar(' '); 487344d93782SGreg Clayton 487444d93782SGreg Clayton if (highlight_attr) 487544d93782SGreg Clayton window.AttributeOn(highlight_attr); 487644d93782SGreg Clayton const uint32_t line_len = m_file_sp->GetLineLength(curr_line + 1, false); 487744d93782SGreg Clayton if (line_len > 0) 487844d93782SGreg Clayton window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len); 487944d93782SGreg Clayton 488044d93782SGreg Clayton if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0) 488144d93782SGreg Clayton { 488244d93782SGreg Clayton StopInfoSP stop_info_sp; 488344d93782SGreg Clayton if (thread) 488444d93782SGreg Clayton stop_info_sp = thread->GetStopInfo(); 488544d93782SGreg Clayton if (stop_info_sp) 488644d93782SGreg Clayton { 488744d93782SGreg Clayton const char *stop_description = stop_info_sp->GetDescription(); 488844d93782SGreg Clayton if (stop_description && stop_description[0]) 488944d93782SGreg Clayton { 489044d93782SGreg Clayton size_t stop_description_len = strlen(stop_description); 4891ec990867SGreg Clayton int desc_x = window_width - stop_description_len - 16; 489244d93782SGreg Clayton window.Printf ("%*s", desc_x - window.GetCursorX(), ""); 4893ec990867SGreg Clayton //window.MoveCursor(window_width - stop_description_len - 15, line_y); 489444d93782SGreg Clayton window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description); 489544d93782SGreg Clayton } 489644d93782SGreg Clayton } 489744d93782SGreg Clayton else 489844d93782SGreg Clayton { 4899ec990867SGreg Clayton window.Printf ("%*s", window_width - window.GetCursorX() - 1, ""); 490044d93782SGreg Clayton } 490144d93782SGreg Clayton } 490244d93782SGreg Clayton if (highlight_attr) 490344d93782SGreg Clayton window.AttributeOff(highlight_attr); 490444d93782SGreg Clayton 490544d93782SGreg Clayton } 490644d93782SGreg Clayton else 490744d93782SGreg Clayton { 490844d93782SGreg Clayton break; 490944d93782SGreg Clayton } 491044d93782SGreg Clayton } 491144d93782SGreg Clayton } 491244d93782SGreg Clayton else 491344d93782SGreg Clayton { 491444d93782SGreg Clayton size_t num_disassembly_lines = GetNumDisassemblyLines(); 491544d93782SGreg Clayton if (num_disassembly_lines > 0) 491644d93782SGreg Clayton { 491744d93782SGreg Clayton // Display disassembly 491844d93782SGreg Clayton BreakpointAddrs bp_file_addrs; 491944d93782SGreg Clayton Target *target = exe_ctx.GetTargetPtr(); 492044d93782SGreg Clayton if (target) 492144d93782SGreg Clayton { 492244d93782SGreg Clayton BreakpointList &bp_list = target->GetBreakpointList(); 492344d93782SGreg Clayton const size_t num_bps = bp_list.GetSize(); 492444d93782SGreg Clayton for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx) 492544d93782SGreg Clayton { 492644d93782SGreg Clayton BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx); 492744d93782SGreg Clayton const size_t num_bps_locs = bp_sp->GetNumLocations(); 492844d93782SGreg Clayton for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx) 492944d93782SGreg Clayton { 493044d93782SGreg Clayton BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx); 493144d93782SGreg Clayton LineEntry bp_loc_line_entry; 493244d93782SGreg Clayton const lldb::addr_t file_addr = bp_loc_sp->GetAddress().GetFileAddress(); 493344d93782SGreg Clayton if (file_addr != LLDB_INVALID_ADDRESS) 493444d93782SGreg Clayton { 493544d93782SGreg Clayton if (m_disassembly_range.ContainsFileAddress(file_addr)) 493644d93782SGreg Clayton bp_file_addrs.insert(file_addr); 493744d93782SGreg Clayton } 493844d93782SGreg Clayton } 493944d93782SGreg Clayton } 494044d93782SGreg Clayton } 494144d93782SGreg Clayton 494244d93782SGreg Clayton const attr_t selected_highlight_attr = A_REVERSE; 494344d93782SGreg Clayton const attr_t pc_highlight_attr = COLOR_PAIR(1); 494444d93782SGreg Clayton 494544d93782SGreg Clayton StreamString strm; 494644d93782SGreg Clayton 494744d93782SGreg Clayton InstructionList &insts = m_disassembly_sp->GetInstructionList(); 494844d93782SGreg Clayton Address pc_address; 494944d93782SGreg Clayton 495044d93782SGreg Clayton if (frame_sp) 495144d93782SGreg Clayton pc_address = frame_sp->GetFrameCodeAddress(); 495244d93782SGreg Clayton const uint32_t pc_idx = pc_address.IsValid() ? insts.GetIndexOfInstructionAtAddress (pc_address) : UINT32_MAX; 495344d93782SGreg Clayton if (set_selected_line_to_pc) 495444d93782SGreg Clayton { 495544d93782SGreg Clayton m_selected_line = pc_idx; 495644d93782SGreg Clayton } 495744d93782SGreg Clayton 495844d93782SGreg Clayton const uint32_t non_visible_pc_offset = (num_visible_lines / 5); 49593985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines) 496044d93782SGreg Clayton m_first_visible_line = 0; 496144d93782SGreg Clayton 496244d93782SGreg Clayton if (pc_idx < num_disassembly_lines) 496344d93782SGreg Clayton { 49643985c8c6SSaleem Abdulrasool if (pc_idx < static_cast<uint32_t>(m_first_visible_line) || 496544d93782SGreg Clayton pc_idx >= m_first_visible_line + num_visible_lines) 496644d93782SGreg Clayton m_first_visible_line = pc_idx - non_visible_pc_offset; 496744d93782SGreg Clayton } 496844d93782SGreg Clayton 496944d93782SGreg Clayton for (size_t i=0; i<num_visible_lines; ++i) 497044d93782SGreg Clayton { 497144d93782SGreg Clayton const uint32_t inst_idx = m_first_visible_line + i; 497244d93782SGreg Clayton Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get(); 497344d93782SGreg Clayton if (!inst) 497444d93782SGreg Clayton break; 497544d93782SGreg Clayton 4976ec990867SGreg Clayton const int line_y = m_min_y+i; 4977ec990867SGreg Clayton window.MoveCursor(1, line_y); 497844d93782SGreg Clayton const bool is_pc_line = frame_sp && inst_idx == pc_idx; 497944d93782SGreg Clayton const bool line_is_selected = m_selected_line == inst_idx; 498044d93782SGreg Clayton // Highlight the line as the PC line first, then if the selected line 498144d93782SGreg Clayton // isn't the same as the PC line, highlight it differently 498244d93782SGreg Clayton attr_t highlight_attr = 0; 498344d93782SGreg Clayton attr_t bp_attr = 0; 498444d93782SGreg Clayton if (is_pc_line) 498544d93782SGreg Clayton highlight_attr = pc_highlight_attr; 498644d93782SGreg Clayton else if (line_is_selected) 498744d93782SGreg Clayton highlight_attr = selected_highlight_attr; 498844d93782SGreg Clayton 498944d93782SGreg Clayton if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) != bp_file_addrs.end()) 499044d93782SGreg Clayton bp_attr = COLOR_PAIR(2); 499144d93782SGreg Clayton 499244d93782SGreg Clayton if (bp_attr) 499344d93782SGreg Clayton window.AttributeOn(bp_attr); 499444d93782SGreg Clayton 4995324a1036SSaleem Abdulrasool window.Printf (" 0x%16.16llx ", 4996324a1036SSaleem Abdulrasool static_cast<unsigned long long>(inst->GetAddress().GetLoadAddress(target))); 499744d93782SGreg Clayton 499844d93782SGreg Clayton if (bp_attr) 499944d93782SGreg Clayton window.AttributeOff(bp_attr); 500044d93782SGreg Clayton 500144d93782SGreg Clayton window.PutChar(ACS_VLINE); 500244d93782SGreg Clayton // Mark the line with the PC with a diamond 500344d93782SGreg Clayton if (is_pc_line) 500444d93782SGreg Clayton window.PutChar(ACS_DIAMOND); 500544d93782SGreg Clayton else 500644d93782SGreg Clayton window.PutChar(' '); 500744d93782SGreg Clayton 500844d93782SGreg Clayton if (highlight_attr) 500944d93782SGreg Clayton window.AttributeOn(highlight_attr); 501044d93782SGreg Clayton 501144d93782SGreg Clayton const char *mnemonic = inst->GetMnemonic(&exe_ctx); 501244d93782SGreg Clayton const char *operands = inst->GetOperands(&exe_ctx); 501344d93782SGreg Clayton const char *comment = inst->GetComment(&exe_ctx); 501444d93782SGreg Clayton 501544d93782SGreg Clayton if (mnemonic && mnemonic[0] == '\0') 501644d93782SGreg Clayton mnemonic = NULL; 501744d93782SGreg Clayton if (operands && operands[0] == '\0') 501844d93782SGreg Clayton operands = NULL; 501944d93782SGreg Clayton if (comment && comment[0] == '\0') 502044d93782SGreg Clayton comment = NULL; 502144d93782SGreg Clayton 502244d93782SGreg Clayton strm.Clear(); 502344d93782SGreg Clayton 502444d93782SGreg Clayton if (mnemonic && operands && comment) 502544d93782SGreg Clayton strm.Printf ("%-8s %-25s ; %s", mnemonic, operands, comment); 502644d93782SGreg Clayton else if (mnemonic && operands) 502744d93782SGreg Clayton strm.Printf ("%-8s %s", mnemonic, operands); 502844d93782SGreg Clayton else if (mnemonic) 502944d93782SGreg Clayton strm.Printf ("%s", mnemonic); 503044d93782SGreg Clayton 503144d93782SGreg Clayton int right_pad = 1; 503244d93782SGreg Clayton window.PutCStringTruncated(strm.GetString().c_str(), right_pad); 503344d93782SGreg Clayton 503444d93782SGreg Clayton if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0) 503544d93782SGreg Clayton { 503644d93782SGreg Clayton StopInfoSP stop_info_sp; 503744d93782SGreg Clayton if (thread) 503844d93782SGreg Clayton stop_info_sp = thread->GetStopInfo(); 503944d93782SGreg Clayton if (stop_info_sp) 504044d93782SGreg Clayton { 504144d93782SGreg Clayton const char *stop_description = stop_info_sp->GetDescription(); 504244d93782SGreg Clayton if (stop_description && stop_description[0]) 504344d93782SGreg Clayton { 504444d93782SGreg Clayton size_t stop_description_len = strlen(stop_description); 5045ec990867SGreg Clayton int desc_x = window_width - stop_description_len - 16; 504644d93782SGreg Clayton window.Printf ("%*s", desc_x - window.GetCursorX(), ""); 5047ec990867SGreg Clayton //window.MoveCursor(window_width - stop_description_len - 15, line_y); 504844d93782SGreg Clayton window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description); 504944d93782SGreg Clayton } 505044d93782SGreg Clayton } 505144d93782SGreg Clayton else 505244d93782SGreg Clayton { 5053ec990867SGreg Clayton window.Printf ("%*s", window_width - window.GetCursorX() - 1, ""); 505444d93782SGreg Clayton } 505544d93782SGreg Clayton } 505644d93782SGreg Clayton if (highlight_attr) 505744d93782SGreg Clayton window.AttributeOff(highlight_attr); 505844d93782SGreg Clayton } 505944d93782SGreg Clayton } 506044d93782SGreg Clayton } 506144d93782SGreg Clayton window.DeferredRefresh(); 506244d93782SGreg Clayton return true; // Drawing handled 506344d93782SGreg Clayton } 506444d93782SGreg Clayton 506544d93782SGreg Clayton size_t 506644d93782SGreg Clayton GetNumLines () 506744d93782SGreg Clayton { 506844d93782SGreg Clayton size_t num_lines = GetNumSourceLines(); 506944d93782SGreg Clayton if (num_lines == 0) 507044d93782SGreg Clayton num_lines = GetNumDisassemblyLines(); 507144d93782SGreg Clayton return num_lines; 507244d93782SGreg Clayton } 507344d93782SGreg Clayton 507444d93782SGreg Clayton size_t 507544d93782SGreg Clayton GetNumSourceLines () const 507644d93782SGreg Clayton { 507744d93782SGreg Clayton if (m_file_sp) 507844d93782SGreg Clayton return m_file_sp->GetNumLines(); 507944d93782SGreg Clayton return 0; 508044d93782SGreg Clayton } 508144d93782SGreg Clayton size_t 508244d93782SGreg Clayton GetNumDisassemblyLines () const 508344d93782SGreg Clayton { 508444d93782SGreg Clayton if (m_disassembly_sp) 508544d93782SGreg Clayton return m_disassembly_sp->GetInstructionList().GetSize(); 508644d93782SGreg Clayton return 0; 508744d93782SGreg Clayton } 508844d93782SGreg Clayton 508944d93782SGreg Clayton virtual HandleCharResult 509044d93782SGreg Clayton WindowDelegateHandleChar (Window &window, int c) 509144d93782SGreg Clayton { 509244d93782SGreg Clayton const uint32_t num_visible_lines = NumVisibleLines(); 509344d93782SGreg Clayton const size_t num_lines = GetNumLines (); 509444d93782SGreg Clayton 509544d93782SGreg Clayton switch (c) 509644d93782SGreg Clayton { 509744d93782SGreg Clayton case ',': 509844d93782SGreg Clayton case KEY_PPAGE: 509944d93782SGreg Clayton // Page up key 51003985c8c6SSaleem Abdulrasool if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines) 510144d93782SGreg Clayton m_first_visible_line -= num_visible_lines; 510244d93782SGreg Clayton else 510344d93782SGreg Clayton m_first_visible_line = 0; 510444d93782SGreg Clayton m_selected_line = m_first_visible_line; 510544d93782SGreg Clayton return eKeyHandled; 510644d93782SGreg Clayton 510744d93782SGreg Clayton case '.': 510844d93782SGreg Clayton case KEY_NPAGE: 510944d93782SGreg Clayton // Page down key 511044d93782SGreg Clayton { 511144d93782SGreg Clayton if (m_first_visible_line + num_visible_lines < num_lines) 511244d93782SGreg Clayton m_first_visible_line += num_visible_lines; 511344d93782SGreg Clayton else if (num_lines < num_visible_lines) 511444d93782SGreg Clayton m_first_visible_line = 0; 511544d93782SGreg Clayton else 511644d93782SGreg Clayton m_first_visible_line = num_lines - num_visible_lines; 511744d93782SGreg Clayton m_selected_line = m_first_visible_line; 511844d93782SGreg Clayton } 511944d93782SGreg Clayton return eKeyHandled; 512044d93782SGreg Clayton 512144d93782SGreg Clayton case KEY_UP: 512244d93782SGreg Clayton if (m_selected_line > 0) 512344d93782SGreg Clayton { 512444d93782SGreg Clayton m_selected_line--; 51253985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_first_visible_line) > m_selected_line) 512644d93782SGreg Clayton m_first_visible_line = m_selected_line; 512744d93782SGreg Clayton } 512844d93782SGreg Clayton return eKeyHandled; 512944d93782SGreg Clayton 513044d93782SGreg Clayton case KEY_DOWN: 513144d93782SGreg Clayton if (m_selected_line + 1 < num_lines) 513244d93782SGreg Clayton { 513344d93782SGreg Clayton m_selected_line++; 513444d93782SGreg Clayton if (m_first_visible_line + num_visible_lines < m_selected_line) 513544d93782SGreg Clayton m_first_visible_line++; 513644d93782SGreg Clayton } 513744d93782SGreg Clayton return eKeyHandled; 513844d93782SGreg Clayton 513944d93782SGreg Clayton case '\r': 514044d93782SGreg Clayton case '\n': 514144d93782SGreg Clayton case KEY_ENTER: 514244d93782SGreg Clayton // Set a breakpoint and run to the line using a one shot breakpoint 514344d93782SGreg Clayton if (GetNumSourceLines() > 0) 514444d93782SGreg Clayton { 514544d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 514644d93782SGreg Clayton if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) 514744d93782SGreg Clayton { 514844d93782SGreg Clayton BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL, // Don't limit the breakpoint to certain modules 514944d93782SGreg Clayton m_file_sp->GetFileSpec(), // Source file 515044d93782SGreg Clayton m_selected_line + 1, // Source line number (m_selected_line is zero based) 515144d93782SGreg Clayton eLazyBoolCalculate, // Check inlines using global setting 515244d93782SGreg Clayton eLazyBoolCalculate, // Skip prologue using global setting, 515344d93782SGreg Clayton false, // internal 515444d93782SGreg Clayton false); // request_hardware 515544d93782SGreg Clayton // Make breakpoint one shot 515644d93782SGreg Clayton bp_sp->GetOptions()->SetOneShot(true); 515744d93782SGreg Clayton exe_ctx.GetProcessRef().Resume(); 515844d93782SGreg Clayton } 515944d93782SGreg Clayton } 516044d93782SGreg Clayton else if (m_selected_line < GetNumDisassemblyLines()) 516144d93782SGreg Clayton { 516244d93782SGreg Clayton const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get(); 516344d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 516444d93782SGreg Clayton if (exe_ctx.HasTargetScope()) 516544d93782SGreg Clayton { 516644d93782SGreg Clayton Address addr = inst->GetAddress(); 516744d93782SGreg Clayton BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr, // lldb_private::Address 516844d93782SGreg Clayton false, // internal 516944d93782SGreg Clayton false); // request_hardware 517044d93782SGreg Clayton // Make breakpoint one shot 517144d93782SGreg Clayton bp_sp->GetOptions()->SetOneShot(true); 517244d93782SGreg Clayton exe_ctx.GetProcessRef().Resume(); 517344d93782SGreg Clayton } 517444d93782SGreg Clayton } 517544d93782SGreg Clayton return eKeyHandled; 517644d93782SGreg Clayton 517744d93782SGreg Clayton case 'b': // 'b' == toggle breakpoint on currently selected line 517844d93782SGreg Clayton if (m_selected_line < GetNumSourceLines()) 517944d93782SGreg Clayton { 518044d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 518144d93782SGreg Clayton if (exe_ctx.HasTargetScope()) 518244d93782SGreg Clayton { 518344d93782SGreg Clayton BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL, // Don't limit the breakpoint to certain modules 518444d93782SGreg Clayton m_file_sp->GetFileSpec(), // Source file 518544d93782SGreg Clayton m_selected_line + 1, // Source line number (m_selected_line is zero based) 518644d93782SGreg Clayton eLazyBoolCalculate, // Check inlines using global setting 518744d93782SGreg Clayton eLazyBoolCalculate, // Skip prologue using global setting, 518844d93782SGreg Clayton false, // internal 518944d93782SGreg Clayton false); // request_hardware 519044d93782SGreg Clayton } 519144d93782SGreg Clayton } 519244d93782SGreg Clayton else if (m_selected_line < GetNumDisassemblyLines()) 519344d93782SGreg Clayton { 519444d93782SGreg Clayton const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get(); 519544d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 519644d93782SGreg Clayton if (exe_ctx.HasTargetScope()) 519744d93782SGreg Clayton { 519844d93782SGreg Clayton Address addr = inst->GetAddress(); 519944d93782SGreg Clayton BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr, // lldb_private::Address 520044d93782SGreg Clayton false, // internal 520144d93782SGreg Clayton false); // request_hardware 520244d93782SGreg Clayton } 520344d93782SGreg Clayton } 520444d93782SGreg Clayton return eKeyHandled; 520544d93782SGreg Clayton 520644d93782SGreg Clayton case 'd': // 'd' == detach and let run 520744d93782SGreg Clayton case 'D': // 'D' == detach and keep stopped 520844d93782SGreg Clayton { 520944d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 521044d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 521144d93782SGreg Clayton exe_ctx.GetProcessRef().Detach(c == 'D'); 521244d93782SGreg Clayton } 521344d93782SGreg Clayton return eKeyHandled; 521444d93782SGreg Clayton 521544d93782SGreg Clayton case 'k': 521644d93782SGreg Clayton // 'k' == kill 521744d93782SGreg Clayton { 521844d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 521944d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 522044d93782SGreg Clayton exe_ctx.GetProcessRef().Destroy(); 522144d93782SGreg Clayton } 522244d93782SGreg Clayton return eKeyHandled; 522344d93782SGreg Clayton 522444d93782SGreg Clayton case 'c': 522544d93782SGreg Clayton // 'c' == continue 522644d93782SGreg Clayton { 522744d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 522844d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 522944d93782SGreg Clayton exe_ctx.GetProcessRef().Resume(); 523044d93782SGreg Clayton } 523144d93782SGreg Clayton return eKeyHandled; 523244d93782SGreg Clayton 523344d93782SGreg Clayton case 'o': 523444d93782SGreg Clayton // 'o' == step out 523544d93782SGreg Clayton { 523644d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 523744d93782SGreg Clayton if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true)) 523844d93782SGreg Clayton { 523944d93782SGreg Clayton exe_ctx.GetThreadRef().StepOut(); 524044d93782SGreg Clayton } 524144d93782SGreg Clayton } 524244d93782SGreg Clayton return eKeyHandled; 524344d93782SGreg Clayton case 'n': // 'n' == step over 524444d93782SGreg Clayton case 'N': // 'N' == step over instruction 524544d93782SGreg Clayton { 524644d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 524744d93782SGreg Clayton if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true)) 524844d93782SGreg Clayton { 524944d93782SGreg Clayton bool source_step = (c == 'n'); 525044d93782SGreg Clayton exe_ctx.GetThreadRef().StepOver(source_step); 525144d93782SGreg Clayton } 525244d93782SGreg Clayton } 525344d93782SGreg Clayton return eKeyHandled; 525444d93782SGreg Clayton case 's': // 's' == step into 525544d93782SGreg Clayton case 'S': // 'S' == step into instruction 525644d93782SGreg Clayton { 525744d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 525844d93782SGreg Clayton if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true)) 525944d93782SGreg Clayton { 526044d93782SGreg Clayton bool source_step = (c == 's'); 52614b4b2478SJim Ingham exe_ctx.GetThreadRef().StepIn(source_step); 526244d93782SGreg Clayton } 526344d93782SGreg Clayton } 526444d93782SGreg Clayton return eKeyHandled; 526544d93782SGreg Clayton 526644d93782SGreg Clayton case 'h': 526744d93782SGreg Clayton window.CreateHelpSubwindow (); 526844d93782SGreg Clayton return eKeyHandled; 526944d93782SGreg Clayton 527044d93782SGreg Clayton default: 527144d93782SGreg Clayton break; 527244d93782SGreg Clayton } 527344d93782SGreg Clayton return eKeyNotHandled; 527444d93782SGreg Clayton } 527544d93782SGreg Clayton 527644d93782SGreg Clayton protected: 527744d93782SGreg Clayton typedef std::set<uint32_t> BreakpointLines; 527844d93782SGreg Clayton typedef std::set<lldb::addr_t> BreakpointAddrs; 527944d93782SGreg Clayton 528044d93782SGreg Clayton Debugger &m_debugger; 528144d93782SGreg Clayton SymbolContext m_sc; 528244d93782SGreg Clayton SourceManager::FileSP m_file_sp; 528344d93782SGreg Clayton SymbolContextScope *m_disassembly_scope; 528444d93782SGreg Clayton lldb::DisassemblerSP m_disassembly_sp; 528544d93782SGreg Clayton AddressRange m_disassembly_range; 5286ec990867SGreg Clayton StreamString m_title; 528744d93782SGreg Clayton lldb::user_id_t m_tid; 528844d93782SGreg Clayton char m_line_format[8]; 528944d93782SGreg Clayton int m_line_width; 529044d93782SGreg Clayton uint32_t m_selected_line; // The selected line 529144d93782SGreg Clayton uint32_t m_pc_line; // The line with the PC 529244d93782SGreg Clayton uint32_t m_stop_id; 529344d93782SGreg Clayton uint32_t m_frame_idx; 529444d93782SGreg Clayton int m_first_visible_line; 529544d93782SGreg Clayton int m_min_x; 529644d93782SGreg Clayton int m_min_y; 529744d93782SGreg Clayton int m_max_x; 529844d93782SGreg Clayton int m_max_y; 529944d93782SGreg Clayton 530044d93782SGreg Clayton }; 530144d93782SGreg Clayton 530244d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = { true }; 530344d93782SGreg Clayton 530444d93782SGreg Clayton IOHandlerCursesGUI::IOHandlerCursesGUI (Debugger &debugger) : 530544d93782SGreg Clayton IOHandler (debugger) 530644d93782SGreg Clayton { 530744d93782SGreg Clayton } 530844d93782SGreg Clayton 530944d93782SGreg Clayton void 531044d93782SGreg Clayton IOHandlerCursesGUI::Activate () 531144d93782SGreg Clayton { 531244d93782SGreg Clayton IOHandler::Activate(); 531344d93782SGreg Clayton if (!m_app_ap) 531444d93782SGreg Clayton { 531544d93782SGreg Clayton m_app_ap.reset (new Application (GetInputFILE(), GetOutputFILE())); 531644d93782SGreg Clayton 531744d93782SGreg Clayton 531844d93782SGreg Clayton // This is both a window and a menu delegate 531944d93782SGreg Clayton std::shared_ptr<ApplicationDelegate> app_delegate_sp(new ApplicationDelegate(*m_app_ap, m_debugger)); 532044d93782SGreg Clayton 532144d93782SGreg Clayton MenuDelegateSP app_menu_delegate_sp = std::static_pointer_cast<MenuDelegate>(app_delegate_sp); 532244d93782SGreg Clayton MenuSP lldb_menu_sp(new Menu("LLDB" , "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB)); 532344d93782SGreg Clayton MenuSP exit_menuitem_sp(new Menu("Exit", NULL, 'x', ApplicationDelegate::eMenuID_LLDBExit)); 532444d93782SGreg Clayton exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit); 532544d93782SGreg Clayton lldb_menu_sp->AddSubmenu (MenuSP (new Menu("About LLDB", NULL, 'a', ApplicationDelegate::eMenuID_LLDBAbout))); 532644d93782SGreg Clayton lldb_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator))); 532744d93782SGreg Clayton lldb_menu_sp->AddSubmenu (exit_menuitem_sp); 532844d93782SGreg Clayton 532944d93782SGreg Clayton MenuSP target_menu_sp(new Menu("Target" ,"F2", KEY_F(2), ApplicationDelegate::eMenuID_Target)); 533044d93782SGreg Clayton target_menu_sp->AddSubmenu (MenuSP (new Menu("Create", NULL, 'c', ApplicationDelegate::eMenuID_TargetCreate))); 533144d93782SGreg Clayton target_menu_sp->AddSubmenu (MenuSP (new Menu("Delete", NULL, 'd', ApplicationDelegate::eMenuID_TargetDelete))); 533244d93782SGreg Clayton 533344d93782SGreg Clayton MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3), ApplicationDelegate::eMenuID_Process)); 533444d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Attach" , NULL, 'a', ApplicationDelegate::eMenuID_ProcessAttach))); 533544d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Detach" , NULL, 'd', ApplicationDelegate::eMenuID_ProcessDetach))); 533644d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Launch" , NULL, 'l', ApplicationDelegate::eMenuID_ProcessLaunch))); 533744d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator))); 533844d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Continue", NULL, 'c', ApplicationDelegate::eMenuID_ProcessContinue))); 533944d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Halt" , NULL, 'h', ApplicationDelegate::eMenuID_ProcessHalt))); 534044d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Kill" , NULL, 'k', ApplicationDelegate::eMenuID_ProcessKill))); 534144d93782SGreg Clayton 534244d93782SGreg Clayton MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4), ApplicationDelegate::eMenuID_Thread)); 534344d93782SGreg Clayton thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step In" , NULL, 'i', ApplicationDelegate::eMenuID_ThreadStepIn))); 534444d93782SGreg Clayton thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Over", NULL, 'v', ApplicationDelegate::eMenuID_ThreadStepOver))); 534544d93782SGreg Clayton thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Out" , NULL, 'o', ApplicationDelegate::eMenuID_ThreadStepOut))); 534644d93782SGreg Clayton 534744d93782SGreg Clayton MenuSP view_menu_sp(new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View)); 534844d93782SGreg Clayton view_menu_sp->AddSubmenu (MenuSP (new Menu("Backtrace", NULL, 'b', ApplicationDelegate::eMenuID_ViewBacktrace))); 534944d93782SGreg Clayton view_menu_sp->AddSubmenu (MenuSP (new Menu("Registers", NULL, 'r', ApplicationDelegate::eMenuID_ViewRegisters))); 535044d93782SGreg Clayton view_menu_sp->AddSubmenu (MenuSP (new Menu("Source" , NULL, 's', ApplicationDelegate::eMenuID_ViewSource))); 535144d93782SGreg Clayton view_menu_sp->AddSubmenu (MenuSP (new Menu("Variables", NULL, 'v', ApplicationDelegate::eMenuID_ViewVariables))); 535244d93782SGreg Clayton 535344d93782SGreg Clayton MenuSP help_menu_sp(new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help)); 535444d93782SGreg Clayton help_menu_sp->AddSubmenu (MenuSP (new Menu("GUI Help", NULL, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp))); 535544d93782SGreg Clayton 535644d93782SGreg Clayton m_app_ap->Initialize(); 535744d93782SGreg Clayton WindowSP &main_window_sp = m_app_ap->GetMainWindow(); 535844d93782SGreg Clayton 535944d93782SGreg Clayton MenuSP menubar_sp(new Menu(Menu::Type::Bar)); 536044d93782SGreg Clayton menubar_sp->AddSubmenu (lldb_menu_sp); 536144d93782SGreg Clayton menubar_sp->AddSubmenu (target_menu_sp); 536244d93782SGreg Clayton menubar_sp->AddSubmenu (process_menu_sp); 536344d93782SGreg Clayton menubar_sp->AddSubmenu (thread_menu_sp); 536444d93782SGreg Clayton menubar_sp->AddSubmenu (view_menu_sp); 536544d93782SGreg Clayton menubar_sp->AddSubmenu (help_menu_sp); 536644d93782SGreg Clayton menubar_sp->SetDelegate(app_menu_delegate_sp); 536744d93782SGreg Clayton 536844d93782SGreg Clayton Rect content_bounds = main_window_sp->GetFrame(); 536944d93782SGreg Clayton Rect menubar_bounds = content_bounds.MakeMenuBar(); 537044d93782SGreg Clayton Rect status_bounds = content_bounds.MakeStatusBar(); 537144d93782SGreg Clayton Rect source_bounds; 537244d93782SGreg Clayton Rect variables_bounds; 537344d93782SGreg Clayton Rect threads_bounds; 537444d93782SGreg Clayton Rect source_variables_bounds; 537544d93782SGreg Clayton content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds, threads_bounds); 537644d93782SGreg Clayton source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds, variables_bounds); 537744d93782SGreg Clayton 537844d93782SGreg Clayton WindowSP menubar_window_sp = main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false); 537944d93782SGreg Clayton // Let the menubar get keys if the active window doesn't handle the 538044d93782SGreg Clayton // keys that are typed so it can respond to menubar key presses. 538144d93782SGreg Clayton menubar_window_sp->SetCanBeActive(false); // Don't let the menubar become the active window 538244d93782SGreg Clayton menubar_window_sp->SetDelegate(menubar_sp); 538344d93782SGreg Clayton 538444d93782SGreg Clayton WindowSP source_window_sp (main_window_sp->CreateSubWindow("Source", 538544d93782SGreg Clayton source_bounds, 538644d93782SGreg Clayton true)); 538744d93782SGreg Clayton WindowSP variables_window_sp (main_window_sp->CreateSubWindow("Variables", 538844d93782SGreg Clayton variables_bounds, 538944d93782SGreg Clayton false)); 539044d93782SGreg Clayton WindowSP threads_window_sp (main_window_sp->CreateSubWindow("Threads", 539144d93782SGreg Clayton threads_bounds, 539244d93782SGreg Clayton false)); 539344d93782SGreg Clayton WindowSP status_window_sp (main_window_sp->CreateSubWindow("Status", 539444d93782SGreg Clayton status_bounds, 539544d93782SGreg Clayton false)); 539644d93782SGreg Clayton status_window_sp->SetCanBeActive(false); // Don't let the status bar become the active window 539744d93782SGreg Clayton main_window_sp->SetDelegate (std::static_pointer_cast<WindowDelegate>(app_delegate_sp)); 539844d93782SGreg Clayton source_window_sp->SetDelegate (WindowDelegateSP(new SourceFileWindowDelegate(m_debugger))); 539944d93782SGreg Clayton variables_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger))); 5400ec990867SGreg Clayton TreeDelegateSP thread_delegate_sp (new ThreadsTreeDelegate(m_debugger)); 540144d93782SGreg Clayton threads_window_sp->SetDelegate (WindowDelegateSP(new TreeWindowDelegate(m_debugger, thread_delegate_sp))); 540244d93782SGreg Clayton status_window_sp->SetDelegate (WindowDelegateSP(new StatusBarWindowDelegate(m_debugger))); 540344d93782SGreg Clayton 54045fdb09bbSGreg Clayton // Show the main help window once the first time the curses GUI is launched 54055fdb09bbSGreg Clayton static bool g_showed_help = false; 54065fdb09bbSGreg Clayton if (!g_showed_help) 54075fdb09bbSGreg Clayton { 54085fdb09bbSGreg Clayton g_showed_help = true; 54095fdb09bbSGreg Clayton main_window_sp->CreateHelpSubwindow(); 54105fdb09bbSGreg Clayton } 54115fdb09bbSGreg Clayton 541244d93782SGreg Clayton init_pair (1, COLOR_WHITE , COLOR_BLUE ); 541344d93782SGreg Clayton init_pair (2, COLOR_BLACK , COLOR_WHITE ); 541444d93782SGreg Clayton init_pair (3, COLOR_MAGENTA , COLOR_WHITE ); 541544d93782SGreg Clayton init_pair (4, COLOR_MAGENTA , COLOR_BLACK ); 541644d93782SGreg Clayton init_pair (5, COLOR_RED , COLOR_BLACK ); 541744d93782SGreg Clayton 541844d93782SGreg Clayton } 541944d93782SGreg Clayton } 542044d93782SGreg Clayton 542144d93782SGreg Clayton void 542244d93782SGreg Clayton IOHandlerCursesGUI::Deactivate () 542344d93782SGreg Clayton { 542444d93782SGreg Clayton m_app_ap->Terminate(); 542544d93782SGreg Clayton } 542644d93782SGreg Clayton 542744d93782SGreg Clayton void 542844d93782SGreg Clayton IOHandlerCursesGUI::Run () 542944d93782SGreg Clayton { 543044d93782SGreg Clayton m_app_ap->Run(m_debugger); 543144d93782SGreg Clayton SetIsDone(true); 543244d93782SGreg Clayton } 543344d93782SGreg Clayton 543444d93782SGreg Clayton 543544d93782SGreg Clayton IOHandlerCursesGUI::~IOHandlerCursesGUI () 543644d93782SGreg Clayton { 543744d93782SGreg Clayton 543844d93782SGreg Clayton } 543944d93782SGreg Clayton 544044d93782SGreg Clayton void 544144d93782SGreg Clayton IOHandlerCursesGUI::Hide () 544244d93782SGreg Clayton { 544344d93782SGreg Clayton } 544444d93782SGreg Clayton 544544d93782SGreg Clayton 544644d93782SGreg Clayton void 544744d93782SGreg Clayton IOHandlerCursesGUI::Refresh () 544844d93782SGreg Clayton { 544944d93782SGreg Clayton } 545044d93782SGreg Clayton 5451e68f5d6bSGreg Clayton void 5452e68f5d6bSGreg Clayton IOHandlerCursesGUI::Cancel () 5453e68f5d6bSGreg Clayton { 5454e68f5d6bSGreg Clayton } 545544d93782SGreg Clayton 545644d93782SGreg Clayton void 545744d93782SGreg Clayton IOHandlerCursesGUI::Interrupt () 545844d93782SGreg Clayton { 545944d93782SGreg Clayton } 546044d93782SGreg Clayton 546144d93782SGreg Clayton 546244d93782SGreg Clayton void 546344d93782SGreg Clayton IOHandlerCursesGUI::GotEOF() 546444d93782SGreg Clayton { 546544d93782SGreg Clayton } 546644d93782SGreg Clayton 5467914b8d98SDeepak Panickal #endif // #ifndef LLDB_DISABLE_CURSES 5468