130fdc8d8SChris Lattner //===-- Debugger.cpp --------------------------------------------*- C++ -*-===// 230fdc8d8SChris Lattner // 330fdc8d8SChris Lattner // The LLVM Compiler Infrastructure 430fdc8d8SChris Lattner // 530fdc8d8SChris Lattner // This file is distributed under the University of Illinois Open Source 630fdc8d8SChris Lattner // License. See LICENSE.TXT for details. 730fdc8d8SChris Lattner // 830fdc8d8SChris Lattner //===----------------------------------------------------------------------===// 930fdc8d8SChris Lattner 104a33d318SGreg Clayton #include "lldb/Core/Debugger.h" 114a33d318SGreg Clayton 124a33d318SGreg Clayton #include <map> 134a33d318SGreg Clayton 144becb37eSEnrico Granata #include "clang/AST/DeclCXX.h" 154becb37eSEnrico Granata #include "clang/AST/Type.h" 164becb37eSEnrico Granata 1730fdc8d8SChris Lattner #include "lldb/lldb-private.h" 1830fdc8d8SChris Lattner #include "lldb/Core/ConnectionFileDescriptor.h" 194a33d318SGreg Clayton #include "lldb/Core/FormatManager.h" 2030fdc8d8SChris Lattner #include "lldb/Core/InputReader.h" 217349bd90SGreg Clayton #include "lldb/Core/RegisterValue.h" 2230fdc8d8SChris Lattner #include "lldb/Core/State.h" 235b52f0c7SJim Ingham #include "lldb/Core/StreamAsynchronousIO.h" 241b654882SGreg Clayton #include "lldb/Core/StreamString.h" 2530fdc8d8SChris Lattner #include "lldb/Core/Timer.h" 264becb37eSEnrico Granata #include "lldb/Core/ValueObject.h" 27a3406614SGreg Clayton #include "lldb/Host/Terminal.h" 286611103cSGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h" 2930fdc8d8SChris Lattner #include "lldb/Target/TargetList.h" 3030fdc8d8SChris Lattner #include "lldb/Target/Process.h" 311b654882SGreg Clayton #include "lldb/Target/RegisterContext.h" 321b654882SGreg Clayton #include "lldb/Target/StopInfo.h" 3330fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 345a31471eSGreg Clayton #include "lldb/Utility/AnsiTerminal.h" 3530fdc8d8SChris Lattner 3630fdc8d8SChris Lattner using namespace lldb; 3730fdc8d8SChris Lattner using namespace lldb_private; 3830fdc8d8SChris Lattner 3930fdc8d8SChris Lattner 401b654882SGreg Clayton static uint32_t g_shared_debugger_refcount = 0; 41ebc1bb27SCaroline Tice static lldb::user_id_t g_unique_id = 1; 42ebc1bb27SCaroline Tice 431b654882SGreg Clayton #pragma mark Static Functions 441b654882SGreg Clayton 451b654882SGreg Clayton static Mutex & 461b654882SGreg Clayton GetDebuggerListMutex () 471b654882SGreg Clayton { 481b654882SGreg Clayton static Mutex g_mutex(Mutex::eMutexTypeRecursive); 491b654882SGreg Clayton return g_mutex; 501b654882SGreg Clayton } 511b654882SGreg Clayton 521b654882SGreg Clayton typedef std::vector<DebuggerSP> DebuggerList; 531b654882SGreg Clayton 541b654882SGreg Clayton static DebuggerList & 551b654882SGreg Clayton GetDebuggerList() 561b654882SGreg Clayton { 571b654882SGreg Clayton // hide the static debugger list inside a singleton accessor to avoid 581b654882SGreg Clayton // global init contructors 591b654882SGreg Clayton static DebuggerList g_list; 601b654882SGreg Clayton return g_list; 611b654882SGreg Clayton } 621b654882SGreg Clayton 631b654882SGreg Clayton 64e372b98dSGreg Clayton static const ConstString & 65e372b98dSGreg Clayton PromptVarName () 66e372b98dSGreg Clayton { 67e372b98dSGreg Clayton static ConstString g_const_string ("prompt"); 68e372b98dSGreg Clayton return g_const_string; 69e372b98dSGreg Clayton } 70e372b98dSGreg Clayton 71e372b98dSGreg Clayton static const ConstString & 72e372b98dSGreg Clayton GetFrameFormatName () 73e372b98dSGreg Clayton { 74e372b98dSGreg Clayton static ConstString g_const_string ("frame-format"); 75e372b98dSGreg Clayton return g_const_string; 76e372b98dSGreg Clayton } 77e372b98dSGreg Clayton 78e372b98dSGreg Clayton static const ConstString & 79e372b98dSGreg Clayton GetThreadFormatName () 80e372b98dSGreg Clayton { 81e372b98dSGreg Clayton static ConstString g_const_string ("thread-format"); 82e372b98dSGreg Clayton return g_const_string; 83e372b98dSGreg Clayton } 84e372b98dSGreg Clayton 85e372b98dSGreg Clayton static const ConstString & 86e372b98dSGreg Clayton ScriptLangVarName () 87e372b98dSGreg Clayton { 88e372b98dSGreg Clayton static ConstString g_const_string ("script-lang"); 89e372b98dSGreg Clayton return g_const_string; 90e372b98dSGreg Clayton } 91e372b98dSGreg Clayton 92e372b98dSGreg Clayton static const ConstString & 93e372b98dSGreg Clayton TermWidthVarName () 94e372b98dSGreg Clayton { 95e372b98dSGreg Clayton static ConstString g_const_string ("term-width"); 96e372b98dSGreg Clayton return g_const_string; 97e372b98dSGreg Clayton } 98e372b98dSGreg Clayton 99e372b98dSGreg Clayton static const ConstString & 100e372b98dSGreg Clayton UseExternalEditorVarName () 101e372b98dSGreg Clayton { 102e372b98dSGreg Clayton static ConstString g_const_string ("use-external-editor"); 103e372b98dSGreg Clayton return g_const_string; 104e372b98dSGreg Clayton } 105e372b98dSGreg Clayton 106e372b98dSGreg Clayton static const ConstString & 107e372b98dSGreg Clayton AutoConfirmName () 108e372b98dSGreg Clayton { 109e372b98dSGreg Clayton static ConstString g_const_string ("auto-confirm"); 110e372b98dSGreg Clayton return g_const_string; 111e372b98dSGreg Clayton } 112e372b98dSGreg Clayton 113e372b98dSGreg Clayton static const ConstString & 114e372b98dSGreg Clayton StopSourceContextBeforeName () 115e372b98dSGreg Clayton { 116e372b98dSGreg Clayton static ConstString g_const_string ("stop-line-count-before"); 117e372b98dSGreg Clayton return g_const_string; 118e372b98dSGreg Clayton } 119e372b98dSGreg Clayton 120e372b98dSGreg Clayton static const ConstString & 121e372b98dSGreg Clayton StopSourceContextAfterName () 122e372b98dSGreg Clayton { 123e372b98dSGreg Clayton static ConstString g_const_string ("stop-line-count-after"); 124e372b98dSGreg Clayton return g_const_string; 125e372b98dSGreg Clayton } 126e372b98dSGreg Clayton 127e372b98dSGreg Clayton static const ConstString & 128e372b98dSGreg Clayton StopDisassemblyCountName () 129e372b98dSGreg Clayton { 130e372b98dSGreg Clayton static ConstString g_const_string ("stop-disassembly-count"); 131e372b98dSGreg Clayton return g_const_string; 132e372b98dSGreg Clayton } 133e372b98dSGreg Clayton 134e372b98dSGreg Clayton static const ConstString & 135e372b98dSGreg Clayton StopDisassemblyDisplayName () 136e372b98dSGreg Clayton { 137e372b98dSGreg Clayton static ConstString g_const_string ("stop-disassembly-display"); 138e372b98dSGreg Clayton return g_const_string; 139e372b98dSGreg Clayton } 140e372b98dSGreg Clayton 141e372b98dSGreg Clayton OptionEnumValueElement 142e372b98dSGreg Clayton DebuggerInstanceSettings::g_show_disassembly_enum_values[] = 143e372b98dSGreg Clayton { 144e372b98dSGreg Clayton { eStopDisassemblyTypeNever, "never", "Never show disassembly when displaying a stop context."}, 145e372b98dSGreg Clayton { eStopDisassemblyTypeNoSource, "no-source", "Show disassembly when there is no source information, or the source file is missing when displaying a stop context."}, 146e372b98dSGreg Clayton { eStopDisassemblyTypeAlways, "always", "Always show disassembly when displaying a stop context."}, 147e372b98dSGreg Clayton { 0, NULL, NULL } 148e372b98dSGreg Clayton }; 149e372b98dSGreg Clayton 150e372b98dSGreg Clayton 151e372b98dSGreg Clayton 1521b654882SGreg Clayton #pragma mark Debugger 1531b654882SGreg Clayton 15499d0faf2SGreg Clayton UserSettingsControllerSP & 15599d0faf2SGreg Clayton Debugger::GetSettingsController () 15699d0faf2SGreg Clayton { 15799d0faf2SGreg Clayton static UserSettingsControllerSP g_settings_controller; 15899d0faf2SGreg Clayton return g_settings_controller; 15999d0faf2SGreg Clayton } 16099d0faf2SGreg Clayton 1612f88aadfSCaroline Tice int 1622f88aadfSCaroline Tice Debugger::TestDebuggerRefCount () 1632f88aadfSCaroline Tice { 1642f88aadfSCaroline Tice return g_shared_debugger_refcount; 1652f88aadfSCaroline Tice } 1662f88aadfSCaroline Tice 16730fdc8d8SChris Lattner void 16830fdc8d8SChris Lattner Debugger::Initialize () 16930fdc8d8SChris Lattner { 1706611103cSGreg Clayton if (g_shared_debugger_refcount == 0) 17199d0faf2SGreg Clayton { 172dbe54508SGreg Clayton lldb_private::Initialize(); 17399d0faf2SGreg Clayton } 1746611103cSGreg Clayton g_shared_debugger_refcount++; 17599d0faf2SGreg Clayton 17630fdc8d8SChris Lattner } 17730fdc8d8SChris Lattner 17830fdc8d8SChris Lattner void 17930fdc8d8SChris Lattner Debugger::Terminate () 18030fdc8d8SChris Lattner { 1816611103cSGreg Clayton if (g_shared_debugger_refcount > 0) 1826611103cSGreg Clayton { 18330fdc8d8SChris Lattner g_shared_debugger_refcount--; 18430fdc8d8SChris Lattner if (g_shared_debugger_refcount == 0) 18530fdc8d8SChris Lattner { 186dbe54508SGreg Clayton lldb_private::WillTerminate(); 187dbe54508SGreg Clayton lldb_private::Terminate(); 1886760a517SCaroline Tice 18999d0faf2SGreg Clayton // Clear our master list of debugger objects 19099d0faf2SGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 19199d0faf2SGreg Clayton GetDebuggerList().clear(); 19230fdc8d8SChris Lattner } 1936760a517SCaroline Tice } 1946760a517SCaroline Tice } 19530fdc8d8SChris Lattner 19620bd37f7SCaroline Tice void 19720bd37f7SCaroline Tice Debugger::SettingsInitialize () 19820bd37f7SCaroline Tice { 19920bd37f7SCaroline Tice static bool g_initialized = false; 20020bd37f7SCaroline Tice 20120bd37f7SCaroline Tice if (!g_initialized) 20220bd37f7SCaroline Tice { 20320bd37f7SCaroline Tice g_initialized = true; 20420bd37f7SCaroline Tice UserSettingsControllerSP &usc = GetSettingsController(); 20520bd37f7SCaroline Tice usc.reset (new SettingsController); 20620bd37f7SCaroline Tice UserSettingsController::InitializeSettingsController (usc, 20720bd37f7SCaroline Tice SettingsController::global_settings_table, 20820bd37f7SCaroline Tice SettingsController::instance_settings_table); 20920bd37f7SCaroline Tice // Now call SettingsInitialize for each settings 'child' of Debugger 21020bd37f7SCaroline Tice Target::SettingsInitialize (); 21120bd37f7SCaroline Tice } 21220bd37f7SCaroline Tice } 21320bd37f7SCaroline Tice 21420bd37f7SCaroline Tice void 21520bd37f7SCaroline Tice Debugger::SettingsTerminate () 21620bd37f7SCaroline Tice { 21720bd37f7SCaroline Tice 21820bd37f7SCaroline Tice // Must call SettingsTerminate() for each settings 'child' of Debugger, before terminating the Debugger's 21920bd37f7SCaroline Tice // Settings. 22020bd37f7SCaroline Tice 22120bd37f7SCaroline Tice Target::SettingsTerminate (); 22220bd37f7SCaroline Tice 22320bd37f7SCaroline Tice // Now terminate the Debugger Settings. 22420bd37f7SCaroline Tice 22520bd37f7SCaroline Tice UserSettingsControllerSP &usc = GetSettingsController(); 22620bd37f7SCaroline Tice UserSettingsController::FinalizeSettingsController (usc); 22720bd37f7SCaroline Tice usc.reset(); 22820bd37f7SCaroline Tice } 22920bd37f7SCaroline Tice 2306611103cSGreg Clayton DebuggerSP 2316611103cSGreg Clayton Debugger::CreateInstance () 2326611103cSGreg Clayton { 2336611103cSGreg Clayton DebuggerSP debugger_sp (new Debugger); 2346611103cSGreg Clayton // Scope for locker 2356611103cSGreg Clayton { 2366611103cSGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 2376611103cSGreg Clayton GetDebuggerList().push_back(debugger_sp); 2386611103cSGreg Clayton } 2396611103cSGreg Clayton return debugger_sp; 2406611103cSGreg Clayton } 2416611103cSGreg Clayton 242e02657b1SCaroline Tice void 2434d122c40SGreg Clayton Debugger::Destroy (DebuggerSP &debugger_sp) 244e02657b1SCaroline Tice { 245e02657b1SCaroline Tice if (debugger_sp.get() == NULL) 246e02657b1SCaroline Tice return; 247e02657b1SCaroline Tice 2488314c525SJim Ingham debugger_sp->Clear(); 2498314c525SJim Ingham 250e02657b1SCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 251e02657b1SCaroline Tice DebuggerList &debugger_list = GetDebuggerList (); 252e02657b1SCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 253e02657b1SCaroline Tice for (pos = debugger_list.begin (); pos != end; ++pos) 254e02657b1SCaroline Tice { 255e02657b1SCaroline Tice if ((*pos).get() == debugger_sp.get()) 256e02657b1SCaroline Tice { 257e02657b1SCaroline Tice debugger_list.erase (pos); 258e02657b1SCaroline Tice return; 259e02657b1SCaroline Tice } 260e02657b1SCaroline Tice } 261e02657b1SCaroline Tice } 262e02657b1SCaroline Tice 2634d122c40SGreg Clayton DebuggerSP 2646611103cSGreg Clayton Debugger::GetSP () 2656611103cSGreg Clayton { 2664d122c40SGreg Clayton // This object contains an instrusive ref count base class so we can 2674d122c40SGreg Clayton // easily make a shared pointer to this object 2684d122c40SGreg Clayton return DebuggerSP (this); 2696611103cSGreg Clayton } 2706611103cSGreg Clayton 2714d122c40SGreg Clayton DebuggerSP 2723df9a8dfSCaroline Tice Debugger::FindDebuggerWithInstanceName (const ConstString &instance_name) 2733df9a8dfSCaroline Tice { 2744d122c40SGreg Clayton DebuggerSP debugger_sp; 2753df9a8dfSCaroline Tice 2763df9a8dfSCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 2773df9a8dfSCaroline Tice DebuggerList &debugger_list = GetDebuggerList(); 2783df9a8dfSCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 2793df9a8dfSCaroline Tice 2803df9a8dfSCaroline Tice for (pos = debugger_list.begin(); pos != end; ++pos) 2813df9a8dfSCaroline Tice { 2823df9a8dfSCaroline Tice if ((*pos).get()->m_instance_name == instance_name) 2833df9a8dfSCaroline Tice { 2843df9a8dfSCaroline Tice debugger_sp = *pos; 2853df9a8dfSCaroline Tice break; 2863df9a8dfSCaroline Tice } 2873df9a8dfSCaroline Tice } 2883df9a8dfSCaroline Tice return debugger_sp; 2893df9a8dfSCaroline Tice } 2906611103cSGreg Clayton 2916611103cSGreg Clayton TargetSP 2926611103cSGreg Clayton Debugger::FindTargetWithProcessID (lldb::pid_t pid) 2936611103cSGreg Clayton { 2944d122c40SGreg Clayton TargetSP target_sp; 2956611103cSGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 2966611103cSGreg Clayton DebuggerList &debugger_list = GetDebuggerList(); 2976611103cSGreg Clayton DebuggerList::iterator pos, end = debugger_list.end(); 2986611103cSGreg Clayton for (pos = debugger_list.begin(); pos != end; ++pos) 2996611103cSGreg Clayton { 3006611103cSGreg Clayton target_sp = (*pos)->GetTargetList().FindTargetWithProcessID (pid); 3016611103cSGreg Clayton if (target_sp) 3026611103cSGreg Clayton break; 3036611103cSGreg Clayton } 3046611103cSGreg Clayton return target_sp; 3056611103cSGreg Clayton } 3066611103cSGreg Clayton 307e4e45924SGreg Clayton TargetSP 308e4e45924SGreg Clayton Debugger::FindTargetWithProcess (Process *process) 309e4e45924SGreg Clayton { 310e4e45924SGreg Clayton TargetSP target_sp; 311e4e45924SGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 312e4e45924SGreg Clayton DebuggerList &debugger_list = GetDebuggerList(); 313e4e45924SGreg Clayton DebuggerList::iterator pos, end = debugger_list.end(); 314e4e45924SGreg Clayton for (pos = debugger_list.begin(); pos != end; ++pos) 315e4e45924SGreg Clayton { 316e4e45924SGreg Clayton target_sp = (*pos)->GetTargetList().FindTargetWithProcess (process); 317e4e45924SGreg Clayton if (target_sp) 318e4e45924SGreg Clayton break; 319e4e45924SGreg Clayton } 320e4e45924SGreg Clayton return target_sp; 321e4e45924SGreg Clayton } 322e4e45924SGreg Clayton 3236611103cSGreg Clayton 32430fdc8d8SChris Lattner Debugger::Debugger () : 325ebc1bb27SCaroline Tice UserID (g_unique_id++), 326dbe54508SGreg Clayton DebuggerInstanceSettings (*GetSettingsController()), 327d46c87a1SGreg Clayton m_input_comm("debugger.input"), 32830fdc8d8SChris Lattner m_input_file (), 32930fdc8d8SChris Lattner m_output_file (), 33030fdc8d8SChris Lattner m_error_file (), 33130fdc8d8SChris Lattner m_target_list (), 332ded470d3SGreg Clayton m_platform_list (), 33330fdc8d8SChris Lattner m_listener ("lldb.Debugger"), 334e37d605eSJim Ingham m_source_manager(*this), 335e37d605eSJim Ingham m_source_file_cache(), 3366611103cSGreg Clayton m_command_interpreter_ap (new CommandInterpreter (*this, eScriptLanguageDefault, false)), 337d5a0a01bSCaroline Tice m_input_reader_stack (), 3384957bf69SGreg Clayton m_input_reader_data () 33930fdc8d8SChris Lattner { 3406611103cSGreg Clayton m_command_interpreter_ap->Initialize (); 341ded470d3SGreg Clayton // Always add our default platform to the platform list 342ded470d3SGreg Clayton PlatformSP default_platform_sp (Platform::GetDefaultPlatform()); 343ded470d3SGreg Clayton assert (default_platform_sp.get()); 344ded470d3SGreg Clayton m_platform_list.Append (default_platform_sp, true); 34530fdc8d8SChris Lattner } 34630fdc8d8SChris Lattner 34730fdc8d8SChris Lattner Debugger::~Debugger () 34830fdc8d8SChris Lattner { 3498314c525SJim Ingham Clear(); 3508314c525SJim Ingham } 3518314c525SJim Ingham 3528314c525SJim Ingham void 3538314c525SJim Ingham Debugger::Clear() 3548314c525SJim Ingham { 3553d6086f6SCaroline Tice CleanUpInputReaders(); 3561ed54f50SGreg Clayton m_listener.Clear(); 3576611103cSGreg Clayton int num_targets = m_target_list.GetNumTargets(); 3586611103cSGreg Clayton for (int i = 0; i < num_targets; i++) 3596611103cSGreg Clayton { 3606611103cSGreg Clayton ProcessSP process_sp (m_target_list.GetTargetAtIndex (i)->GetProcessSP()); 3616611103cSGreg Clayton if (process_sp) 3628314c525SJim Ingham { 363e24c4acfSGreg Clayton if (process_sp->GetShouldDetach()) 3648314c525SJim Ingham process_sp->Detach(); 3658314c525SJim Ingham else 3666611103cSGreg Clayton process_sp->Destroy(); 3676611103cSGreg Clayton } 36830fdc8d8SChris Lattner } 3698314c525SJim Ingham DisconnectInput(); 37030fdc8d8SChris Lattner 3718314c525SJim Ingham } 37230fdc8d8SChris Lattner 37330fdc8d8SChris Lattner bool 374fc3f027dSGreg Clayton Debugger::GetCloseInputOnEOF () const 375fc3f027dSGreg Clayton { 376fc3f027dSGreg Clayton return m_input_comm.GetCloseOnEOF(); 377fc3f027dSGreg Clayton } 378fc3f027dSGreg Clayton 379fc3f027dSGreg Clayton void 380fc3f027dSGreg Clayton Debugger::SetCloseInputOnEOF (bool b) 381fc3f027dSGreg Clayton { 382fc3f027dSGreg Clayton m_input_comm.SetCloseOnEOF(b); 383fc3f027dSGreg Clayton } 384fc3f027dSGreg Clayton 385fc3f027dSGreg Clayton bool 38630fdc8d8SChris Lattner Debugger::GetAsyncExecution () 38730fdc8d8SChris Lattner { 3886611103cSGreg Clayton return !m_command_interpreter_ap->GetSynchronous(); 38930fdc8d8SChris Lattner } 39030fdc8d8SChris Lattner 39130fdc8d8SChris Lattner void 39230fdc8d8SChris Lattner Debugger::SetAsyncExecution (bool async_execution) 39330fdc8d8SChris Lattner { 3946611103cSGreg Clayton m_command_interpreter_ap->SetSynchronous (!async_execution); 39530fdc8d8SChris Lattner } 39630fdc8d8SChris Lattner 39730fdc8d8SChris Lattner 39830fdc8d8SChris Lattner void 39930fdc8d8SChris Lattner Debugger::SetInputFileHandle (FILE *fh, bool tranfer_ownership) 40030fdc8d8SChris Lattner { 40151b1e2d2SGreg Clayton File &in_file = GetInputFile(); 40251b1e2d2SGreg Clayton in_file.SetStream (fh, tranfer_ownership); 40351b1e2d2SGreg Clayton if (in_file.IsValid() == false) 40451b1e2d2SGreg Clayton in_file.SetStream (stdin, true); 40530fdc8d8SChris Lattner 40630fdc8d8SChris Lattner // Disconnect from any old connection if we had one 40730fdc8d8SChris Lattner m_input_comm.Disconnect (); 40851b1e2d2SGreg Clayton m_input_comm.SetConnection (new ConnectionFileDescriptor (in_file.GetDescriptor(), true)); 40930fdc8d8SChris Lattner m_input_comm.SetReadThreadBytesReceivedCallback (Debugger::DispatchInputCallback, this); 41030fdc8d8SChris Lattner 41130fdc8d8SChris Lattner Error error; 41230fdc8d8SChris Lattner if (m_input_comm.StartReadThread (&error) == false) 41330fdc8d8SChris Lattner { 41451b1e2d2SGreg Clayton File &err_file = GetErrorFile(); 41551b1e2d2SGreg Clayton 41651b1e2d2SGreg Clayton err_file.Printf ("error: failed to main input read thread: %s", error.AsCString() ? error.AsCString() : "unkown error"); 41730fdc8d8SChris Lattner exit(1); 41830fdc8d8SChris Lattner } 41930fdc8d8SChris Lattner } 42030fdc8d8SChris Lattner 42130fdc8d8SChris Lattner void 42230fdc8d8SChris Lattner Debugger::SetOutputFileHandle (FILE *fh, bool tranfer_ownership) 42330fdc8d8SChris Lattner { 42451b1e2d2SGreg Clayton File &out_file = GetOutputFile(); 42551b1e2d2SGreg Clayton out_file.SetStream (fh, tranfer_ownership); 42651b1e2d2SGreg Clayton if (out_file.IsValid() == false) 42751b1e2d2SGreg Clayton out_file.SetStream (stdout, false); 4282f88aadfSCaroline Tice 4292f88aadfSCaroline Tice GetCommandInterpreter().GetScriptInterpreter()->ResetOutputFileHandle (fh); 43030fdc8d8SChris Lattner } 43130fdc8d8SChris Lattner 43230fdc8d8SChris Lattner void 43330fdc8d8SChris Lattner Debugger::SetErrorFileHandle (FILE *fh, bool tranfer_ownership) 43430fdc8d8SChris Lattner { 43551b1e2d2SGreg Clayton File &err_file = GetErrorFile(); 43651b1e2d2SGreg Clayton err_file.SetStream (fh, tranfer_ownership); 43751b1e2d2SGreg Clayton if (err_file.IsValid() == false) 43851b1e2d2SGreg Clayton err_file.SetStream (stderr, false); 43930fdc8d8SChris Lattner } 44030fdc8d8SChris Lattner 44130fdc8d8SChris Lattner ExecutionContext 4422976d00aSJim Ingham Debugger::GetSelectedExecutionContext () 44330fdc8d8SChris Lattner { 44430fdc8d8SChris Lattner ExecutionContext exe_ctx; 445c14ee32dSGreg Clayton TargetSP target_sp(GetSelectedTarget()); 446c14ee32dSGreg Clayton exe_ctx.SetTargetSP (target_sp); 44730fdc8d8SChris Lattner 44830fdc8d8SChris Lattner if (target_sp) 44930fdc8d8SChris Lattner { 450c14ee32dSGreg Clayton ProcessSP process_sp (target_sp->GetProcessSP()); 451c14ee32dSGreg Clayton exe_ctx.SetProcessSP (process_sp); 452c14ee32dSGreg Clayton if (process_sp && process_sp->IsRunning() == false) 45330fdc8d8SChris Lattner { 454c14ee32dSGreg Clayton ThreadSP thread_sp (process_sp->GetThreadList().GetSelectedThread()); 455c14ee32dSGreg Clayton if (thread_sp) 45630fdc8d8SChris Lattner { 457c14ee32dSGreg Clayton exe_ctx.SetThreadSP (thread_sp); 458c14ee32dSGreg Clayton exe_ctx.SetFrameSP (thread_sp->GetSelectedFrame()); 459c14ee32dSGreg Clayton if (exe_ctx.GetFramePtr() == NULL) 460c14ee32dSGreg Clayton exe_ctx.SetFrameSP (thread_sp->GetStackFrameAtIndex (0)); 46130fdc8d8SChris Lattner } 46230fdc8d8SChris Lattner } 46330fdc8d8SChris Lattner } 46430fdc8d8SChris Lattner return exe_ctx; 46530fdc8d8SChris Lattner 46630fdc8d8SChris Lattner } 46730fdc8d8SChris Lattner 468b44880caSCaroline Tice InputReaderSP 469b44880caSCaroline Tice Debugger::GetCurrentInputReader () 470b44880caSCaroline Tice { 471b44880caSCaroline Tice InputReaderSP reader_sp; 472b44880caSCaroline Tice 473d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 474b44880caSCaroline Tice { 475b44880caSCaroline Tice // Clear any finished readers from the stack 476b44880caSCaroline Tice while (CheckIfTopInputReaderIsDone()) ; 477b44880caSCaroline Tice 478d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 479d5a0a01bSCaroline Tice reader_sp = m_input_reader_stack.Top(); 480b44880caSCaroline Tice } 481b44880caSCaroline Tice 482b44880caSCaroline Tice return reader_sp; 483b44880caSCaroline Tice } 484b44880caSCaroline Tice 48530fdc8d8SChris Lattner void 48630fdc8d8SChris Lattner Debugger::DispatchInputCallback (void *baton, const void *bytes, size_t bytes_len) 48730fdc8d8SChris Lattner { 488efed6131SCaroline Tice if (bytes_len > 0) 48930fdc8d8SChris Lattner ((Debugger *)baton)->DispatchInput ((char *)bytes, bytes_len); 490efed6131SCaroline Tice else 491efed6131SCaroline Tice ((Debugger *)baton)->DispatchInputEndOfFile (); 49230fdc8d8SChris Lattner } 49330fdc8d8SChris Lattner 49430fdc8d8SChris Lattner 49530fdc8d8SChris Lattner void 49630fdc8d8SChris Lattner Debugger::DispatchInput (const char *bytes, size_t bytes_len) 49730fdc8d8SChris Lattner { 498efed6131SCaroline Tice if (bytes == NULL || bytes_len == 0) 499efed6131SCaroline Tice return; 50030fdc8d8SChris Lattner 50130fdc8d8SChris Lattner WriteToDefaultReader (bytes, bytes_len); 50230fdc8d8SChris Lattner } 50330fdc8d8SChris Lattner 50430fdc8d8SChris Lattner void 505efed6131SCaroline Tice Debugger::DispatchInputInterrupt () 506efed6131SCaroline Tice { 507efed6131SCaroline Tice m_input_reader_data.clear(); 508efed6131SCaroline Tice 509b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 510efed6131SCaroline Tice if (reader_sp) 511b44880caSCaroline Tice { 512efed6131SCaroline Tice reader_sp->Notify (eInputReaderInterrupt); 513efed6131SCaroline Tice 514b44880caSCaroline Tice // If notifying the reader of the interrupt finished the reader, we should pop it off the stack. 515efed6131SCaroline Tice while (CheckIfTopInputReaderIsDone ()) ; 516efed6131SCaroline Tice } 517efed6131SCaroline Tice } 518efed6131SCaroline Tice 519efed6131SCaroline Tice void 520efed6131SCaroline Tice Debugger::DispatchInputEndOfFile () 521efed6131SCaroline Tice { 522efed6131SCaroline Tice m_input_reader_data.clear(); 523efed6131SCaroline Tice 524b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 525efed6131SCaroline Tice if (reader_sp) 526b44880caSCaroline Tice { 527efed6131SCaroline Tice reader_sp->Notify (eInputReaderEndOfFile); 528efed6131SCaroline Tice 529b44880caSCaroline Tice // If notifying the reader of the end-of-file finished the reader, we should pop it off the stack. 530efed6131SCaroline Tice while (CheckIfTopInputReaderIsDone ()) ; 531efed6131SCaroline Tice } 532efed6131SCaroline Tice } 533efed6131SCaroline Tice 534efed6131SCaroline Tice void 5353d6086f6SCaroline Tice Debugger::CleanUpInputReaders () 5363d6086f6SCaroline Tice { 5373d6086f6SCaroline Tice m_input_reader_data.clear(); 5383d6086f6SCaroline Tice 539b44880caSCaroline Tice // The bottom input reader should be the main debugger input reader. We do not want to close that one here. 540d5a0a01bSCaroline Tice while (m_input_reader_stack.GetSize() > 1) 5413d6086f6SCaroline Tice { 542b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 5433d6086f6SCaroline Tice if (reader_sp) 5443d6086f6SCaroline Tice { 5453d6086f6SCaroline Tice reader_sp->Notify (eInputReaderEndOfFile); 5463d6086f6SCaroline Tice reader_sp->SetIsDone (true); 5473d6086f6SCaroline Tice } 5483d6086f6SCaroline Tice } 5493d6086f6SCaroline Tice } 5503d6086f6SCaroline Tice 5513d6086f6SCaroline Tice void 552969ed3d1SCaroline Tice Debugger::NotifyTopInputReader (InputReaderAction notification) 553969ed3d1SCaroline Tice { 554969ed3d1SCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader()); 555969ed3d1SCaroline Tice if (reader_sp) 556969ed3d1SCaroline Tice { 557969ed3d1SCaroline Tice reader_sp->Notify (notification); 558969ed3d1SCaroline Tice 559969ed3d1SCaroline Tice // Flush out any input readers that are done. 560969ed3d1SCaroline Tice while (CheckIfTopInputReaderIsDone ()) 561969ed3d1SCaroline Tice /* Do nothing. */; 562969ed3d1SCaroline Tice } 563969ed3d1SCaroline Tice } 564969ed3d1SCaroline Tice 5659088b068SCaroline Tice bool 5664d122c40SGreg Clayton Debugger::InputReaderIsTopReader (const InputReaderSP& reader_sp) 5679088b068SCaroline Tice { 5689088b068SCaroline Tice InputReaderSP top_reader_sp (GetCurrentInputReader()); 5699088b068SCaroline Tice 570d61c10bcSCaroline Tice return (reader_sp.get() == top_reader_sp.get()); 5719088b068SCaroline Tice } 5729088b068SCaroline Tice 5739088b068SCaroline Tice 574969ed3d1SCaroline Tice void 57530fdc8d8SChris Lattner Debugger::WriteToDefaultReader (const char *bytes, size_t bytes_len) 57630fdc8d8SChris Lattner { 57730fdc8d8SChris Lattner if (bytes && bytes_len) 57830fdc8d8SChris Lattner m_input_reader_data.append (bytes, bytes_len); 57930fdc8d8SChris Lattner 58030fdc8d8SChris Lattner if (m_input_reader_data.empty()) 58130fdc8d8SChris Lattner return; 58230fdc8d8SChris Lattner 583d5a0a01bSCaroline Tice while (!m_input_reader_stack.IsEmpty() && !m_input_reader_data.empty()) 58430fdc8d8SChris Lattner { 58530fdc8d8SChris Lattner // Get the input reader from the top of the stack 586b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 58730fdc8d8SChris Lattner if (!reader_sp) 58830fdc8d8SChris Lattner break; 58930fdc8d8SChris Lattner 590471b31ceSGreg Clayton size_t bytes_handled = reader_sp->HandleRawBytes (m_input_reader_data.c_str(), 59130fdc8d8SChris Lattner m_input_reader_data.size()); 59230fdc8d8SChris Lattner if (bytes_handled) 59330fdc8d8SChris Lattner { 59430fdc8d8SChris Lattner m_input_reader_data.erase (0, bytes_handled); 59530fdc8d8SChris Lattner } 59630fdc8d8SChris Lattner else 59730fdc8d8SChris Lattner { 59830fdc8d8SChris Lattner // No bytes were handled, we might not have reached our 59930fdc8d8SChris Lattner // granularity, just return and wait for more data 60030fdc8d8SChris Lattner break; 60130fdc8d8SChris Lattner } 60230fdc8d8SChris Lattner } 60330fdc8d8SChris Lattner 604b44880caSCaroline Tice // Flush out any input readers that are done. 60530fdc8d8SChris Lattner while (CheckIfTopInputReaderIsDone ()) 60630fdc8d8SChris Lattner /* Do nothing. */; 60730fdc8d8SChris Lattner 60830fdc8d8SChris Lattner } 60930fdc8d8SChris Lattner 61030fdc8d8SChris Lattner void 61130fdc8d8SChris Lattner Debugger::PushInputReader (const InputReaderSP& reader_sp) 61230fdc8d8SChris Lattner { 61330fdc8d8SChris Lattner if (!reader_sp) 61430fdc8d8SChris Lattner return; 615b44880caSCaroline Tice 61630fdc8d8SChris Lattner // Deactivate the old top reader 617b44880caSCaroline Tice InputReaderSP top_reader_sp (GetCurrentInputReader ()); 618b44880caSCaroline Tice 61930fdc8d8SChris Lattner if (top_reader_sp) 62030fdc8d8SChris Lattner top_reader_sp->Notify (eInputReaderDeactivate); 621b44880caSCaroline Tice 622d5a0a01bSCaroline Tice m_input_reader_stack.Push (reader_sp); 62330fdc8d8SChris Lattner reader_sp->Notify (eInputReaderActivate); 62430fdc8d8SChris Lattner ActivateInputReader (reader_sp); 62530fdc8d8SChris Lattner } 62630fdc8d8SChris Lattner 62730fdc8d8SChris Lattner bool 6284d122c40SGreg Clayton Debugger::PopInputReader (const InputReaderSP& pop_reader_sp) 62930fdc8d8SChris Lattner { 63030fdc8d8SChris Lattner bool result = false; 63130fdc8d8SChris Lattner 63230fdc8d8SChris Lattner // The reader on the stop of the stack is done, so let the next 63330fdc8d8SChris Lattner // read on the stack referesh its prompt and if there is one... 634d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 63530fdc8d8SChris Lattner { 636b44880caSCaroline Tice // Cannot call GetCurrentInputReader here, as that would cause an infinite loop. 637d5a0a01bSCaroline Tice InputReaderSP reader_sp(m_input_reader_stack.Top()); 63830fdc8d8SChris Lattner 63930fdc8d8SChris Lattner if (!pop_reader_sp || pop_reader_sp.get() == reader_sp.get()) 64030fdc8d8SChris Lattner { 641d5a0a01bSCaroline Tice m_input_reader_stack.Pop (); 64230fdc8d8SChris Lattner reader_sp->Notify (eInputReaderDeactivate); 64330fdc8d8SChris Lattner reader_sp->Notify (eInputReaderDone); 64430fdc8d8SChris Lattner result = true; 64530fdc8d8SChris Lattner 646d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 64730fdc8d8SChris Lattner { 648d5a0a01bSCaroline Tice reader_sp = m_input_reader_stack.Top(); 64930fdc8d8SChris Lattner if (reader_sp) 65030fdc8d8SChris Lattner { 65130fdc8d8SChris Lattner ActivateInputReader (reader_sp); 65230fdc8d8SChris Lattner reader_sp->Notify (eInputReaderReactivate); 65330fdc8d8SChris Lattner } 65430fdc8d8SChris Lattner } 65530fdc8d8SChris Lattner } 65630fdc8d8SChris Lattner } 65730fdc8d8SChris Lattner return result; 65830fdc8d8SChris Lattner } 65930fdc8d8SChris Lattner 66030fdc8d8SChris Lattner bool 66130fdc8d8SChris Lattner Debugger::CheckIfTopInputReaderIsDone () 66230fdc8d8SChris Lattner { 66330fdc8d8SChris Lattner bool result = false; 664d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 66530fdc8d8SChris Lattner { 666b44880caSCaroline Tice // Cannot call GetCurrentInputReader here, as that would cause an infinite loop. 667d5a0a01bSCaroline Tice InputReaderSP reader_sp(m_input_reader_stack.Top()); 66830fdc8d8SChris Lattner 66930fdc8d8SChris Lattner if (reader_sp && reader_sp->IsDone()) 67030fdc8d8SChris Lattner { 67130fdc8d8SChris Lattner result = true; 67230fdc8d8SChris Lattner PopInputReader (reader_sp); 67330fdc8d8SChris Lattner } 67430fdc8d8SChris Lattner } 67530fdc8d8SChris Lattner return result; 67630fdc8d8SChris Lattner } 67730fdc8d8SChris Lattner 67830fdc8d8SChris Lattner void 67930fdc8d8SChris Lattner Debugger::ActivateInputReader (const InputReaderSP &reader_sp) 68030fdc8d8SChris Lattner { 68151b1e2d2SGreg Clayton int input_fd = m_input_file.GetFile().GetDescriptor(); 68230fdc8d8SChris Lattner 68351b1e2d2SGreg Clayton if (input_fd >= 0) 68430fdc8d8SChris Lattner { 68551b1e2d2SGreg Clayton Terminal tty(input_fd); 686a3406614SGreg Clayton 687a3406614SGreg Clayton tty.SetEcho(reader_sp->GetEcho()); 68830fdc8d8SChris Lattner 68930fdc8d8SChris Lattner switch (reader_sp->GetGranularity()) 69030fdc8d8SChris Lattner { 69130fdc8d8SChris Lattner case eInputReaderGranularityByte: 69230fdc8d8SChris Lattner case eInputReaderGranularityWord: 693a3406614SGreg Clayton tty.SetCanonical (false); 69430fdc8d8SChris Lattner break; 69530fdc8d8SChris Lattner 69630fdc8d8SChris Lattner case eInputReaderGranularityLine: 69730fdc8d8SChris Lattner case eInputReaderGranularityAll: 698a3406614SGreg Clayton tty.SetCanonical (true); 69930fdc8d8SChris Lattner break; 70030fdc8d8SChris Lattner 70130fdc8d8SChris Lattner default: 70230fdc8d8SChris Lattner break; 70330fdc8d8SChris Lattner } 70430fdc8d8SChris Lattner } 70530fdc8d8SChris Lattner } 7066611103cSGreg Clayton 7075b52f0c7SJim Ingham StreamSP 7085b52f0c7SJim Ingham Debugger::GetAsyncOutputStream () 7095b52f0c7SJim Ingham { 7105b52f0c7SJim Ingham return StreamSP (new StreamAsynchronousIO (GetCommandInterpreter(), 7115b52f0c7SJim Ingham CommandInterpreter::eBroadcastBitAsynchronousOutputData)); 7125b52f0c7SJim Ingham } 7135b52f0c7SJim Ingham 7145b52f0c7SJim Ingham StreamSP 7155b52f0c7SJim Ingham Debugger::GetAsyncErrorStream () 7165b52f0c7SJim Ingham { 7175b52f0c7SJim Ingham return StreamSP (new StreamAsynchronousIO (GetCommandInterpreter(), 7185b52f0c7SJim Ingham CommandInterpreter::eBroadcastBitAsynchronousErrorData)); 7195b52f0c7SJim Ingham } 7205b52f0c7SJim Ingham 721ebc1bb27SCaroline Tice DebuggerSP 722ebc1bb27SCaroline Tice Debugger::FindDebuggerWithID (lldb::user_id_t id) 723ebc1bb27SCaroline Tice { 7244d122c40SGreg Clayton DebuggerSP debugger_sp; 725ebc1bb27SCaroline Tice 726ebc1bb27SCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 727ebc1bb27SCaroline Tice DebuggerList &debugger_list = GetDebuggerList(); 728ebc1bb27SCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 729ebc1bb27SCaroline Tice for (pos = debugger_list.begin(); pos != end; ++pos) 730ebc1bb27SCaroline Tice { 731ebc1bb27SCaroline Tice if ((*pos).get()->GetID() == id) 732ebc1bb27SCaroline Tice { 733ebc1bb27SCaroline Tice debugger_sp = *pos; 734ebc1bb27SCaroline Tice break; 735ebc1bb27SCaroline Tice } 736ebc1bb27SCaroline Tice } 737ebc1bb27SCaroline Tice return debugger_sp; 738ebc1bb27SCaroline Tice } 7393df9a8dfSCaroline Tice 7401b654882SGreg Clayton static void 7411b654882SGreg Clayton TestPromptFormats (StackFrame *frame) 7421b654882SGreg Clayton { 7431b654882SGreg Clayton if (frame == NULL) 7441b654882SGreg Clayton return; 7451b654882SGreg Clayton 7461b654882SGreg Clayton StreamString s; 7471b654882SGreg Clayton const char *prompt_format = 7481b654882SGreg Clayton "{addr = '${addr}'\n}" 7491b654882SGreg Clayton "{process.id = '${process.id}'\n}" 7501b654882SGreg Clayton "{process.name = '${process.name}'\n}" 7511b654882SGreg Clayton "{process.file.basename = '${process.file.basename}'\n}" 7521b654882SGreg Clayton "{process.file.fullpath = '${process.file.fullpath}'\n}" 7531b654882SGreg Clayton "{thread.id = '${thread.id}'\n}" 7541b654882SGreg Clayton "{thread.index = '${thread.index}'\n}" 7551b654882SGreg Clayton "{thread.name = '${thread.name}'\n}" 7561b654882SGreg Clayton "{thread.queue = '${thread.queue}'\n}" 7571b654882SGreg Clayton "{thread.stop-reason = '${thread.stop-reason}'\n}" 7581b654882SGreg Clayton "{target.arch = '${target.arch}'\n}" 7591b654882SGreg Clayton "{module.file.basename = '${module.file.basename}'\n}" 7601b654882SGreg Clayton "{module.file.fullpath = '${module.file.fullpath}'\n}" 7611b654882SGreg Clayton "{file.basename = '${file.basename}'\n}" 7621b654882SGreg Clayton "{file.fullpath = '${file.fullpath}'\n}" 7631b654882SGreg Clayton "{frame.index = '${frame.index}'\n}" 7641b654882SGreg Clayton "{frame.pc = '${frame.pc}'\n}" 7651b654882SGreg Clayton "{frame.sp = '${frame.sp}'\n}" 7661b654882SGreg Clayton "{frame.fp = '${frame.fp}'\n}" 7671b654882SGreg Clayton "{frame.flags = '${frame.flags}'\n}" 7681b654882SGreg Clayton "{frame.reg.rdi = '${frame.reg.rdi}'\n}" 7691b654882SGreg Clayton "{frame.reg.rip = '${frame.reg.rip}'\n}" 7701b654882SGreg Clayton "{frame.reg.rsp = '${frame.reg.rsp}'\n}" 7711b654882SGreg Clayton "{frame.reg.rbp = '${frame.reg.rbp}'\n}" 7721b654882SGreg Clayton "{frame.reg.rflags = '${frame.reg.rflags}'\n}" 7731b654882SGreg Clayton "{frame.reg.xmm0 = '${frame.reg.xmm0}'\n}" 7741b654882SGreg Clayton "{frame.reg.carp = '${frame.reg.carp}'\n}" 7751b654882SGreg Clayton "{function.id = '${function.id}'\n}" 7761b654882SGreg Clayton "{function.name = '${function.name}'\n}" 7771b654882SGreg Clayton "{function.addr-offset = '${function.addr-offset}'\n}" 7781b654882SGreg Clayton "{function.line-offset = '${function.line-offset}'\n}" 7791b654882SGreg Clayton "{function.pc-offset = '${function.pc-offset}'\n}" 7801b654882SGreg Clayton "{line.file.basename = '${line.file.basename}'\n}" 7811b654882SGreg Clayton "{line.file.fullpath = '${line.file.fullpath}'\n}" 7821b654882SGreg Clayton "{line.number = '${line.number}'\n}" 7831b654882SGreg Clayton "{line.start-addr = '${line.start-addr}'\n}" 7841b654882SGreg Clayton "{line.end-addr = '${line.end-addr}'\n}" 7851b654882SGreg Clayton ; 7861b654882SGreg Clayton 7871b654882SGreg Clayton SymbolContext sc (frame->GetSymbolContext(eSymbolContextEverything)); 7881b654882SGreg Clayton ExecutionContext exe_ctx; 7890603aa9dSGreg Clayton frame->CalculateExecutionContext(exe_ctx); 7901b654882SGreg Clayton const char *end = NULL; 7911b654882SGreg Clayton if (Debugger::FormatPrompt (prompt_format, &sc, &exe_ctx, &sc.line_entry.range.GetBaseAddress(), s, &end)) 7921b654882SGreg Clayton { 7931b654882SGreg Clayton printf("%s\n", s.GetData()); 7941b654882SGreg Clayton } 7951b654882SGreg Clayton else 7961b654882SGreg Clayton { 7971b654882SGreg Clayton printf ("error: at '%s'\n", end); 7981b654882SGreg Clayton printf ("what we got: %s\n", s.GetData()); 7991b654882SGreg Clayton } 8001b654882SGreg Clayton } 8011b654882SGreg Clayton 8029fc1944eSEnrico Granata static bool 8039fc1944eSEnrico Granata ScanFormatDescriptor (const char* var_name_begin, 8049fc1944eSEnrico Granata const char* var_name_end, 8059fc1944eSEnrico Granata const char** var_name_final, 8069fc1944eSEnrico Granata const char** percent_position, 8074d122c40SGreg Clayton Format* custom_format, 8089fc1944eSEnrico Granata ValueObject::ValueObjectRepresentationStyle* val_obj_display) 8099fc1944eSEnrico Granata { 810e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 8119fc1944eSEnrico Granata *percent_position = ::strchr(var_name_begin,'%'); 8129fc1944eSEnrico Granata if (!*percent_position || *percent_position > var_name_end) 813e992a089SEnrico Granata { 814e992a089SEnrico Granata if (log) 815e992a089SEnrico Granata log->Printf("no format descriptor in string, skipping"); 8169fc1944eSEnrico Granata *var_name_final = var_name_end; 817e992a089SEnrico Granata } 8189fc1944eSEnrico Granata else 8199fc1944eSEnrico Granata { 8209fc1944eSEnrico Granata *var_name_final = *percent_position; 8219fc1944eSEnrico Granata char* format_name = new char[var_name_end-*var_name_final]; format_name[var_name_end-*var_name_final-1] = '\0'; 8229fc1944eSEnrico Granata memcpy(format_name, *var_name_final+1, var_name_end-*var_name_final-1); 823e992a089SEnrico Granata if (log) 824e992a089SEnrico Granata log->Printf("parsing %s as a format descriptor", format_name); 8259fc1944eSEnrico Granata if ( !FormatManager::GetFormatFromCString(format_name, 8269fc1944eSEnrico Granata true, 8279fc1944eSEnrico Granata *custom_format) ) 8289fc1944eSEnrico Granata { 829e992a089SEnrico Granata if (log) 830e992a089SEnrico Granata log->Printf("%s is an unknown format", format_name); 8319fc1944eSEnrico Granata // if this is an @ sign, print ObjC description 8329fc1944eSEnrico Granata if (*format_name == '@') 8339fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayLanguageSpecific; 8349fc1944eSEnrico Granata // if this is a V, print the value using the default format 835e992a089SEnrico Granata else if (*format_name == 'V') 8369fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayValue; 837d55546b2SEnrico Granata // if this is an L, print the location of the value 838e992a089SEnrico Granata else if (*format_name == 'L') 839f2bbf717SEnrico Granata *val_obj_display = ValueObject::eDisplayLocation; 840d55546b2SEnrico Granata // if this is an S, print the summary after all 841e992a089SEnrico Granata else if (*format_name == 'S') 842d55546b2SEnrico Granata *val_obj_display = ValueObject::eDisplaySummary; 8435dfd49ccSEnrico Granata else if (*format_name == '#') 8445dfd49ccSEnrico Granata *val_obj_display = ValueObject::eDisplayChildrenCount; 845d64d0bc0SEnrico Granata else if (*format_name == 'T') 846d64d0bc0SEnrico Granata *val_obj_display = ValueObject::eDisplayType; 847e992a089SEnrico Granata else if (log) 848e992a089SEnrico Granata log->Printf("%s is an error, leaving the previous value alone", format_name); 8499fc1944eSEnrico Granata } 8509fc1944eSEnrico Granata // a good custom format tells us to print the value using it 8519fc1944eSEnrico Granata else 852e992a089SEnrico Granata { 853e992a089SEnrico Granata if (log) 854e992a089SEnrico Granata log->Printf("will display value for this VO"); 8559fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayValue; 856e992a089SEnrico Granata } 8579fc1944eSEnrico Granata delete format_name; 8589fc1944eSEnrico Granata } 859e992a089SEnrico Granata if (log) 860e992a089SEnrico Granata log->Printf("final format description outcome: custom_format = %d, val_obj_display = %d", 861e992a089SEnrico Granata *custom_format, 862e992a089SEnrico Granata *val_obj_display); 8639fc1944eSEnrico Granata return true; 8649fc1944eSEnrico Granata } 8659fc1944eSEnrico Granata 8669fc1944eSEnrico Granata static bool 8679fc1944eSEnrico Granata ScanBracketedRange (const char* var_name_begin, 8689fc1944eSEnrico Granata const char* var_name_end, 8699fc1944eSEnrico Granata const char* var_name_final, 8709fc1944eSEnrico Granata const char** open_bracket_position, 8719fc1944eSEnrico Granata const char** separator_position, 8729fc1944eSEnrico Granata const char** close_bracket_position, 8739fc1944eSEnrico Granata const char** var_name_final_if_array_range, 8749fc1944eSEnrico Granata int64_t* index_lower, 8759fc1944eSEnrico Granata int64_t* index_higher) 8769fc1944eSEnrico Granata { 877e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 8789fc1944eSEnrico Granata *open_bracket_position = ::strchr(var_name_begin,'['); 8799fc1944eSEnrico Granata if (*open_bracket_position && *open_bracket_position < var_name_final) 8809fc1944eSEnrico Granata { 8819fc1944eSEnrico Granata *separator_position = ::strchr(*open_bracket_position,'-'); // might be NULL if this is a simple var[N] bitfield 8829fc1944eSEnrico Granata *close_bracket_position = ::strchr(*open_bracket_position,']'); 8839fc1944eSEnrico Granata // as usual, we assume that [] will come before % 8849fc1944eSEnrico Granata //printf("trying to expand a []\n"); 8859fc1944eSEnrico Granata *var_name_final_if_array_range = *open_bracket_position; 8869fc1944eSEnrico Granata if (*close_bracket_position - *open_bracket_position == 1) 8879fc1944eSEnrico Granata { 888e992a089SEnrico Granata if (log) 889e992a089SEnrico Granata log->Printf("[] detected.. going from 0 to end of data"); 8909fc1944eSEnrico Granata *index_lower = 0; 8919fc1944eSEnrico Granata } 8929fc1944eSEnrico Granata else if (*separator_position == NULL || *separator_position > var_name_end) 8939fc1944eSEnrico Granata { 8949fc1944eSEnrico Granata char *end = NULL; 8959fc1944eSEnrico Granata *index_lower = ::strtoul (*open_bracket_position+1, &end, 0); 8969fc1944eSEnrico Granata *index_higher = *index_lower; 897e992a089SEnrico Granata if (log) 898fd54b368SJason Molenda log->Printf("[%lld] detected, high index is same", *index_lower); 8999fc1944eSEnrico Granata } 9009fc1944eSEnrico Granata else if (*close_bracket_position && *close_bracket_position < var_name_end) 9019fc1944eSEnrico Granata { 9029fc1944eSEnrico Granata char *end = NULL; 9039fc1944eSEnrico Granata *index_lower = ::strtoul (*open_bracket_position+1, &end, 0); 9049fc1944eSEnrico Granata *index_higher = ::strtoul (*separator_position+1, &end, 0); 905e992a089SEnrico Granata if (log) 906fd54b368SJason Molenda log->Printf("[%lld-%lld] detected", *index_lower, *index_higher); 9079fc1944eSEnrico Granata } 9089fc1944eSEnrico Granata else 909e992a089SEnrico Granata { 910e992a089SEnrico Granata if (log) 911e992a089SEnrico Granata log->Printf("expression is erroneous, cannot extract indices out of it"); 9129fc1944eSEnrico Granata return false; 913e992a089SEnrico Granata } 9149fc1944eSEnrico Granata if (*index_lower > *index_higher && *index_higher > 0) 9159fc1944eSEnrico Granata { 916e992a089SEnrico Granata if (log) 917e992a089SEnrico Granata log->Printf("swapping indices"); 9189fc1944eSEnrico Granata int temp = *index_lower; 9199fc1944eSEnrico Granata *index_lower = *index_higher; 9209fc1944eSEnrico Granata *index_higher = temp; 9219fc1944eSEnrico Granata } 9229fc1944eSEnrico Granata } 923e992a089SEnrico Granata else if (log) 924e992a089SEnrico Granata log->Printf("no bracketed range, skipping entirely"); 9259fc1944eSEnrico Granata return true; 9269fc1944eSEnrico Granata } 9279fc1944eSEnrico Granata 9289fc1944eSEnrico Granata 9299fc1944eSEnrico Granata static ValueObjectSP 930c482a192SEnrico Granata ExpandExpressionPath (ValueObject* valobj, 9319fc1944eSEnrico Granata StackFrame* frame, 9329fc1944eSEnrico Granata bool* do_deref_pointer, 9339fc1944eSEnrico Granata const char* var_name_begin, 9349fc1944eSEnrico Granata const char* var_name_final, 9359fc1944eSEnrico Granata Error& error) 9369fc1944eSEnrico Granata { 937e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 9389fc1944eSEnrico Granata StreamString sstring; 9399fc1944eSEnrico Granata VariableSP var_sp; 9409fc1944eSEnrico Granata 9419fc1944eSEnrico Granata if (*do_deref_pointer) 942e992a089SEnrico Granata { 943e992a089SEnrico Granata if (log) 944e992a089SEnrico Granata log->Printf("been told to deref_pointer by caller"); 9459fc1944eSEnrico Granata sstring.PutChar('*'); 946e992a089SEnrico Granata } 947c482a192SEnrico Granata else if (valobj->IsDereferenceOfParent() && ClangASTContext::IsPointerType(valobj->GetParent()->GetClangType()) && !valobj->IsArrayItemForPointer()) 9489fc1944eSEnrico Granata { 949e992a089SEnrico Granata if (log) 950e992a089SEnrico Granata log->Printf("decided to deref_pointer myself"); 9519fc1944eSEnrico Granata sstring.PutChar('*'); 9529fc1944eSEnrico Granata *do_deref_pointer = true; 9539fc1944eSEnrico Granata } 9549fc1944eSEnrico Granata 955c482a192SEnrico Granata valobj->GetExpressionPath(sstring, true, ValueObject::eHonorPointers); 956e992a089SEnrico Granata if (log) 957e992a089SEnrico Granata log->Printf("expression path to expand in phase 0: %s",sstring.GetData()); 9589fc1944eSEnrico Granata sstring.PutRawBytes(var_name_begin+3, var_name_final-var_name_begin-3); 959e992a089SEnrico Granata if (log) 960e992a089SEnrico Granata log->Printf("expression path to expand in phase 1: %s",sstring.GetData()); 9619fc1944eSEnrico Granata std::string name = std::string(sstring.GetData()); 9629fc1944eSEnrico Granata ValueObjectSP target = frame->GetValueForVariableExpressionPath (name.c_str(), 9639fc1944eSEnrico Granata eNoDynamicValues, 9649fc1944eSEnrico Granata 0, 9659fc1944eSEnrico Granata var_sp, 9669fc1944eSEnrico Granata error); 9679fc1944eSEnrico Granata return target; 9689fc1944eSEnrico Granata } 9699fc1944eSEnrico Granata 9709fc1944eSEnrico Granata static ValueObjectSP 971c482a192SEnrico Granata ExpandIndexedExpression (ValueObject* valobj, 9729fc1944eSEnrico Granata uint32_t index, 9739fc1944eSEnrico Granata StackFrame* frame, 974fc7a7f3bSEnrico Granata bool deref_pointer) 9759fc1944eSEnrico Granata { 976e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 977fc7a7f3bSEnrico Granata const char* ptr_deref_format = "[%d]"; 978fc7a7f3bSEnrico Granata std::auto_ptr<char> ptr_deref_buffer(new char[10]); 979fc7a7f3bSEnrico Granata ::sprintf(ptr_deref_buffer.get(), ptr_deref_format, index); 980e992a089SEnrico Granata if (log) 981e992a089SEnrico Granata log->Printf("name to deref: %s",ptr_deref_buffer.get()); 982fc7a7f3bSEnrico Granata const char* first_unparsed; 983fc7a7f3bSEnrico Granata ValueObject::GetValueForExpressionPathOptions options; 984fc7a7f3bSEnrico Granata ValueObject::ExpressionPathEndResultType final_value_type; 985fc7a7f3bSEnrico Granata ValueObject::ExpressionPathScanEndReason reason_to_stop; 986fc7a7f3bSEnrico Granata ValueObject::ExpressionPathAftermath what_next = (deref_pointer ? ValueObject::eDereference : ValueObject::eNothing); 987c482a192SEnrico Granata ValueObjectSP item = valobj->GetValueForExpressionPath (ptr_deref_buffer.get(), 988fc7a7f3bSEnrico Granata &first_unparsed, 989fc7a7f3bSEnrico Granata &reason_to_stop, 990fc7a7f3bSEnrico Granata &final_value_type, 991fc7a7f3bSEnrico Granata options, 992fc7a7f3bSEnrico Granata &what_next); 993fc7a7f3bSEnrico Granata if (!item) 994fc7a7f3bSEnrico Granata { 995e992a089SEnrico Granata if (log) 996e992a089SEnrico Granata log->Printf("ERROR: unparsed portion = %s, why stopping = %d," 997e992a089SEnrico Granata " final_value_type %d", 998fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 999fc7a7f3bSEnrico Granata } 10009fc1944eSEnrico Granata else 10019fc1944eSEnrico Granata { 1002e992a089SEnrico Granata if (log) 1003e992a089SEnrico Granata log->Printf("ALL RIGHT: unparsed portion = %s, why stopping = %d," 1004e992a089SEnrico Granata " final_value_type %d", 1005fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 10069fc1944eSEnrico Granata } 10079fc1944eSEnrico Granata return item; 10089fc1944eSEnrico Granata } 10099fc1944eSEnrico Granata 10101b654882SGreg Clayton bool 10111b654882SGreg Clayton Debugger::FormatPrompt 10121b654882SGreg Clayton ( 10131b654882SGreg Clayton const char *format, 10141b654882SGreg Clayton const SymbolContext *sc, 10151b654882SGreg Clayton const ExecutionContext *exe_ctx, 10161b654882SGreg Clayton const Address *addr, 10171b654882SGreg Clayton Stream &s, 10184becb37eSEnrico Granata const char **end, 1019c482a192SEnrico Granata ValueObject* valobj 10201b654882SGreg Clayton ) 10211b654882SGreg Clayton { 1022c482a192SEnrico Granata ValueObject* realvalobj = NULL; // makes it super-easy to parse pointers 10231b654882SGreg Clayton bool success = true; 10241b654882SGreg Clayton const char *p; 1025e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 10261b654882SGreg Clayton for (p = format; *p != '\0'; ++p) 10271b654882SGreg Clayton { 1028c482a192SEnrico Granata if (realvalobj) 10294becb37eSEnrico Granata { 1030c482a192SEnrico Granata valobj = realvalobj; 1031c482a192SEnrico Granata realvalobj = NULL; 10324becb37eSEnrico Granata } 10331b654882SGreg Clayton size_t non_special_chars = ::strcspn (p, "${}\\"); 10341b654882SGreg Clayton if (non_special_chars > 0) 10351b654882SGreg Clayton { 10361b654882SGreg Clayton if (success) 10371b654882SGreg Clayton s.Write (p, non_special_chars); 10381b654882SGreg Clayton p += non_special_chars; 10391b654882SGreg Clayton } 10401b654882SGreg Clayton 10411b654882SGreg Clayton if (*p == '\0') 10421b654882SGreg Clayton { 10431b654882SGreg Clayton break; 10441b654882SGreg Clayton } 10451b654882SGreg Clayton else if (*p == '{') 10461b654882SGreg Clayton { 10471b654882SGreg Clayton // Start a new scope that must have everything it needs if it is to 10481b654882SGreg Clayton // to make it into the final output stream "s". If you want to make 10491b654882SGreg Clayton // a format that only prints out the function or symbol name if there 10501b654882SGreg Clayton // is one in the symbol context you can use: 10511b654882SGreg Clayton // "{function =${function.name}}" 10521b654882SGreg Clayton // The first '{' starts a new scope that end with the matching '}' at 10531b654882SGreg Clayton // the end of the string. The contents "function =${function.name}" 10541b654882SGreg Clayton // will then be evaluated and only be output if there is a function 10551b654882SGreg Clayton // or symbol with a valid name. 10561b654882SGreg Clayton StreamString sub_strm; 10571b654882SGreg Clayton 10581b654882SGreg Clayton ++p; // Skip the '{' 10591b654882SGreg Clayton 1060c482a192SEnrico Granata if (FormatPrompt (p, sc, exe_ctx, addr, sub_strm, &p, valobj)) 10611b654882SGreg Clayton { 10621b654882SGreg Clayton // The stream had all it needed 10631b654882SGreg Clayton s.Write(sub_strm.GetData(), sub_strm.GetSize()); 10641b654882SGreg Clayton } 10651b654882SGreg Clayton if (*p != '}') 10661b654882SGreg Clayton { 10671b654882SGreg Clayton success = false; 10681b654882SGreg Clayton break; 10691b654882SGreg Clayton } 10701b654882SGreg Clayton } 10711b654882SGreg Clayton else if (*p == '}') 10721b654882SGreg Clayton { 10731b654882SGreg Clayton // End of a enclosing scope 10741b654882SGreg Clayton break; 10751b654882SGreg Clayton } 10761b654882SGreg Clayton else if (*p == '$') 10771b654882SGreg Clayton { 10781b654882SGreg Clayton // We have a prompt variable to print 10791b654882SGreg Clayton ++p; 10801b654882SGreg Clayton if (*p == '{') 10811b654882SGreg Clayton { 10821b654882SGreg Clayton ++p; 10831b654882SGreg Clayton const char *var_name_begin = p; 10841b654882SGreg Clayton const char *var_name_end = ::strchr (p, '}'); 10851b654882SGreg Clayton 10861b654882SGreg Clayton if (var_name_end && var_name_begin < var_name_end) 10871b654882SGreg Clayton { 10881b654882SGreg Clayton // if we have already failed to parse, skip this variable 10891b654882SGreg Clayton if (success) 10901b654882SGreg Clayton { 10911b654882SGreg Clayton const char *cstr = NULL; 10921b654882SGreg Clayton Address format_addr; 10931b654882SGreg Clayton bool calculate_format_addr_function_offset = false; 10941b654882SGreg Clayton // Set reg_kind and reg_num to invalid values 10951b654882SGreg Clayton RegisterKind reg_kind = kNumRegisterKinds; 10961b654882SGreg Clayton uint32_t reg_num = LLDB_INVALID_REGNUM; 10971b654882SGreg Clayton FileSpec format_file_spec; 1098e0d378b3SGreg Clayton const RegisterInfo *reg_info = NULL; 10991b654882SGreg Clayton RegisterContext *reg_ctx = NULL; 11009fc1944eSEnrico Granata bool do_deref_pointer = false; 1101e992a089SEnrico Granata ValueObject::ExpressionPathScanEndReason reason_to_stop = ValueObject::eEndOfString; 1102e992a089SEnrico Granata ValueObject::ExpressionPathEndResultType final_value_type = ValueObject::ePlain; 11031b654882SGreg Clayton 11041b654882SGreg Clayton // Each variable must set success to true below... 11051b654882SGreg Clayton bool var_success = false; 11061b654882SGreg Clayton switch (var_name_begin[0]) 11071b654882SGreg Clayton { 11084becb37eSEnrico Granata case '*': 11096f3533fbSEnrico Granata case 'v': 11106f3533fbSEnrico Granata case 's': 11114becb37eSEnrico Granata { 1112c482a192SEnrico Granata if (!valobj) 111334132754SGreg Clayton break; 11146f3533fbSEnrico Granata 1115c3e320a7SEnrico Granata if (log) 1116c3e320a7SEnrico Granata log->Printf("initial string: %s",var_name_begin); 1117c3e320a7SEnrico Granata 11186f3533fbSEnrico Granata // check for *var and *svar 11196f3533fbSEnrico Granata if (*var_name_begin == '*') 11206f3533fbSEnrico Granata { 11219fc1944eSEnrico Granata do_deref_pointer = true; 11229fc1944eSEnrico Granata var_name_begin++; 11239fc1944eSEnrico Granata } 1124c3e320a7SEnrico Granata 1125c3e320a7SEnrico Granata if (log) 1126c3e320a7SEnrico Granata log->Printf("initial string: %s",var_name_begin); 1127c3e320a7SEnrico Granata 11286f3533fbSEnrico Granata if (*var_name_begin == 's') 11294becb37eSEnrico Granata { 11304d122c40SGreg Clayton valobj = valobj->GetSyntheticValue(eUseSyntheticFilter).get(); 11316f3533fbSEnrico Granata var_name_begin++; 11326f3533fbSEnrico Granata } 11336f3533fbSEnrico Granata 1134c3e320a7SEnrico Granata if (log) 1135c3e320a7SEnrico Granata log->Printf("initial string: %s",var_name_begin); 1136c3e320a7SEnrico Granata 11376f3533fbSEnrico Granata // should be a 'v' by now 11386f3533fbSEnrico Granata if (*var_name_begin != 'v') 11396f3533fbSEnrico Granata break; 11406f3533fbSEnrico Granata 1141c3e320a7SEnrico Granata if (log) 1142c3e320a7SEnrico Granata log->Printf("initial string: %s",var_name_begin); 1143c3e320a7SEnrico Granata 1144fc7a7f3bSEnrico Granata ValueObject::ExpressionPathAftermath what_next = (do_deref_pointer ? 1145fc7a7f3bSEnrico Granata ValueObject::eDereference : ValueObject::eNothing); 1146fc7a7f3bSEnrico Granata ValueObject::GetValueForExpressionPathOptions options; 11478c9d3560SEnrico Granata options.DontCheckDotVsArrowSyntax().DoAllowBitfieldSyntax().DoAllowFragileIVar().DoAllowSyntheticChildren(); 11480a3958e0SEnrico Granata ValueObject::ValueObjectRepresentationStyle val_obj_display = ValueObject::eDisplaySummary; 114934132754SGreg Clayton ValueObject* target = NULL; 11504d122c40SGreg Clayton Format custom_format = eFormatInvalid; 115134132754SGreg Clayton const char* var_name_final = NULL; 11529fc1944eSEnrico Granata const char* var_name_final_if_array_range = NULL; 115334132754SGreg Clayton const char* close_bracket_position = NULL; 115434132754SGreg Clayton int64_t index_lower = -1; 115534132754SGreg Clayton int64_t index_higher = -1; 11569fc1944eSEnrico Granata bool is_array_range = false; 1157fc7a7f3bSEnrico Granata const char* first_unparsed; 115885933ed4SEnrico Granata bool was_plain_var = false; 115985933ed4SEnrico Granata bool was_var_format = false; 1160fc7a7f3bSEnrico Granata 1161c482a192SEnrico Granata if (!valobj) break; 1162c482a192SEnrico Granata // simplest case ${var}, just print valobj's value 11639fc1944eSEnrico Granata if (::strncmp (var_name_begin, "var}", strlen("var}")) == 0) 11640a3958e0SEnrico Granata { 116585933ed4SEnrico Granata was_plain_var = true; 1166c482a192SEnrico Granata target = valobj; 11670a3958e0SEnrico Granata val_obj_display = ValueObject::eDisplayValue; 11680a3958e0SEnrico Granata } 11699fc1944eSEnrico Granata else if (::strncmp(var_name_begin,"var%",strlen("var%")) == 0) 11709fc1944eSEnrico Granata { 117185933ed4SEnrico Granata was_var_format = true; 11729fc1944eSEnrico Granata // this is a variable with some custom format applied to it 11739fc1944eSEnrico Granata const char* percent_position; 1174c482a192SEnrico Granata target = valobj; 11750a3958e0SEnrico Granata val_obj_display = ValueObject::eDisplayValue; 11769fc1944eSEnrico Granata ScanFormatDescriptor (var_name_begin, 11779fc1944eSEnrico Granata var_name_end, 11789fc1944eSEnrico Granata &var_name_final, 11799fc1944eSEnrico Granata &percent_position, 11809fc1944eSEnrico Granata &custom_format, 11819fc1944eSEnrico Granata &val_obj_display); 11820a3958e0SEnrico Granata } 11839fc1944eSEnrico Granata // this is ${var.something} or multiple .something nested 11849fc1944eSEnrico Granata else if (::strncmp (var_name_begin, "var", strlen("var")) == 0) 11859fc1944eSEnrico Granata { 11869fc1944eSEnrico Granata 11879fc1944eSEnrico Granata const char* percent_position; 11889fc1944eSEnrico Granata ScanFormatDescriptor (var_name_begin, 11899fc1944eSEnrico Granata var_name_end, 11909fc1944eSEnrico Granata &var_name_final, 11919fc1944eSEnrico Granata &percent_position, 11929fc1944eSEnrico Granata &custom_format, 11939fc1944eSEnrico Granata &val_obj_display); 11949fc1944eSEnrico Granata 11959fc1944eSEnrico Granata const char* open_bracket_position; 11969fc1944eSEnrico Granata const char* separator_position; 11979fc1944eSEnrico Granata ScanBracketedRange (var_name_begin, 11989fc1944eSEnrico Granata var_name_end, 11999fc1944eSEnrico Granata var_name_final, 12009fc1944eSEnrico Granata &open_bracket_position, 12019fc1944eSEnrico Granata &separator_position, 12029fc1944eSEnrico Granata &close_bracket_position, 12039fc1944eSEnrico Granata &var_name_final_if_array_range, 12049fc1944eSEnrico Granata &index_lower, 12059fc1944eSEnrico Granata &index_higher); 12069fc1944eSEnrico Granata 12079fc1944eSEnrico Granata Error error; 12089fc1944eSEnrico Granata 1209fc7a7f3bSEnrico Granata std::auto_ptr<char> expr_path(new char[var_name_final-var_name_begin-1]); 1210fc7a7f3bSEnrico Granata ::memset(expr_path.get(), 0, var_name_final-var_name_begin-1); 1211fc7a7f3bSEnrico Granata memcpy(expr_path.get(), var_name_begin+3,var_name_final-var_name_begin-3); 1212fc7a7f3bSEnrico Granata 1213e992a089SEnrico Granata if (log) 1214e992a089SEnrico Granata log->Printf("symbol to expand: %s",expr_path.get()); 1215fc7a7f3bSEnrico Granata 1216c482a192SEnrico Granata target = valobj->GetValueForExpressionPath(expr_path.get(), 1217fc7a7f3bSEnrico Granata &first_unparsed, 1218fc7a7f3bSEnrico Granata &reason_to_stop, 1219fc7a7f3bSEnrico Granata &final_value_type, 1220fc7a7f3bSEnrico Granata options, 1221fc7a7f3bSEnrico Granata &what_next).get(); 1222fc7a7f3bSEnrico Granata 1223fc7a7f3bSEnrico Granata if (!target) 12249fc1944eSEnrico Granata { 1225e992a089SEnrico Granata if (log) 1226e992a089SEnrico Granata log->Printf("ERROR: unparsed portion = %s, why stopping = %d," 1227e992a089SEnrico Granata " final_value_type %d", 1228fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 1229fc7a7f3bSEnrico Granata break; 12300a3958e0SEnrico Granata } 1231a7187d00SEnrico Granata else 1232fc7a7f3bSEnrico Granata { 1233e992a089SEnrico Granata if (log) 1234e992a089SEnrico Granata log->Printf("ALL RIGHT: unparsed portion = %s, why stopping = %d," 1235e992a089SEnrico Granata " final_value_type %d", 1236fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 1237a7187d00SEnrico Granata } 12380a3958e0SEnrico Granata } 12390a3958e0SEnrico Granata else 12400a3958e0SEnrico Granata break; 12419fc1944eSEnrico Granata 1242fc7a7f3bSEnrico Granata is_array_range = (final_value_type == ValueObject::eBoundedRange || 1243fc7a7f3bSEnrico Granata final_value_type == ValueObject::eUnboundedRange); 1244fc7a7f3bSEnrico Granata 1245fc7a7f3bSEnrico Granata do_deref_pointer = (what_next == ValueObject::eDereference); 1246fc7a7f3bSEnrico Granata 1247a7187d00SEnrico Granata if (do_deref_pointer && !is_array_range) 12480a3958e0SEnrico Granata { 12499fc1944eSEnrico Granata // I have not deref-ed yet, let's do it 12509fc1944eSEnrico Granata // this happens when we are not going through GetValueForVariableExpressionPath 12519fc1944eSEnrico Granata // to get to the target ValueObject 12529fc1944eSEnrico Granata Error error; 12539fc1944eSEnrico Granata target = target->Dereference(error).get(); 1254dc940730SEnrico Granata if (error.Fail()) 1255dc940730SEnrico Granata { 1256dc940730SEnrico Granata if (log) 1257dc940730SEnrico Granata log->Printf("ERROR: %s\n", error.AsCString("unknown")); \ 1258dc940730SEnrico Granata break; 1259dc940730SEnrico Granata } 12609fc1944eSEnrico Granata do_deref_pointer = false; 12610a3958e0SEnrico Granata } 12620a3958e0SEnrico Granata 126385933ed4SEnrico Granata // TODO use flags for these 1264f4efecd9SEnrico Granata bool is_array = ClangASTContext::IsArrayType(target->GetClangType()); 1265f4efecd9SEnrico Granata bool is_pointer = ClangASTContext::IsPointerType(target->GetClangType()); 126685933ed4SEnrico Granata bool is_aggregate = ClangASTContext::IsAggregateType(target->GetClangType()); 1267f4efecd9SEnrico Granata 1268f4efecd9SEnrico Granata if ((is_array || is_pointer) && (!is_array_range) && val_obj_display == ValueObject::eDisplayValue) // this should be wrong, but there are some exceptions 1269f4efecd9SEnrico Granata { 127085933ed4SEnrico Granata StreamString str_temp; 1271e992a089SEnrico Granata if (log) 1272e992a089SEnrico Granata log->Printf("I am into array || pointer && !range"); 1273d64d0bc0SEnrico Granata 1274d64d0bc0SEnrico Granata if (target->HasSpecialCasesForPrintableRepresentation(val_obj_display, 1275d64d0bc0SEnrico Granata custom_format)) 1276d64d0bc0SEnrico Granata { 1277f4efecd9SEnrico Granata // try to use the special cases 127885933ed4SEnrico Granata var_success = target->DumpPrintableRepresentation(str_temp, 127985933ed4SEnrico Granata val_obj_display, 128085933ed4SEnrico Granata custom_format); 1281e992a089SEnrico Granata if (log) 1282e992a089SEnrico Granata log->Printf("special cases did%s match", var_success ? "" : "n't"); 1283d64d0bc0SEnrico Granata 1284d64d0bc0SEnrico Granata // should not happen 128585933ed4SEnrico Granata if (!var_success) 128685933ed4SEnrico Granata s << "<invalid usage of pointer value as object>"; 128785933ed4SEnrico Granata else 128885933ed4SEnrico Granata s << str_temp.GetData(); 1289d64d0bc0SEnrico Granata var_success = true; 1290d64d0bc0SEnrico Granata break; 1291d64d0bc0SEnrico Granata } 1292d64d0bc0SEnrico Granata else 1293d64d0bc0SEnrico Granata { 129488da35f8SEnrico Granata if (was_plain_var) // if ${var} 1295d64d0bc0SEnrico Granata { 1296d64d0bc0SEnrico Granata s << target->GetTypeName() << " @ " << target->GetLocationAsCString(); 1297d64d0bc0SEnrico Granata } 129888da35f8SEnrico Granata else if (is_pointer) // if pointer, value is the address stored 129988da35f8SEnrico Granata { 130088da35f8SEnrico Granata var_success = target->GetPrintableRepresentation(s, 130188da35f8SEnrico Granata val_obj_display, 130288da35f8SEnrico Granata custom_format); 130388da35f8SEnrico Granata } 1304d64d0bc0SEnrico Granata else 1305d64d0bc0SEnrico Granata { 1306d64d0bc0SEnrico Granata s << "<invalid usage of pointer value as object>"; 1307d64d0bc0SEnrico Granata } 1308d64d0bc0SEnrico Granata var_success = true; 1309d64d0bc0SEnrico Granata break; 1310d64d0bc0SEnrico Granata } 1311d64d0bc0SEnrico Granata } 1312d64d0bc0SEnrico Granata 1313d64d0bc0SEnrico Granata // if directly trying to print ${var}, and this is an aggregate, display a nice 1314d64d0bc0SEnrico Granata // type @ location message 1315d64d0bc0SEnrico Granata if (is_aggregate && was_plain_var) 1316d64d0bc0SEnrico Granata { 1317d64d0bc0SEnrico Granata s << target->GetTypeName() << " @ " << target->GetLocationAsCString(); 1318d64d0bc0SEnrico Granata var_success = true; 131985933ed4SEnrico Granata break; 132085933ed4SEnrico Granata } 132185933ed4SEnrico Granata 1322d64d0bc0SEnrico Granata // if directly trying to print ${var%V}, and this is an aggregate, do not let the user do it 1323d64d0bc0SEnrico Granata if (is_aggregate && ((was_var_format && val_obj_display == ValueObject::eDisplayValue))) 132485933ed4SEnrico Granata { 132585933ed4SEnrico Granata s << "<invalid use of aggregate type>"; 132685933ed4SEnrico Granata var_success = true; 1327f4efecd9SEnrico Granata break; 1328f4efecd9SEnrico Granata } 1329f4efecd9SEnrico Granata 13309fc1944eSEnrico Granata if (!is_array_range) 1331e992a089SEnrico Granata { 1332e992a089SEnrico Granata if (log) 1333e992a089SEnrico Granata log->Printf("dumping ordinary printable output"); 13349fc1944eSEnrico Granata var_success = target->DumpPrintableRepresentation(s,val_obj_display, custom_format); 1335e992a089SEnrico Granata } 13369fc1944eSEnrico Granata else 13379fc1944eSEnrico Granata { 1338e992a089SEnrico Granata if (log) 1339e992a089SEnrico Granata log->Printf("checking if I can handle as array"); 13409fc1944eSEnrico Granata if (!is_array && !is_pointer) 13419fc1944eSEnrico Granata break; 1342e992a089SEnrico Granata if (log) 1343e992a089SEnrico Granata log->Printf("handle as array"); 1344fc7a7f3bSEnrico Granata const char* special_directions = NULL; 1345fc7a7f3bSEnrico Granata StreamString special_directions_writer; 13460a3958e0SEnrico Granata if (close_bracket_position && (var_name_end-close_bracket_position > 1)) 13470a3958e0SEnrico Granata { 1348fc7a7f3bSEnrico Granata ConstString additional_data; 1349fc7a7f3bSEnrico Granata additional_data.SetCStringWithLength(close_bracket_position+1, var_name_end-close_bracket_position-1); 1350fc7a7f3bSEnrico Granata special_directions_writer.Printf("${%svar%s}", 1351fc7a7f3bSEnrico Granata do_deref_pointer ? "*" : "", 1352fc7a7f3bSEnrico Granata additional_data.GetCString()); 1353fc7a7f3bSEnrico Granata special_directions = special_directions_writer.GetData(); 13540a3958e0SEnrico Granata } 13550a3958e0SEnrico Granata 13560a3958e0SEnrico Granata // let us display items index_lower thru index_higher of this array 13570a3958e0SEnrico Granata s.PutChar('['); 13580a3958e0SEnrico Granata var_success = true; 13590a3958e0SEnrico Granata 13609fc1944eSEnrico Granata if (index_higher < 0) 1361c482a192SEnrico Granata index_higher = valobj->GetNumChildren() - 1; 13620a3958e0SEnrico Granata 136322c55d18SEnrico Granata uint32_t max_num_children = target->GetUpdatePoint().GetTargetSP()->GetMaximumNumberOfChildrenToDisplay(); 136422c55d18SEnrico Granata 13650a3958e0SEnrico Granata for (;index_lower<=index_higher;index_lower++) 13660a3958e0SEnrico Granata { 1367fc7a7f3bSEnrico Granata ValueObject* item = ExpandIndexedExpression (target, 13689fc1944eSEnrico Granata index_lower, 1369c14ee32dSGreg Clayton exe_ctx->GetFramePtr(), 1370fc7a7f3bSEnrico Granata false).get(); 13710a3958e0SEnrico Granata 1372fc7a7f3bSEnrico Granata if (!item) 1373fc7a7f3bSEnrico Granata { 1374e992a089SEnrico Granata if (log) 1375fd54b368SJason Molenda log->Printf("ERROR in getting child item at index %lld", index_lower); 1376fc7a7f3bSEnrico Granata } 1377fc7a7f3bSEnrico Granata else 1378fc7a7f3bSEnrico Granata { 1379e992a089SEnrico Granata if (log) 1380e992a089SEnrico Granata log->Printf("special_directions for child item: %s",special_directions); 1381fc7a7f3bSEnrico Granata } 1382fc7a7f3bSEnrico Granata 13830a3958e0SEnrico Granata if (!special_directions) 13849fc1944eSEnrico Granata var_success &= item->DumpPrintableRepresentation(s,val_obj_display, custom_format); 13850a3958e0SEnrico Granata else 13860a3958e0SEnrico Granata var_success &= FormatPrompt(special_directions, sc, exe_ctx, addr, s, NULL, item); 13870a3958e0SEnrico Granata 138822c55d18SEnrico Granata if (--max_num_children == 0) 138922c55d18SEnrico Granata { 139022c55d18SEnrico Granata s.PutCString(", ..."); 139122c55d18SEnrico Granata break; 139222c55d18SEnrico Granata } 139322c55d18SEnrico Granata 13940a3958e0SEnrico Granata if (index_lower < index_higher) 13950a3958e0SEnrico Granata s.PutChar(','); 13960a3958e0SEnrico Granata } 13970a3958e0SEnrico Granata s.PutChar(']'); 13984becb37eSEnrico Granata } 13994becb37eSEnrico Granata } 140034132754SGreg Clayton break; 14011b654882SGreg Clayton case 'a': 14021b654882SGreg Clayton if (::strncmp (var_name_begin, "addr}", strlen("addr}")) == 0) 14031b654882SGreg Clayton { 14041b654882SGreg Clayton if (addr && addr->IsValid()) 14051b654882SGreg Clayton { 14061b654882SGreg Clayton var_success = true; 14071b654882SGreg Clayton format_addr = *addr; 14081b654882SGreg Clayton } 14091b654882SGreg Clayton } 14105a31471eSGreg Clayton else if (::strncmp (var_name_begin, "ansi.", strlen("ansi.")) == 0) 14115a31471eSGreg Clayton { 14125a31471eSGreg Clayton var_success = true; 14135a31471eSGreg Clayton var_name_begin += strlen("ansi."); // Skip the "ansi." 14145a31471eSGreg Clayton if (::strncmp (var_name_begin, "fg.", strlen("fg.")) == 0) 14155a31471eSGreg Clayton { 14165a31471eSGreg Clayton var_name_begin += strlen("fg."); // Skip the "fg." 14175a31471eSGreg Clayton if (::strncmp (var_name_begin, "black}", strlen("black}")) == 0) 14185a31471eSGreg Clayton { 14195a31471eSGreg Clayton s.Printf ("%s%s%s", 14205a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14215a31471eSGreg Clayton lldb_utility::ansi::k_fg_black, 14225a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14235a31471eSGreg Clayton } 14245a31471eSGreg Clayton else if (::strncmp (var_name_begin, "red}", strlen("red}")) == 0) 14255a31471eSGreg Clayton { 14265a31471eSGreg Clayton s.Printf ("%s%s%s", 14275a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14285a31471eSGreg Clayton lldb_utility::ansi::k_fg_red, 14295a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14305a31471eSGreg Clayton } 14315a31471eSGreg Clayton else if (::strncmp (var_name_begin, "green}", strlen("green}")) == 0) 14325a31471eSGreg Clayton { 14335a31471eSGreg Clayton s.Printf ("%s%s%s", 14345a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14355a31471eSGreg Clayton lldb_utility::ansi::k_fg_green, 14365a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14375a31471eSGreg Clayton } 14385a31471eSGreg Clayton else if (::strncmp (var_name_begin, "yellow}", strlen("yellow}")) == 0) 14395a31471eSGreg Clayton { 14405a31471eSGreg Clayton s.Printf ("%s%s%s", 14415a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14425a31471eSGreg Clayton lldb_utility::ansi::k_fg_yellow, 14435a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14445a31471eSGreg Clayton } 14455a31471eSGreg Clayton else if (::strncmp (var_name_begin, "blue}", strlen("blue}")) == 0) 14465a31471eSGreg Clayton { 14475a31471eSGreg Clayton s.Printf ("%s%s%s", 14485a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14495a31471eSGreg Clayton lldb_utility::ansi::k_fg_blue, 14505a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14515a31471eSGreg Clayton } 14525a31471eSGreg Clayton else if (::strncmp (var_name_begin, "purple}", strlen("purple}")) == 0) 14535a31471eSGreg Clayton { 14545a31471eSGreg Clayton s.Printf ("%s%s%s", 14555a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14565a31471eSGreg Clayton lldb_utility::ansi::k_fg_purple, 14575a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14585a31471eSGreg Clayton } 14595a31471eSGreg Clayton else if (::strncmp (var_name_begin, "cyan}", strlen("cyan}")) == 0) 14605a31471eSGreg Clayton { 14615a31471eSGreg Clayton s.Printf ("%s%s%s", 14625a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14635a31471eSGreg Clayton lldb_utility::ansi::k_fg_cyan, 14645a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14655a31471eSGreg Clayton } 14665a31471eSGreg Clayton else if (::strncmp (var_name_begin, "white}", strlen("white}")) == 0) 14675a31471eSGreg Clayton { 14685a31471eSGreg Clayton s.Printf ("%s%s%s", 14695a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14705a31471eSGreg Clayton lldb_utility::ansi::k_fg_white, 14715a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14725a31471eSGreg Clayton } 14735a31471eSGreg Clayton else 14745a31471eSGreg Clayton { 14755a31471eSGreg Clayton var_success = false; 14765a31471eSGreg Clayton } 14775a31471eSGreg Clayton } 14785a31471eSGreg Clayton else if (::strncmp (var_name_begin, "bg.", strlen("bg.")) == 0) 14795a31471eSGreg Clayton { 14805a31471eSGreg Clayton var_name_begin += strlen("bg."); // Skip the "bg." 14815a31471eSGreg Clayton if (::strncmp (var_name_begin, "black}", strlen("black}")) == 0) 14825a31471eSGreg Clayton { 14835a31471eSGreg Clayton s.Printf ("%s%s%s", 14845a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14855a31471eSGreg Clayton lldb_utility::ansi::k_bg_black, 14865a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14875a31471eSGreg Clayton } 14885a31471eSGreg Clayton else if (::strncmp (var_name_begin, "red}", strlen("red}")) == 0) 14895a31471eSGreg Clayton { 14905a31471eSGreg Clayton s.Printf ("%s%s%s", 14915a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14925a31471eSGreg Clayton lldb_utility::ansi::k_bg_red, 14935a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14945a31471eSGreg Clayton } 14955a31471eSGreg Clayton else if (::strncmp (var_name_begin, "green}", strlen("green}")) == 0) 14965a31471eSGreg Clayton { 14975a31471eSGreg Clayton s.Printf ("%s%s%s", 14985a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14995a31471eSGreg Clayton lldb_utility::ansi::k_bg_green, 15005a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15015a31471eSGreg Clayton } 15025a31471eSGreg Clayton else if (::strncmp (var_name_begin, "yellow}", strlen("yellow}")) == 0) 15035a31471eSGreg Clayton { 15045a31471eSGreg Clayton s.Printf ("%s%s%s", 15055a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15065a31471eSGreg Clayton lldb_utility::ansi::k_bg_yellow, 15075a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15085a31471eSGreg Clayton } 15095a31471eSGreg Clayton else if (::strncmp (var_name_begin, "blue}", strlen("blue}")) == 0) 15105a31471eSGreg Clayton { 15115a31471eSGreg Clayton s.Printf ("%s%s%s", 15125a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15135a31471eSGreg Clayton lldb_utility::ansi::k_bg_blue, 15145a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15155a31471eSGreg Clayton } 15165a31471eSGreg Clayton else if (::strncmp (var_name_begin, "purple}", strlen("purple}")) == 0) 15175a31471eSGreg Clayton { 15185a31471eSGreg Clayton s.Printf ("%s%s%s", 15195a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15205a31471eSGreg Clayton lldb_utility::ansi::k_bg_purple, 15215a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15225a31471eSGreg Clayton } 15235a31471eSGreg Clayton else if (::strncmp (var_name_begin, "cyan}", strlen("cyan}")) == 0) 15245a31471eSGreg Clayton { 15255a31471eSGreg Clayton s.Printf ("%s%s%s", 15265a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15275a31471eSGreg Clayton lldb_utility::ansi::k_bg_cyan, 15285a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15295a31471eSGreg Clayton } 15305a31471eSGreg Clayton else if (::strncmp (var_name_begin, "white}", strlen("white}")) == 0) 15315a31471eSGreg Clayton { 15325a31471eSGreg Clayton s.Printf ("%s%s%s", 15335a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15345a31471eSGreg Clayton lldb_utility::ansi::k_bg_white, 15355a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15365a31471eSGreg Clayton } 15375a31471eSGreg Clayton else 15385a31471eSGreg Clayton { 15395a31471eSGreg Clayton var_success = false; 15405a31471eSGreg Clayton } 15415a31471eSGreg Clayton } 15425a31471eSGreg Clayton else if (::strncmp (var_name_begin, "normal}", strlen ("normal}")) == 0) 15435a31471eSGreg Clayton { 15445a31471eSGreg Clayton s.Printf ("%s%s%s", 15455a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15465a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_normal, 15475a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15485a31471eSGreg Clayton } 15495a31471eSGreg Clayton else if (::strncmp (var_name_begin, "bold}", strlen("bold}")) == 0) 15505a31471eSGreg Clayton { 15515a31471eSGreg Clayton s.Printf ("%s%s%s", 15525a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15535a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_bold, 15545a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15555a31471eSGreg Clayton } 15565a31471eSGreg Clayton else if (::strncmp (var_name_begin, "faint}", strlen("faint}")) == 0) 15575a31471eSGreg Clayton { 15585a31471eSGreg Clayton s.Printf ("%s%s%s", 15595a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15605a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_faint, 15615a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15625a31471eSGreg Clayton } 15635a31471eSGreg Clayton else if (::strncmp (var_name_begin, "italic}", strlen("italic}")) == 0) 15645a31471eSGreg Clayton { 15655a31471eSGreg Clayton s.Printf ("%s%s%s", 15665a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15675a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_italic, 15685a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15695a31471eSGreg Clayton } 15705a31471eSGreg Clayton else if (::strncmp (var_name_begin, "underline}", strlen("underline}")) == 0) 15715a31471eSGreg Clayton { 15725a31471eSGreg Clayton s.Printf ("%s%s%s", 15735a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15745a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_underline, 15755a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15765a31471eSGreg Clayton } 15775a31471eSGreg Clayton else if (::strncmp (var_name_begin, "slow-blink}", strlen("slow-blink}")) == 0) 15785a31471eSGreg Clayton { 15795a31471eSGreg Clayton s.Printf ("%s%s%s", 15805a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15815a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_slow_blink, 15825a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15835a31471eSGreg Clayton } 15845a31471eSGreg Clayton else if (::strncmp (var_name_begin, "fast-blink}", strlen("fast-blink}")) == 0) 15855a31471eSGreg Clayton { 15865a31471eSGreg Clayton s.Printf ("%s%s%s", 15875a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15885a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_fast_blink, 15895a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15905a31471eSGreg Clayton } 15915a31471eSGreg Clayton else if (::strncmp (var_name_begin, "negative}", strlen("negative}")) == 0) 15925a31471eSGreg Clayton { 15935a31471eSGreg Clayton s.Printf ("%s%s%s", 15945a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15955a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_negative, 15965a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15975a31471eSGreg Clayton } 15985a31471eSGreg Clayton else if (::strncmp (var_name_begin, "conceal}", strlen("conceal}")) == 0) 15995a31471eSGreg Clayton { 16005a31471eSGreg Clayton s.Printf ("%s%s%s", 16015a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 16025a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_conceal, 16035a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 16045a31471eSGreg Clayton 16055a31471eSGreg Clayton } 16065a31471eSGreg Clayton else if (::strncmp (var_name_begin, "crossed-out}", strlen("crossed-out}")) == 0) 16075a31471eSGreg Clayton { 16085a31471eSGreg Clayton s.Printf ("%s%s%s", 16095a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 16105a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_crossed_out, 16115a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 16125a31471eSGreg Clayton } 16135a31471eSGreg Clayton else 16145a31471eSGreg Clayton { 16155a31471eSGreg Clayton var_success = false; 16165a31471eSGreg Clayton } 16175a31471eSGreg Clayton } 16181b654882SGreg Clayton break; 16191b654882SGreg Clayton 16201b654882SGreg Clayton case 'p': 16211b654882SGreg Clayton if (::strncmp (var_name_begin, "process.", strlen("process.")) == 0) 16221b654882SGreg Clayton { 1623c14ee32dSGreg Clayton if (exe_ctx) 1624c14ee32dSGreg Clayton { 1625c14ee32dSGreg Clayton Process *process = exe_ctx->GetProcessPtr(); 1626c14ee32dSGreg Clayton if (process) 16271b654882SGreg Clayton { 16281b654882SGreg Clayton var_name_begin += ::strlen ("process."); 16291b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 16301b654882SGreg Clayton { 163181c22f61SGreg Clayton s.Printf("%llu", process->GetID()); 16321b654882SGreg Clayton var_success = true; 16331b654882SGreg Clayton } 16341b654882SGreg Clayton else if ((::strncmp (var_name_begin, "name}", strlen("name}")) == 0) || 16351b654882SGreg Clayton (::strncmp (var_name_begin, "file.basename}", strlen("file.basename}")) == 0) || 16361b654882SGreg Clayton (::strncmp (var_name_begin, "file.fullpath}", strlen("file.fullpath}")) == 0)) 16371b654882SGreg Clayton { 1638c14ee32dSGreg Clayton Module *exe_module = process->GetTarget().GetExecutableModulePointer(); 1639aa149cbdSGreg Clayton if (exe_module) 16401b654882SGreg Clayton { 16411b654882SGreg Clayton if (var_name_begin[0] == 'n' || var_name_begin[5] == 'f') 16421b654882SGreg Clayton { 1643aa149cbdSGreg Clayton format_file_spec.GetFilename() = exe_module->GetFileSpec().GetFilename(); 16441b654882SGreg Clayton var_success = format_file_spec; 16451b654882SGreg Clayton } 16461b654882SGreg Clayton else 16471b654882SGreg Clayton { 1648aa149cbdSGreg Clayton format_file_spec = exe_module->GetFileSpec(); 16491b654882SGreg Clayton var_success = format_file_spec; 16501b654882SGreg Clayton } 16511b654882SGreg Clayton } 16521b654882SGreg Clayton } 16531b654882SGreg Clayton } 16541b654882SGreg Clayton } 1655c14ee32dSGreg Clayton } 16561b654882SGreg Clayton break; 16571b654882SGreg Clayton 16581b654882SGreg Clayton case 't': 16591b654882SGreg Clayton if (::strncmp (var_name_begin, "thread.", strlen("thread.")) == 0) 16601b654882SGreg Clayton { 1661c14ee32dSGreg Clayton if (exe_ctx) 1662c14ee32dSGreg Clayton { 1663c14ee32dSGreg Clayton Thread *thread = exe_ctx->GetThreadPtr(); 1664c14ee32dSGreg Clayton if (thread) 16651b654882SGreg Clayton { 16661b654882SGreg Clayton var_name_begin += ::strlen ("thread."); 16671b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 16681b654882SGreg Clayton { 166981c22f61SGreg Clayton s.Printf("0x%4.4llx", thread->GetID()); 16701b654882SGreg Clayton var_success = true; 16711b654882SGreg Clayton } 16721b654882SGreg Clayton else if (::strncmp (var_name_begin, "index}", strlen("index}")) == 0) 16731b654882SGreg Clayton { 1674c14ee32dSGreg Clayton s.Printf("%u", thread->GetIndexID()); 16751b654882SGreg Clayton var_success = true; 16761b654882SGreg Clayton } 16771b654882SGreg Clayton else if (::strncmp (var_name_begin, "name}", strlen("name}")) == 0) 16781b654882SGreg Clayton { 1679c14ee32dSGreg Clayton cstr = thread->GetName(); 16801b654882SGreg Clayton var_success = cstr && cstr[0]; 16811b654882SGreg Clayton if (var_success) 16821b654882SGreg Clayton s.PutCString(cstr); 16831b654882SGreg Clayton } 16841b654882SGreg Clayton else if (::strncmp (var_name_begin, "queue}", strlen("queue}")) == 0) 16851b654882SGreg Clayton { 1686c14ee32dSGreg Clayton cstr = thread->GetQueueName(); 16871b654882SGreg Clayton var_success = cstr && cstr[0]; 16881b654882SGreg Clayton if (var_success) 16891b654882SGreg Clayton s.PutCString(cstr); 16901b654882SGreg Clayton } 16911b654882SGreg Clayton else if (::strncmp (var_name_begin, "stop-reason}", strlen("stop-reason}")) == 0) 16921b654882SGreg Clayton { 1693c14ee32dSGreg Clayton StopInfoSP stop_info_sp = thread->GetStopInfo (); 1694b15bfc75SJim Ingham if (stop_info_sp) 16951b654882SGreg Clayton { 1696b15bfc75SJim Ingham cstr = stop_info_sp->GetDescription(); 16971b654882SGreg Clayton if (cstr && cstr[0]) 16981b654882SGreg Clayton { 16991b654882SGreg Clayton s.PutCString(cstr); 17001b654882SGreg Clayton var_success = true; 17011b654882SGreg Clayton } 17021b654882SGreg Clayton } 17031b654882SGreg Clayton } 1704*73ca05a2SJim Ingham else if (::strncmp (var_name_begin, "return-value}", strlen("return-value}")) == 0) 1705*73ca05a2SJim Ingham { 1706*73ca05a2SJim Ingham StopInfoSP stop_info_sp = thread->GetStopInfo (); 1707*73ca05a2SJim Ingham if (stop_info_sp) 1708*73ca05a2SJim Ingham { 1709*73ca05a2SJim Ingham ValueObjectSP return_valobj_sp = StopInfo::GetReturnValueObject (stop_info_sp); 1710*73ca05a2SJim Ingham if (return_valobj_sp) 1711*73ca05a2SJim Ingham { 1712*73ca05a2SJim Ingham cstr = return_valobj_sp->GetValueAsCString (); 1713*73ca05a2SJim Ingham if (cstr && cstr[0]) 1714*73ca05a2SJim Ingham { 1715*73ca05a2SJim Ingham s.PutCString(cstr); 1716*73ca05a2SJim Ingham var_success = true; 1717*73ca05a2SJim Ingham } 1718*73ca05a2SJim Ingham } 1719*73ca05a2SJim Ingham } 1720*73ca05a2SJim Ingham } 17211b654882SGreg Clayton } 17221b654882SGreg Clayton } 1723c14ee32dSGreg Clayton } 17241b654882SGreg Clayton else if (::strncmp (var_name_begin, "target.", strlen("target.")) == 0) 17251b654882SGreg Clayton { 17260603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 17270603aa9dSGreg Clayton if (target) 17281b654882SGreg Clayton { 17291b654882SGreg Clayton var_name_begin += ::strlen ("target."); 17301b654882SGreg Clayton if (::strncmp (var_name_begin, "arch}", strlen("arch}")) == 0) 17311b654882SGreg Clayton { 17321b654882SGreg Clayton ArchSpec arch (target->GetArchitecture ()); 17331b654882SGreg Clayton if (arch.IsValid()) 17341b654882SGreg Clayton { 173564195a2cSGreg Clayton s.PutCString (arch.GetArchitectureName()); 17361b654882SGreg Clayton var_success = true; 17371b654882SGreg Clayton } 17381b654882SGreg Clayton } 17391b654882SGreg Clayton } 17401b654882SGreg Clayton } 17411b654882SGreg Clayton break; 17421b654882SGreg Clayton 17431b654882SGreg Clayton 17441b654882SGreg Clayton case 'm': 17451b654882SGreg Clayton if (::strncmp (var_name_begin, "module.", strlen("module.")) == 0) 17461b654882SGreg Clayton { 17470603aa9dSGreg Clayton if (sc && sc->module_sp.get()) 17481b654882SGreg Clayton { 17490603aa9dSGreg Clayton Module *module = sc->module_sp.get(); 17501b654882SGreg Clayton var_name_begin += ::strlen ("module."); 17511b654882SGreg Clayton 17521b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 17531b654882SGreg Clayton { 17541b654882SGreg Clayton if (module->GetFileSpec()) 17551b654882SGreg Clayton { 17561b654882SGreg Clayton var_name_begin += ::strlen ("file."); 17571b654882SGreg Clayton 17581b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 17591b654882SGreg Clayton { 17601b654882SGreg Clayton format_file_spec.GetFilename() = module->GetFileSpec().GetFilename(); 17611b654882SGreg Clayton var_success = format_file_spec; 17621b654882SGreg Clayton } 17631b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 17641b654882SGreg Clayton { 17651b654882SGreg Clayton format_file_spec = module->GetFileSpec(); 17661b654882SGreg Clayton var_success = format_file_spec; 17671b654882SGreg Clayton } 17681b654882SGreg Clayton } 17691b654882SGreg Clayton } 17701b654882SGreg Clayton } 17711b654882SGreg Clayton } 17721b654882SGreg Clayton break; 17731b654882SGreg Clayton 17741b654882SGreg Clayton 17751b654882SGreg Clayton case 'f': 17761b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 17771b654882SGreg Clayton { 17781b654882SGreg Clayton if (sc && sc->comp_unit != NULL) 17791b654882SGreg Clayton { 17801b654882SGreg Clayton var_name_begin += ::strlen ("file."); 17811b654882SGreg Clayton 17821b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 17831b654882SGreg Clayton { 17841b654882SGreg Clayton format_file_spec.GetFilename() = sc->comp_unit->GetFilename(); 17851b654882SGreg Clayton var_success = format_file_spec; 17861b654882SGreg Clayton } 17871b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 17881b654882SGreg Clayton { 17891b654882SGreg Clayton format_file_spec = *sc->comp_unit; 17901b654882SGreg Clayton var_success = format_file_spec; 17911b654882SGreg Clayton } 17921b654882SGreg Clayton } 17931b654882SGreg Clayton } 17941b654882SGreg Clayton else if (::strncmp (var_name_begin, "frame.", strlen("frame.")) == 0) 17951b654882SGreg Clayton { 1796c14ee32dSGreg Clayton if (exe_ctx) 1797c14ee32dSGreg Clayton { 1798c14ee32dSGreg Clayton StackFrame *frame = exe_ctx->GetFramePtr(); 1799c14ee32dSGreg Clayton if (frame) 18001b654882SGreg Clayton { 18011b654882SGreg Clayton var_name_begin += ::strlen ("frame."); 18021b654882SGreg Clayton if (::strncmp (var_name_begin, "index}", strlen("index}")) == 0) 18031b654882SGreg Clayton { 1804c14ee32dSGreg Clayton s.Printf("%u", frame->GetFrameIndex()); 18051b654882SGreg Clayton var_success = true; 18061b654882SGreg Clayton } 18071b654882SGreg Clayton else if (::strncmp (var_name_begin, "pc}", strlen("pc}")) == 0) 18081b654882SGreg Clayton { 18091b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 18101b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_PC; 18111b654882SGreg Clayton var_success = true; 18121b654882SGreg Clayton } 18131b654882SGreg Clayton else if (::strncmp (var_name_begin, "sp}", strlen("sp}")) == 0) 18141b654882SGreg Clayton { 18151b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 18161b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_SP; 18171b654882SGreg Clayton var_success = true; 18181b654882SGreg Clayton } 18191b654882SGreg Clayton else if (::strncmp (var_name_begin, "fp}", strlen("fp}")) == 0) 18201b654882SGreg Clayton { 18211b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 18221b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_FP; 18231b654882SGreg Clayton var_success = true; 18241b654882SGreg Clayton } 18251b654882SGreg Clayton else if (::strncmp (var_name_begin, "flags}", strlen("flags}")) == 0) 18261b654882SGreg Clayton { 18271b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 18281b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_FLAGS; 18291b654882SGreg Clayton var_success = true; 18301b654882SGreg Clayton } 18311b654882SGreg Clayton else if (::strncmp (var_name_begin, "reg.", strlen ("reg.")) == 0) 18321b654882SGreg Clayton { 1833c14ee32dSGreg Clayton reg_ctx = frame->GetRegisterContext().get(); 18341b654882SGreg Clayton if (reg_ctx) 18351b654882SGreg Clayton { 18361b654882SGreg Clayton var_name_begin += ::strlen ("reg."); 18371b654882SGreg Clayton if (var_name_begin < var_name_end) 18381b654882SGreg Clayton { 18391b654882SGreg Clayton std::string reg_name (var_name_begin, var_name_end); 18401b654882SGreg Clayton reg_info = reg_ctx->GetRegisterInfoByName (reg_name.c_str()); 18411b654882SGreg Clayton if (reg_info) 18421b654882SGreg Clayton var_success = true; 18431b654882SGreg Clayton } 18441b654882SGreg Clayton } 18451b654882SGreg Clayton } 18461b654882SGreg Clayton } 18471b654882SGreg Clayton } 1848c14ee32dSGreg Clayton } 18491b654882SGreg Clayton else if (::strncmp (var_name_begin, "function.", strlen("function.")) == 0) 18501b654882SGreg Clayton { 18511b654882SGreg Clayton if (sc && (sc->function != NULL || sc->symbol != NULL)) 18521b654882SGreg Clayton { 18531b654882SGreg Clayton var_name_begin += ::strlen ("function."); 18541b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 18551b654882SGreg Clayton { 18561b654882SGreg Clayton if (sc->function) 185781c22f61SGreg Clayton s.Printf("function{0x%8.8llx}", sc->function->GetID()); 18581b654882SGreg Clayton else 18591b654882SGreg Clayton s.Printf("symbol[%u]", sc->symbol->GetID()); 18601b654882SGreg Clayton 18611b654882SGreg Clayton var_success = true; 18621b654882SGreg Clayton } 18631b654882SGreg Clayton else if (::strncmp (var_name_begin, "name}", strlen("name}")) == 0) 18641b654882SGreg Clayton { 18651b654882SGreg Clayton if (sc->function) 18661b654882SGreg Clayton cstr = sc->function->GetName().AsCString (NULL); 18671b654882SGreg Clayton else if (sc->symbol) 18681b654882SGreg Clayton cstr = sc->symbol->GetName().AsCString (NULL); 18691b654882SGreg Clayton if (cstr) 18701b654882SGreg Clayton { 18711b654882SGreg Clayton s.PutCString(cstr); 18720d9c9934SGreg Clayton 18730d9c9934SGreg Clayton if (sc->block) 18740d9c9934SGreg Clayton { 18750d9c9934SGreg Clayton Block *inline_block = sc->block->GetContainingInlinedBlock (); 18760d9c9934SGreg Clayton if (inline_block) 18770d9c9934SGreg Clayton { 18780d9c9934SGreg Clayton const InlineFunctionInfo *inline_info = sc->block->GetInlinedFunctionInfo(); 18790d9c9934SGreg Clayton if (inline_info) 18800d9c9934SGreg Clayton { 18810d9c9934SGreg Clayton s.PutCString(" [inlined] "); 18820d9c9934SGreg Clayton inline_info->GetName().Dump(&s); 18830d9c9934SGreg Clayton } 18840d9c9934SGreg Clayton } 18850d9c9934SGreg Clayton } 18861b654882SGreg Clayton var_success = true; 18871b654882SGreg Clayton } 18881b654882SGreg Clayton } 18891b654882SGreg Clayton else if (::strncmp (var_name_begin, "addr-offset}", strlen("addr-offset}")) == 0) 18901b654882SGreg Clayton { 18911b654882SGreg Clayton var_success = addr != NULL; 18921b654882SGreg Clayton if (var_success) 18931b654882SGreg Clayton { 18941b654882SGreg Clayton format_addr = *addr; 18951b654882SGreg Clayton calculate_format_addr_function_offset = true; 18961b654882SGreg Clayton } 18971b654882SGreg Clayton } 18981b654882SGreg Clayton else if (::strncmp (var_name_begin, "line-offset}", strlen("line-offset}")) == 0) 18991b654882SGreg Clayton { 19001b654882SGreg Clayton var_success = sc->line_entry.range.GetBaseAddress().IsValid(); 19011b654882SGreg Clayton if (var_success) 19021b654882SGreg Clayton { 19031b654882SGreg Clayton format_addr = sc->line_entry.range.GetBaseAddress(); 19041b654882SGreg Clayton calculate_format_addr_function_offset = true; 19051b654882SGreg Clayton } 19061b654882SGreg Clayton } 19071b654882SGreg Clayton else if (::strncmp (var_name_begin, "pc-offset}", strlen("pc-offset}")) == 0) 19081b654882SGreg Clayton { 1909c14ee32dSGreg Clayton StackFrame *frame = exe_ctx->GetFramePtr(); 1910c14ee32dSGreg Clayton var_success = frame != NULL; 19111b654882SGreg Clayton if (var_success) 19121b654882SGreg Clayton { 1913c14ee32dSGreg Clayton format_addr = frame->GetFrameCodeAddress(); 19141b654882SGreg Clayton calculate_format_addr_function_offset = true; 19151b654882SGreg Clayton } 19161b654882SGreg Clayton } 19171b654882SGreg Clayton } 19181b654882SGreg Clayton } 19191b654882SGreg Clayton break; 19201b654882SGreg Clayton 19211b654882SGreg Clayton case 'l': 19221b654882SGreg Clayton if (::strncmp (var_name_begin, "line.", strlen("line.")) == 0) 19231b654882SGreg Clayton { 19241b654882SGreg Clayton if (sc && sc->line_entry.IsValid()) 19251b654882SGreg Clayton { 19261b654882SGreg Clayton var_name_begin += ::strlen ("line."); 19271b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 19281b654882SGreg Clayton { 19291b654882SGreg Clayton var_name_begin += ::strlen ("file."); 19301b654882SGreg Clayton 19311b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 19321b654882SGreg Clayton { 19331b654882SGreg Clayton format_file_spec.GetFilename() = sc->line_entry.file.GetFilename(); 19341b654882SGreg Clayton var_success = format_file_spec; 19351b654882SGreg Clayton } 19361b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 19371b654882SGreg Clayton { 19381b654882SGreg Clayton format_file_spec = sc->line_entry.file; 19391b654882SGreg Clayton var_success = format_file_spec; 19401b654882SGreg Clayton } 19411b654882SGreg Clayton } 19421b654882SGreg Clayton else if (::strncmp (var_name_begin, "number}", strlen("number}")) == 0) 19431b654882SGreg Clayton { 19441b654882SGreg Clayton var_success = true; 19451b654882SGreg Clayton s.Printf("%u", sc->line_entry.line); 19461b654882SGreg Clayton } 19471b654882SGreg Clayton else if ((::strncmp (var_name_begin, "start-addr}", strlen("start-addr}")) == 0) || 19481b654882SGreg Clayton (::strncmp (var_name_begin, "end-addr}", strlen("end-addr}")) == 0)) 19491b654882SGreg Clayton { 19501b654882SGreg Clayton var_success = sc && sc->line_entry.range.GetBaseAddress().IsValid(); 19511b654882SGreg Clayton if (var_success) 19521b654882SGreg Clayton { 19531b654882SGreg Clayton format_addr = sc->line_entry.range.GetBaseAddress(); 19541b654882SGreg Clayton if (var_name_begin[0] == 'e') 19551b654882SGreg Clayton format_addr.Slide (sc->line_entry.range.GetByteSize()); 19561b654882SGreg Clayton } 19571b654882SGreg Clayton } 19581b654882SGreg Clayton } 19591b654882SGreg Clayton } 19601b654882SGreg Clayton break; 19611b654882SGreg Clayton } 19621b654882SGreg Clayton 19631b654882SGreg Clayton if (var_success) 19641b654882SGreg Clayton { 19651b654882SGreg Clayton // If format addr is valid, then we need to print an address 19661b654882SGreg Clayton if (reg_num != LLDB_INVALID_REGNUM) 19671b654882SGreg Clayton { 1968c14ee32dSGreg Clayton StackFrame *frame = exe_ctx->GetFramePtr(); 19691b654882SGreg Clayton // We have a register value to display... 19701b654882SGreg Clayton if (reg_num == LLDB_REGNUM_GENERIC_PC && reg_kind == eRegisterKindGeneric) 19711b654882SGreg Clayton { 1972c14ee32dSGreg Clayton format_addr = frame->GetFrameCodeAddress(); 19731b654882SGreg Clayton } 19741b654882SGreg Clayton else 19751b654882SGreg Clayton { 19761b654882SGreg Clayton if (reg_ctx == NULL) 1977c14ee32dSGreg Clayton reg_ctx = frame->GetRegisterContext().get(); 19781b654882SGreg Clayton 19791b654882SGreg Clayton if (reg_ctx) 19801b654882SGreg Clayton { 19811b654882SGreg Clayton if (reg_kind != kNumRegisterKinds) 19821b654882SGreg Clayton reg_num = reg_ctx->ConvertRegisterKindToRegisterNumber(reg_kind, reg_num); 19831b654882SGreg Clayton reg_info = reg_ctx->GetRegisterInfoAtIndex (reg_num); 19841b654882SGreg Clayton var_success = reg_info != NULL; 19851b654882SGreg Clayton } 19861b654882SGreg Clayton } 19871b654882SGreg Clayton } 19881b654882SGreg Clayton 19891b654882SGreg Clayton if (reg_info != NULL) 19901b654882SGreg Clayton { 19917349bd90SGreg Clayton RegisterValue reg_value; 19927349bd90SGreg Clayton var_success = reg_ctx->ReadRegister (reg_info, reg_value); 19937349bd90SGreg Clayton if (var_success) 19941b654882SGreg Clayton { 19959a8fa916SGreg Clayton reg_value.Dump(&s, reg_info, false, false, eFormatDefault); 19961b654882SGreg Clayton } 19971b654882SGreg Clayton } 19981b654882SGreg Clayton 19991b654882SGreg Clayton if (format_file_spec) 20001b654882SGreg Clayton { 20011b654882SGreg Clayton s << format_file_spec; 20021b654882SGreg Clayton } 20031b654882SGreg Clayton 20041b654882SGreg Clayton // If format addr is valid, then we need to print an address 20051b654882SGreg Clayton if (format_addr.IsValid()) 20061b654882SGreg Clayton { 20070603aa9dSGreg Clayton var_success = false; 20080603aa9dSGreg Clayton 20091b654882SGreg Clayton if (calculate_format_addr_function_offset) 20101b654882SGreg Clayton { 20111b654882SGreg Clayton Address func_addr; 20120603aa9dSGreg Clayton 20130603aa9dSGreg Clayton if (sc) 20140603aa9dSGreg Clayton { 20151b654882SGreg Clayton if (sc->function) 20160d9c9934SGreg Clayton { 20171b654882SGreg Clayton func_addr = sc->function->GetAddressRange().GetBaseAddress(); 20180d9c9934SGreg Clayton if (sc->block) 20190d9c9934SGreg Clayton { 20200d9c9934SGreg Clayton // Check to make sure we aren't in an inline 20210d9c9934SGreg Clayton // function. If we are, use the inline block 20220d9c9934SGreg Clayton // range that contains "format_addr" since 20230d9c9934SGreg Clayton // blocks can be discontiguous. 20240d9c9934SGreg Clayton Block *inline_block = sc->block->GetContainingInlinedBlock (); 20250d9c9934SGreg Clayton AddressRange inline_range; 20260d9c9934SGreg Clayton if (inline_block && inline_block->GetRangeContainingAddress (format_addr, inline_range)) 20270d9c9934SGreg Clayton func_addr = inline_range.GetBaseAddress(); 20280d9c9934SGreg Clayton } 20290d9c9934SGreg Clayton } 20301b654882SGreg Clayton else if (sc->symbol && sc->symbol->GetAddressRangePtr()) 20311b654882SGreg Clayton func_addr = sc->symbol->GetAddressRangePtr()->GetBaseAddress(); 20320603aa9dSGreg Clayton } 20331b654882SGreg Clayton 20340603aa9dSGreg Clayton if (func_addr.IsValid()) 20351b654882SGreg Clayton { 20361b654882SGreg Clayton if (func_addr.GetSection() == format_addr.GetSection()) 20371b654882SGreg Clayton { 20381b654882SGreg Clayton addr_t func_file_addr = func_addr.GetFileAddress(); 20391b654882SGreg Clayton addr_t addr_file_addr = format_addr.GetFileAddress(); 20401b654882SGreg Clayton if (addr_file_addr > func_file_addr) 20411b654882SGreg Clayton s.Printf(" + %llu", addr_file_addr - func_file_addr); 20421b654882SGreg Clayton else if (addr_file_addr < func_file_addr) 20431b654882SGreg Clayton s.Printf(" - %llu", func_file_addr - addr_file_addr); 20440603aa9dSGreg Clayton var_success = true; 20451b654882SGreg Clayton } 20461b654882SGreg Clayton else 20470603aa9dSGreg Clayton { 20480603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 20490603aa9dSGreg Clayton if (target) 20500603aa9dSGreg Clayton { 20510603aa9dSGreg Clayton addr_t func_load_addr = func_addr.GetLoadAddress (target); 20520603aa9dSGreg Clayton addr_t addr_load_addr = format_addr.GetLoadAddress (target); 20530603aa9dSGreg Clayton if (addr_load_addr > func_load_addr) 20540603aa9dSGreg Clayton s.Printf(" + %llu", addr_load_addr - func_load_addr); 20550603aa9dSGreg Clayton else if (addr_load_addr < func_load_addr) 20560603aa9dSGreg Clayton s.Printf(" - %llu", func_load_addr - addr_load_addr); 20570603aa9dSGreg Clayton var_success = true; 20580603aa9dSGreg Clayton } 20590603aa9dSGreg Clayton } 20601b654882SGreg Clayton } 20611b654882SGreg Clayton } 20621b654882SGreg Clayton else 20631b654882SGreg Clayton { 20640603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 20651b654882SGreg Clayton addr_t vaddr = LLDB_INVALID_ADDRESS; 20660603aa9dSGreg Clayton if (exe_ctx && !target->GetSectionLoadList().IsEmpty()) 20670603aa9dSGreg Clayton vaddr = format_addr.GetLoadAddress (target); 20681b654882SGreg Clayton if (vaddr == LLDB_INVALID_ADDRESS) 20691b654882SGreg Clayton vaddr = format_addr.GetFileAddress (); 20701b654882SGreg Clayton 20711b654882SGreg Clayton if (vaddr != LLDB_INVALID_ADDRESS) 20720603aa9dSGreg Clayton { 2073514487e8SGreg Clayton int addr_width = target->GetArchitecture().GetAddressByteSize() * 2; 207435f1a0d5SGreg Clayton if (addr_width == 0) 207535f1a0d5SGreg Clayton addr_width = 16; 207635f1a0d5SGreg Clayton s.Printf("0x%*.*llx", addr_width, addr_width, vaddr); 20770603aa9dSGreg Clayton var_success = true; 20780603aa9dSGreg Clayton } 20791b654882SGreg Clayton } 20801b654882SGreg Clayton } 20811b654882SGreg Clayton } 20821b654882SGreg Clayton 20831b654882SGreg Clayton if (var_success == false) 20841b654882SGreg Clayton success = false; 20851b654882SGreg Clayton } 20861b654882SGreg Clayton p = var_name_end; 20871b654882SGreg Clayton } 20881b654882SGreg Clayton else 20891b654882SGreg Clayton break; 20901b654882SGreg Clayton } 20911b654882SGreg Clayton else 20921b654882SGreg Clayton { 20931b654882SGreg Clayton // We got a dollar sign with no '{' after it, it must just be a dollar sign 20941b654882SGreg Clayton s.PutChar(*p); 20951b654882SGreg Clayton } 20961b654882SGreg Clayton } 20971b654882SGreg Clayton else if (*p == '\\') 20981b654882SGreg Clayton { 20991b654882SGreg Clayton ++p; // skip the slash 21001b654882SGreg Clayton switch (*p) 21011b654882SGreg Clayton { 21021b654882SGreg Clayton case 'a': s.PutChar ('\a'); break; 21031b654882SGreg Clayton case 'b': s.PutChar ('\b'); break; 21041b654882SGreg Clayton case 'f': s.PutChar ('\f'); break; 21051b654882SGreg Clayton case 'n': s.PutChar ('\n'); break; 21061b654882SGreg Clayton case 'r': s.PutChar ('\r'); break; 21071b654882SGreg Clayton case 't': s.PutChar ('\t'); break; 21081b654882SGreg Clayton case 'v': s.PutChar ('\v'); break; 21091b654882SGreg Clayton case '\'': s.PutChar ('\''); break; 21101b654882SGreg Clayton case '\\': s.PutChar ('\\'); break; 21111b654882SGreg Clayton case '0': 21121b654882SGreg Clayton // 1 to 3 octal chars 21131b654882SGreg Clayton { 21140603aa9dSGreg Clayton // Make a string that can hold onto the initial zero char, 21150603aa9dSGreg Clayton // up to 3 octal digits, and a terminating NULL. 21160603aa9dSGreg Clayton char oct_str[5] = { 0, 0, 0, 0, 0 }; 21170603aa9dSGreg Clayton 21180603aa9dSGreg Clayton int i; 21190603aa9dSGreg Clayton for (i=0; (p[i] >= '0' && p[i] <= '7') && i<4; ++i) 21200603aa9dSGreg Clayton oct_str[i] = p[i]; 21210603aa9dSGreg Clayton 21220603aa9dSGreg Clayton // We don't want to consume the last octal character since 21230603aa9dSGreg Clayton // the main for loop will do this for us, so we advance p by 21240603aa9dSGreg Clayton // one less than i (even if i is zero) 21250603aa9dSGreg Clayton p += i - 1; 21260603aa9dSGreg Clayton unsigned long octal_value = ::strtoul (oct_str, NULL, 8); 21270603aa9dSGreg Clayton if (octal_value <= UINT8_MAX) 21281b654882SGreg Clayton { 21290603aa9dSGreg Clayton char octal_char = octal_value; 21300603aa9dSGreg Clayton s.Write (&octal_char, 1); 21311b654882SGreg Clayton } 21321b654882SGreg Clayton } 21331b654882SGreg Clayton break; 21341b654882SGreg Clayton 21351b654882SGreg Clayton case 'x': 21361b654882SGreg Clayton // hex number in the format 21370603aa9dSGreg Clayton if (isxdigit(p[1])) 21381b654882SGreg Clayton { 21390603aa9dSGreg Clayton ++p; // Skip the 'x' 21401b654882SGreg Clayton 21410603aa9dSGreg Clayton // Make a string that can hold onto two hex chars plus a 21420603aa9dSGreg Clayton // NULL terminator 21431b654882SGreg Clayton char hex_str[3] = { 0,0,0 }; 21441b654882SGreg Clayton hex_str[0] = *p; 21450603aa9dSGreg Clayton if (isxdigit(p[1])) 21460603aa9dSGreg Clayton { 21470603aa9dSGreg Clayton ++p; // Skip the first of the two hex chars 21481b654882SGreg Clayton hex_str[1] = *p; 21490603aa9dSGreg Clayton } 21500603aa9dSGreg Clayton 21511b654882SGreg Clayton unsigned long hex_value = strtoul (hex_str, NULL, 16); 21520603aa9dSGreg Clayton if (hex_value <= UINT8_MAX) 21531b654882SGreg Clayton s.PutChar (hex_value); 21541b654882SGreg Clayton } 21551b654882SGreg Clayton else 21561b654882SGreg Clayton { 21570603aa9dSGreg Clayton s.PutChar('x'); 21581b654882SGreg Clayton } 21591b654882SGreg Clayton break; 21601b654882SGreg Clayton 21611b654882SGreg Clayton default: 21620603aa9dSGreg Clayton // Just desensitize any other character by just printing what 21630603aa9dSGreg Clayton // came after the '\' 21640603aa9dSGreg Clayton s << *p; 21651b654882SGreg Clayton break; 21661b654882SGreg Clayton 21671b654882SGreg Clayton } 21681b654882SGreg Clayton 21691b654882SGreg Clayton } 21701b654882SGreg Clayton } 21711b654882SGreg Clayton if (end) 21721b654882SGreg Clayton *end = p; 21731b654882SGreg Clayton return success; 21741b654882SGreg Clayton } 21751b654882SGreg Clayton 21761b654882SGreg Clayton #pragma mark Debugger::SettingsController 21771b654882SGreg Clayton 21783df9a8dfSCaroline Tice //-------------------------------------------------- 21791b654882SGreg Clayton // class Debugger::SettingsController 21803df9a8dfSCaroline Tice //-------------------------------------------------- 21813df9a8dfSCaroline Tice 21821b654882SGreg Clayton Debugger::SettingsController::SettingsController () : 21834d122c40SGreg Clayton UserSettingsController ("", UserSettingsControllerSP()) 21843df9a8dfSCaroline Tice { 218591123da2SCaroline Tice m_default_settings.reset (new DebuggerInstanceSettings (*this, false, 218691123da2SCaroline Tice InstanceSettings::GetDefaultName().AsCString())); 21873df9a8dfSCaroline Tice } 21883df9a8dfSCaroline Tice 21891b654882SGreg Clayton Debugger::SettingsController::~SettingsController () 21903df9a8dfSCaroline Tice { 21913df9a8dfSCaroline Tice } 21923df9a8dfSCaroline Tice 21933df9a8dfSCaroline Tice 21944d122c40SGreg Clayton InstanceSettingsSP 21951b654882SGreg Clayton Debugger::SettingsController::CreateInstanceSettings (const char *instance_name) 21963df9a8dfSCaroline Tice { 2197dbe54508SGreg Clayton DebuggerInstanceSettings *new_settings = new DebuggerInstanceSettings (*GetSettingsController(), 219891123da2SCaroline Tice false, instance_name); 21994d122c40SGreg Clayton InstanceSettingsSP new_settings_sp (new_settings); 22003df9a8dfSCaroline Tice return new_settings_sp; 22013df9a8dfSCaroline Tice } 22023df9a8dfSCaroline Tice 22031b654882SGreg Clayton #pragma mark DebuggerInstanceSettings 22043df9a8dfSCaroline Tice //-------------------------------------------------- 22053df9a8dfSCaroline Tice // class DebuggerInstanceSettings 22063df9a8dfSCaroline Tice //-------------------------------------------------- 22073df9a8dfSCaroline Tice 2208a7015092SGreg Clayton DebuggerInstanceSettings::DebuggerInstanceSettings 2209a7015092SGreg Clayton ( 2210a7015092SGreg Clayton UserSettingsController &owner, 2211a7015092SGreg Clayton bool live_instance, 2212a7015092SGreg Clayton const char *name 2213a7015092SGreg Clayton ) : 221485851ddeSGreg Clayton InstanceSettings (owner, name ? name : InstanceSettings::InvalidName().AsCString(), live_instance), 2215a7015092SGreg Clayton m_term_width (80), 2216e372b98dSGreg Clayton m_stop_source_before_count (3), 2217e372b98dSGreg Clayton m_stop_source_after_count (3), 2218e372b98dSGreg Clayton m_stop_disassembly_count (4), 2219e372b98dSGreg Clayton m_stop_disassembly_display (eStopDisassemblyTypeNoSource), 22203df9a8dfSCaroline Tice m_prompt (), 22210603aa9dSGreg Clayton m_frame_format (), 22220603aa9dSGreg Clayton m_thread_format (), 2223daccaa9eSCaroline Tice m_script_lang (), 22243bcdb29cSJim Ingham m_use_external_editor (false), 22253bcdb29cSJim Ingham m_auto_confirm_on (false) 22263df9a8dfSCaroline Tice { 2227f20e8239SCaroline Tice // CopyInstanceSettings is a pure virtual function in InstanceSettings; it therefore cannot be called 2228f20e8239SCaroline Tice // until the vtables for DebuggerInstanceSettings are properly set up, i.e. AFTER all the initializers. 2229f20e8239SCaroline Tice // For this reason it has to be called here, rather than in the initializer or in the parent constructor. 22309e41c15dSCaroline Tice // The same is true of CreateInstanceName(). 22319e41c15dSCaroline Tice 22329e41c15dSCaroline Tice if (GetInstanceName() == InstanceSettings::InvalidName()) 22339e41c15dSCaroline Tice { 22349e41c15dSCaroline Tice ChangeInstanceName (std::string (CreateInstanceName().AsCString())); 22359e41c15dSCaroline Tice m_owner.RegisterInstanceSettings (this); 22369e41c15dSCaroline Tice } 2237f20e8239SCaroline Tice 2238f20e8239SCaroline Tice if (live_instance) 22393df9a8dfSCaroline Tice { 22404d122c40SGreg Clayton const InstanceSettingsSP &pending_settings = m_owner.FindPendingSettings (m_instance_name); 22413df9a8dfSCaroline Tice CopyInstanceSettings (pending_settings, false); 22423df9a8dfSCaroline Tice } 22433df9a8dfSCaroline Tice } 22443df9a8dfSCaroline Tice 22453df9a8dfSCaroline Tice DebuggerInstanceSettings::DebuggerInstanceSettings (const DebuggerInstanceSettings &rhs) : 224699d0faf2SGreg Clayton InstanceSettings (*Debugger::GetSettingsController(), CreateInstanceName ().AsCString()), 22473df9a8dfSCaroline Tice m_prompt (rhs.m_prompt), 22480603aa9dSGreg Clayton m_frame_format (rhs.m_frame_format), 22490603aa9dSGreg Clayton m_thread_format (rhs.m_thread_format), 2250daccaa9eSCaroline Tice m_script_lang (rhs.m_script_lang), 22513bcdb29cSJim Ingham m_use_external_editor (rhs.m_use_external_editor), 22523bcdb29cSJim Ingham m_auto_confirm_on(rhs.m_auto_confirm_on) 22533df9a8dfSCaroline Tice { 22544d122c40SGreg Clayton const InstanceSettingsSP &pending_settings = m_owner.FindPendingSettings (m_instance_name); 22553df9a8dfSCaroline Tice CopyInstanceSettings (pending_settings, false); 22563df9a8dfSCaroline Tice m_owner.RemovePendingSettings (m_instance_name); 22573df9a8dfSCaroline Tice } 22583df9a8dfSCaroline Tice 22593df9a8dfSCaroline Tice DebuggerInstanceSettings::~DebuggerInstanceSettings () 22603df9a8dfSCaroline Tice { 22613df9a8dfSCaroline Tice } 22623df9a8dfSCaroline Tice 22633df9a8dfSCaroline Tice DebuggerInstanceSettings& 22643df9a8dfSCaroline Tice DebuggerInstanceSettings::operator= (const DebuggerInstanceSettings &rhs) 22653df9a8dfSCaroline Tice { 22663df9a8dfSCaroline Tice if (this != &rhs) 22673df9a8dfSCaroline Tice { 22681b654882SGreg Clayton m_term_width = rhs.m_term_width; 22693df9a8dfSCaroline Tice m_prompt = rhs.m_prompt; 22700603aa9dSGreg Clayton m_frame_format = rhs.m_frame_format; 22710603aa9dSGreg Clayton m_thread_format = rhs.m_thread_format; 22723df9a8dfSCaroline Tice m_script_lang = rhs.m_script_lang; 2273daccaa9eSCaroline Tice m_use_external_editor = rhs.m_use_external_editor; 22743bcdb29cSJim Ingham m_auto_confirm_on = rhs.m_auto_confirm_on; 22753df9a8dfSCaroline Tice } 22763df9a8dfSCaroline Tice 22773df9a8dfSCaroline Tice return *this; 22783df9a8dfSCaroline Tice } 22793df9a8dfSCaroline Tice 22801b654882SGreg Clayton bool 22811b654882SGreg Clayton DebuggerInstanceSettings::ValidTermWidthValue (const char *value, Error err) 22821b654882SGreg Clayton { 22831b654882SGreg Clayton bool valid = false; 22841b654882SGreg Clayton 22851b654882SGreg Clayton // Verify we have a value string. 22861b654882SGreg Clayton if (value == NULL || value[0] == '\0') 22871b654882SGreg Clayton { 228886edbf41SGreg Clayton err.SetErrorString ("missing value, can't set terminal width without a value"); 22891b654882SGreg Clayton } 22901b654882SGreg Clayton else 22911b654882SGreg Clayton { 22921b654882SGreg Clayton char *end = NULL; 22931b654882SGreg Clayton const uint32_t width = ::strtoul (value, &end, 0); 22941b654882SGreg Clayton 2295ea9fc181SJohnny Chen if (end && end[0] == '\0') 22961b654882SGreg Clayton { 2297433d7741SJohnny Chen if (width >= 10 && width <= 1024) 22981b654882SGreg Clayton valid = true; 22991b654882SGreg Clayton else 230086edbf41SGreg Clayton err.SetErrorString ("invalid term-width value; value must be between 10 and 1024"); 23011b654882SGreg Clayton } 23021b654882SGreg Clayton else 230386edbf41SGreg Clayton err.SetErrorStringWithFormat ("'%s' is not a valid unsigned integer string", value); 23041b654882SGreg Clayton } 23051b654882SGreg Clayton 23061b654882SGreg Clayton return valid; 23071b654882SGreg Clayton } 23081b654882SGreg Clayton 23091b654882SGreg Clayton 23103df9a8dfSCaroline Tice void 23113df9a8dfSCaroline Tice DebuggerInstanceSettings::UpdateInstanceSettingsVariable (const ConstString &var_name, 23123df9a8dfSCaroline Tice const char *index_value, 23133df9a8dfSCaroline Tice const char *value, 23143df9a8dfSCaroline Tice const ConstString &instance_name, 23153df9a8dfSCaroline Tice const SettingEntry &entry, 2316e0d378b3SGreg Clayton VarSetOperationType op, 23173df9a8dfSCaroline Tice Error &err, 23183df9a8dfSCaroline Tice bool pending) 23193df9a8dfSCaroline Tice { 23200603aa9dSGreg Clayton 23210603aa9dSGreg Clayton if (var_name == TermWidthVarName()) 23220603aa9dSGreg Clayton { 23230603aa9dSGreg Clayton if (ValidTermWidthValue (value, err)) 23240603aa9dSGreg Clayton { 23250603aa9dSGreg Clayton m_term_width = ::strtoul (value, NULL, 0); 23260603aa9dSGreg Clayton } 23270603aa9dSGreg Clayton } 23280603aa9dSGreg Clayton else if (var_name == PromptVarName()) 23293df9a8dfSCaroline Tice { 23303df9a8dfSCaroline Tice UserSettingsController::UpdateStringVariable (op, m_prompt, value, err); 23313df9a8dfSCaroline Tice if (!pending) 23323df9a8dfSCaroline Tice { 233349e2737eSCaroline Tice // 'instance_name' is actually (probably) in the form '[<instance_name>]'; if so, we need to 233449e2737eSCaroline Tice // strip off the brackets before passing it to BroadcastPromptChange. 233549e2737eSCaroline Tice 233649e2737eSCaroline Tice std::string tmp_instance_name (instance_name.AsCString()); 233749e2737eSCaroline Tice if ((tmp_instance_name[0] == '[') 233849e2737eSCaroline Tice && (tmp_instance_name[instance_name.GetLength() - 1] == ']')) 233949e2737eSCaroline Tice tmp_instance_name = tmp_instance_name.substr (1, instance_name.GetLength() - 2); 234049e2737eSCaroline Tice ConstString new_name (tmp_instance_name.c_str()); 234149e2737eSCaroline Tice 234249e2737eSCaroline Tice BroadcastPromptChange (new_name, m_prompt.c_str()); 23433df9a8dfSCaroline Tice } 23443df9a8dfSCaroline Tice } 23450603aa9dSGreg Clayton else if (var_name == GetFrameFormatName()) 23460603aa9dSGreg Clayton { 23470603aa9dSGreg Clayton UserSettingsController::UpdateStringVariable (op, m_frame_format, value, err); 23480603aa9dSGreg Clayton } 23490603aa9dSGreg Clayton else if (var_name == GetThreadFormatName()) 23500603aa9dSGreg Clayton { 23510603aa9dSGreg Clayton UserSettingsController::UpdateStringVariable (op, m_thread_format, value, err); 23520603aa9dSGreg Clayton } 23533df9a8dfSCaroline Tice else if (var_name == ScriptLangVarName()) 23543df9a8dfSCaroline Tice { 23553df9a8dfSCaroline Tice bool success; 23563df9a8dfSCaroline Tice m_script_lang = Args::StringToScriptLanguage (value, eScriptLanguageDefault, 23573df9a8dfSCaroline Tice &success); 23583df9a8dfSCaroline Tice } 2359daccaa9eSCaroline Tice else if (var_name == UseExternalEditorVarName ()) 2360daccaa9eSCaroline Tice { 2361385aa28cSGreg Clayton UserSettingsController::UpdateBooleanVariable (op, m_use_external_editor, value, false, err); 2362daccaa9eSCaroline Tice } 23633bcdb29cSJim Ingham else if (var_name == AutoConfirmName ()) 23643bcdb29cSJim Ingham { 2365385aa28cSGreg Clayton UserSettingsController::UpdateBooleanVariable (op, m_auto_confirm_on, value, false, err); 23663bcdb29cSJim Ingham } 2367e372b98dSGreg Clayton else if (var_name == StopSourceContextBeforeName ()) 2368e372b98dSGreg Clayton { 2369e372b98dSGreg Clayton uint32_t new_value = Args::StringToUInt32(value, UINT32_MAX, 10, NULL); 2370e372b98dSGreg Clayton if (new_value != UINT32_MAX) 2371e372b98dSGreg Clayton m_stop_source_before_count = new_value; 2372e372b98dSGreg Clayton else 2373e372b98dSGreg Clayton err.SetErrorStringWithFormat("invalid unsigned string value '%s' for the '%s' setting", value, StopSourceContextAfterName ().GetCString()); 2374e372b98dSGreg Clayton } 2375e372b98dSGreg Clayton else if (var_name == StopSourceContextAfterName ()) 2376e372b98dSGreg Clayton { 2377e372b98dSGreg Clayton uint32_t new_value = Args::StringToUInt32(value, UINT32_MAX, 10, NULL); 2378e372b98dSGreg Clayton if (new_value != UINT32_MAX) 2379e372b98dSGreg Clayton m_stop_source_after_count = new_value; 2380e372b98dSGreg Clayton else 2381e372b98dSGreg Clayton err.SetErrorStringWithFormat("invalid unsigned string value '%s' for the '%s' setting", value, StopSourceContextBeforeName ().GetCString()); 2382e372b98dSGreg Clayton } 2383e372b98dSGreg Clayton else if (var_name == StopDisassemblyCountName ()) 2384e372b98dSGreg Clayton { 2385e372b98dSGreg Clayton uint32_t new_value = Args::StringToUInt32(value, UINT32_MAX, 10, NULL); 2386e372b98dSGreg Clayton if (new_value != UINT32_MAX) 2387e372b98dSGreg Clayton m_stop_disassembly_count = new_value; 2388e372b98dSGreg Clayton else 2389e372b98dSGreg Clayton err.SetErrorStringWithFormat("invalid unsigned string value '%s' for the '%s' setting", value, StopDisassemblyCountName ().GetCString()); 2390e372b98dSGreg Clayton } 2391e372b98dSGreg Clayton else if (var_name == StopDisassemblyDisplayName ()) 2392e372b98dSGreg Clayton { 2393e372b98dSGreg Clayton int new_value; 2394e372b98dSGreg Clayton UserSettingsController::UpdateEnumVariable (g_show_disassembly_enum_values, &new_value, value, err); 2395e372b98dSGreg Clayton if (err.Success()) 2396e372b98dSGreg Clayton m_stop_disassembly_display = (StopDisassemblyType)new_value; 2397e372b98dSGreg Clayton } 23983df9a8dfSCaroline Tice } 23993df9a8dfSCaroline Tice 240012cecd74SCaroline Tice bool 24013df9a8dfSCaroline Tice DebuggerInstanceSettings::GetInstanceSettingsValue (const SettingEntry &entry, 24023df9a8dfSCaroline Tice const ConstString &var_name, 2403daccaa9eSCaroline Tice StringList &value, 240412cecd74SCaroline Tice Error *err) 24053df9a8dfSCaroline Tice { 24063df9a8dfSCaroline Tice if (var_name == PromptVarName()) 24073df9a8dfSCaroline Tice { 24080603aa9dSGreg Clayton value.AppendString (m_prompt.c_str(), m_prompt.size()); 24093df9a8dfSCaroline Tice 24103df9a8dfSCaroline Tice } 24113df9a8dfSCaroline Tice else if (var_name == ScriptLangVarName()) 24123df9a8dfSCaroline Tice { 24133df9a8dfSCaroline Tice value.AppendString (ScriptInterpreter::LanguageToString (m_script_lang).c_str()); 24143df9a8dfSCaroline Tice } 2415101c7c20SCaroline Tice else if (var_name == TermWidthVarName()) 2416101c7c20SCaroline Tice { 2417101c7c20SCaroline Tice StreamString width_str; 2418e372b98dSGreg Clayton width_str.Printf ("%u", m_term_width); 2419101c7c20SCaroline Tice value.AppendString (width_str.GetData()); 2420101c7c20SCaroline Tice } 24210603aa9dSGreg Clayton else if (var_name == GetFrameFormatName ()) 24220603aa9dSGreg Clayton { 24230603aa9dSGreg Clayton value.AppendString(m_frame_format.c_str(), m_frame_format.size()); 24240603aa9dSGreg Clayton } 24250603aa9dSGreg Clayton else if (var_name == GetThreadFormatName ()) 24260603aa9dSGreg Clayton { 24270603aa9dSGreg Clayton value.AppendString(m_thread_format.c_str(), m_thread_format.size()); 24280603aa9dSGreg Clayton } 2429daccaa9eSCaroline Tice else if (var_name == UseExternalEditorVarName()) 2430daccaa9eSCaroline Tice { 2431daccaa9eSCaroline Tice if (m_use_external_editor) 2432daccaa9eSCaroline Tice value.AppendString ("true"); 2433daccaa9eSCaroline Tice else 2434daccaa9eSCaroline Tice value.AppendString ("false"); 2435daccaa9eSCaroline Tice } 24363bcdb29cSJim Ingham else if (var_name == AutoConfirmName()) 24373bcdb29cSJim Ingham { 24383bcdb29cSJim Ingham if (m_auto_confirm_on) 24393bcdb29cSJim Ingham value.AppendString ("true"); 24403bcdb29cSJim Ingham else 24413bcdb29cSJim Ingham value.AppendString ("false"); 24423bcdb29cSJim Ingham } 2443e372b98dSGreg Clayton else if (var_name == StopSourceContextAfterName ()) 2444e372b98dSGreg Clayton { 2445e372b98dSGreg Clayton StreamString strm; 2446e372b98dSGreg Clayton strm.Printf ("%u", m_stop_source_before_count); 2447e372b98dSGreg Clayton value.AppendString (strm.GetData()); 2448e372b98dSGreg Clayton } 2449e372b98dSGreg Clayton else if (var_name == StopSourceContextBeforeName ()) 2450e372b98dSGreg Clayton { 2451e372b98dSGreg Clayton StreamString strm; 2452e372b98dSGreg Clayton strm.Printf ("%u", m_stop_source_after_count); 2453e372b98dSGreg Clayton value.AppendString (strm.GetData()); 2454e372b98dSGreg Clayton } 2455e372b98dSGreg Clayton else if (var_name == StopDisassemblyCountName ()) 2456e372b98dSGreg Clayton { 2457e372b98dSGreg Clayton StreamString strm; 2458e372b98dSGreg Clayton strm.Printf ("%u", m_stop_disassembly_count); 2459e372b98dSGreg Clayton value.AppendString (strm.GetData()); 2460e372b98dSGreg Clayton } 2461e372b98dSGreg Clayton else if (var_name == StopDisassemblyDisplayName ()) 2462e372b98dSGreg Clayton { 2463e372b98dSGreg Clayton if (m_stop_disassembly_display >= eStopDisassemblyTypeNever && m_stop_disassembly_display <= eStopDisassemblyTypeAlways) 2464e372b98dSGreg Clayton value.AppendString (g_show_disassembly_enum_values[m_stop_disassembly_display].string_value); 2465e372b98dSGreg Clayton else 2466e372b98dSGreg Clayton value.AppendString ("<invalid>"); 2467e372b98dSGreg Clayton } 2468daccaa9eSCaroline Tice else 246912cecd74SCaroline Tice { 247012cecd74SCaroline Tice if (err) 247112cecd74SCaroline Tice err->SetErrorStringWithFormat ("unrecognized variable name '%s'", var_name.AsCString()); 247212cecd74SCaroline Tice return false; 247312cecd74SCaroline Tice } 247412cecd74SCaroline Tice return true; 24753df9a8dfSCaroline Tice } 24763df9a8dfSCaroline Tice 24773df9a8dfSCaroline Tice void 24784d122c40SGreg Clayton DebuggerInstanceSettings::CopyInstanceSettings (const InstanceSettingsSP &new_settings, 24793df9a8dfSCaroline Tice bool pending) 24803df9a8dfSCaroline Tice { 24813df9a8dfSCaroline Tice if (new_settings.get() == NULL) 24823df9a8dfSCaroline Tice return; 24833df9a8dfSCaroline Tice 24843df9a8dfSCaroline Tice DebuggerInstanceSettings *new_debugger_settings = (DebuggerInstanceSettings *) new_settings.get(); 24853df9a8dfSCaroline Tice 24863df9a8dfSCaroline Tice m_prompt = new_debugger_settings->m_prompt; 24873df9a8dfSCaroline Tice if (!pending) 248849e2737eSCaroline Tice { 248949e2737eSCaroline Tice // 'instance_name' is actually (probably) in the form '[<instance_name>]'; if so, we need to 249049e2737eSCaroline Tice // strip off the brackets before passing it to BroadcastPromptChange. 249149e2737eSCaroline Tice 249249e2737eSCaroline Tice std::string tmp_instance_name (m_instance_name.AsCString()); 249349e2737eSCaroline Tice if ((tmp_instance_name[0] == '[') 249449e2737eSCaroline Tice && (tmp_instance_name[m_instance_name.GetLength() - 1] == ']')) 249549e2737eSCaroline Tice tmp_instance_name = tmp_instance_name.substr (1, m_instance_name.GetLength() - 2); 249649e2737eSCaroline Tice ConstString new_name (tmp_instance_name.c_str()); 249749e2737eSCaroline Tice 249849e2737eSCaroline Tice BroadcastPromptChange (new_name, m_prompt.c_str()); 249949e2737eSCaroline Tice } 25000603aa9dSGreg Clayton m_frame_format = new_debugger_settings->m_frame_format; 25010603aa9dSGreg Clayton m_thread_format = new_debugger_settings->m_thread_format; 2502daccaa9eSCaroline Tice m_term_width = new_debugger_settings->m_term_width; 25033df9a8dfSCaroline Tice m_script_lang = new_debugger_settings->m_script_lang; 2504daccaa9eSCaroline Tice m_use_external_editor = new_debugger_settings->m_use_external_editor; 25053bcdb29cSJim Ingham m_auto_confirm_on = new_debugger_settings->m_auto_confirm_on; 25063df9a8dfSCaroline Tice } 25073df9a8dfSCaroline Tice 25083df9a8dfSCaroline Tice 25093df9a8dfSCaroline Tice bool 25103df9a8dfSCaroline Tice DebuggerInstanceSettings::BroadcastPromptChange (const ConstString &instance_name, const char *new_prompt) 25113df9a8dfSCaroline Tice { 25123df9a8dfSCaroline Tice std::string tmp_prompt; 25133df9a8dfSCaroline Tice 25143df9a8dfSCaroline Tice if (new_prompt != NULL) 25153df9a8dfSCaroline Tice { 25163df9a8dfSCaroline Tice tmp_prompt = new_prompt ; 25173df9a8dfSCaroline Tice int len = tmp_prompt.size(); 25183df9a8dfSCaroline Tice if (len > 1 25193df9a8dfSCaroline Tice && (tmp_prompt[0] == '\'' || tmp_prompt[0] == '"') 25203df9a8dfSCaroline Tice && (tmp_prompt[len-1] == tmp_prompt[0])) 25213df9a8dfSCaroline Tice { 25223df9a8dfSCaroline Tice tmp_prompt = tmp_prompt.substr(1,len-2); 25233df9a8dfSCaroline Tice } 25243df9a8dfSCaroline Tice len = tmp_prompt.size(); 25253df9a8dfSCaroline Tice if (tmp_prompt[len-1] != ' ') 25263df9a8dfSCaroline Tice tmp_prompt.append(" "); 25273df9a8dfSCaroline Tice } 25283df9a8dfSCaroline Tice EventSP new_event_sp; 25293df9a8dfSCaroline Tice new_event_sp.reset (new Event(CommandInterpreter::eBroadcastBitResetPrompt, 25303df9a8dfSCaroline Tice new EventDataBytes (tmp_prompt.c_str()))); 25313df9a8dfSCaroline Tice 25323df9a8dfSCaroline Tice if (instance_name.GetLength() != 0) 25333df9a8dfSCaroline Tice { 25343df9a8dfSCaroline Tice // Set prompt for a particular instance. 25353df9a8dfSCaroline Tice Debugger *dbg = Debugger::FindDebuggerWithInstanceName (instance_name).get(); 25363df9a8dfSCaroline Tice if (dbg != NULL) 25373df9a8dfSCaroline Tice { 25383df9a8dfSCaroline Tice dbg->GetCommandInterpreter().BroadcastEvent (new_event_sp); 25393df9a8dfSCaroline Tice } 25403df9a8dfSCaroline Tice } 25413df9a8dfSCaroline Tice 25423df9a8dfSCaroline Tice return true; 25433df9a8dfSCaroline Tice } 25443df9a8dfSCaroline Tice 25453df9a8dfSCaroline Tice const ConstString 25463df9a8dfSCaroline Tice DebuggerInstanceSettings::CreateInstanceName () 25473df9a8dfSCaroline Tice { 25483df9a8dfSCaroline Tice static int instance_count = 1; 25493df9a8dfSCaroline Tice StreamString sstr; 25503df9a8dfSCaroline Tice 25513df9a8dfSCaroline Tice sstr.Printf ("debugger_%d", instance_count); 25523df9a8dfSCaroline Tice ++instance_count; 25533df9a8dfSCaroline Tice 25543df9a8dfSCaroline Tice const ConstString ret_val (sstr.GetData()); 25553df9a8dfSCaroline Tice 25563df9a8dfSCaroline Tice return ret_val; 25573df9a8dfSCaroline Tice } 25583df9a8dfSCaroline Tice 25593bcdb29cSJim Ingham 25603df9a8dfSCaroline Tice //-------------------------------------------------- 25611b654882SGreg Clayton // SettingsController Variable Tables 25623df9a8dfSCaroline Tice //-------------------------------------------------- 25633df9a8dfSCaroline Tice 25643df9a8dfSCaroline Tice 25653df9a8dfSCaroline Tice SettingEntry 25661b654882SGreg Clayton Debugger::SettingsController::global_settings_table[] = 25673df9a8dfSCaroline Tice { 25683df9a8dfSCaroline Tice //{ "var-name", var-type, "default", enum-table, init'd, hidden, "help-text"}, 2569101c7c20SCaroline Tice // The Debugger level global table should always be empty; all Debugger settable variables should be instance 2570101c7c20SCaroline Tice // variables. 25713df9a8dfSCaroline Tice { NULL, eSetVarTypeNone, NULL, NULL, 0, 0, NULL } 25723df9a8dfSCaroline Tice }; 25733df9a8dfSCaroline Tice 2574bb562b13SGreg Clayton #define MODULE_WITH_FUNC "{ ${module.file.basename}{`${function.name}${function.pc-offset}}}" 25750603aa9dSGreg Clayton #define FILE_AND_LINE "{ at ${line.file.basename}:${line.number}}" 25763df9a8dfSCaroline Tice 25770603aa9dSGreg Clayton #define DEFAULT_THREAD_FORMAT "thread #${thread.index}: tid = ${thread.id}"\ 25780603aa9dSGreg Clayton "{, ${frame.pc}}"\ 25790603aa9dSGreg Clayton MODULE_WITH_FUNC\ 2580cf4b9078SGreg Clayton FILE_AND_LINE\ 25810603aa9dSGreg Clayton "{, stop reason = ${thread.stop-reason}}"\ 2582*73ca05a2SJim Ingham "{, return value = ${thread.return-value}}"\ 25830603aa9dSGreg Clayton "\\n" 25840603aa9dSGreg Clayton 2585315d2cabSGreg Clayton //#define DEFAULT_THREAD_FORMAT "thread #${thread.index}: tid = ${thread.id}"\ 2586315d2cabSGreg Clayton // "{, ${frame.pc}}"\ 2587315d2cabSGreg Clayton // MODULE_WITH_FUNC\ 2588315d2cabSGreg Clayton // FILE_AND_LINE\ 2589315d2cabSGreg Clayton // "{, stop reason = ${thread.stop-reason}}"\ 2590315d2cabSGreg Clayton // "{, name = ${thread.name}}"\ 2591315d2cabSGreg Clayton // "{, queue = ${thread.queue}}"\ 2592315d2cabSGreg Clayton // "\\n" 2593315d2cabSGreg Clayton 25940603aa9dSGreg Clayton #define DEFAULT_FRAME_FORMAT "frame #${frame.index}: ${frame.pc}"\ 25950603aa9dSGreg Clayton MODULE_WITH_FUNC\ 25960603aa9dSGreg Clayton FILE_AND_LINE\ 25970603aa9dSGreg Clayton "\\n" 25983df9a8dfSCaroline Tice 25993df9a8dfSCaroline Tice SettingEntry 26001b654882SGreg Clayton Debugger::SettingsController::instance_settings_table[] = 26013df9a8dfSCaroline Tice { 26020603aa9dSGreg Clayton // NAME Setting variable type Default Enum Init'd Hidden Help 26030603aa9dSGreg Clayton // ======================= ======================= ====================== ==== ====== ====== ====================== 26040603aa9dSGreg Clayton { "frame-format", eSetVarTypeString, DEFAULT_FRAME_FORMAT, NULL, false, false, "The default frame format string to use when displaying thread information." }, 26053bcdb29cSJim Ingham { "prompt", eSetVarTypeString, "(lldb) ", NULL, false, false, "The debugger command line prompt displayed for the user." }, 26063bcdb29cSJim Ingham { "script-lang", eSetVarTypeString, "python", NULL, false, false, "The script language to be used for evaluating user-written scripts." }, 26073bcdb29cSJim Ingham { "term-width", eSetVarTypeInt, "80" , NULL, false, false, "The maximum number of columns to use for displaying text." }, 26080603aa9dSGreg Clayton { "thread-format", eSetVarTypeString, DEFAULT_THREAD_FORMAT, NULL, false, false, "The default thread format string to use when displaying thread information." }, 260906e827ccSJim Ingham { "use-external-editor", eSetVarTypeBoolean, "false", NULL, false, false, "Whether to use an external editor or not." }, 261006e827ccSJim Ingham { "auto-confirm", eSetVarTypeBoolean, "false", NULL, false, false, "If true all confirmation prompts will receive their default reply." }, 2611e372b98dSGreg Clayton { "stop-line-count-before",eSetVarTypeInt, "3", NULL, false, false, "The number of sources lines to display that come before the current source line when displaying a stopped context." }, 2612e372b98dSGreg Clayton { "stop-line-count-after", eSetVarTypeInt, "3", NULL, false, false, "The number of sources lines to display that come after the current source line when displaying a stopped context." }, 2613e372b98dSGreg Clayton { "stop-disassembly-count", eSetVarTypeInt, "0", NULL, false, false, "The number of disassembly lines to show when displaying a stopped context." }, 2614e372b98dSGreg Clayton { "stop-disassembly-display", eSetVarTypeEnum, "no-source", g_show_disassembly_enum_values, false, false, "Control when to display disassembly when displaying a stopped context." }, 26150603aa9dSGreg Clayton { NULL, eSetVarTypeNone, NULL, NULL, false, false, NULL } 26163df9a8dfSCaroline Tice }; 2617