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 641b654882SGreg Clayton #pragma mark Debugger 651b654882SGreg Clayton 6699d0faf2SGreg Clayton UserSettingsControllerSP & 6799d0faf2SGreg Clayton Debugger::GetSettingsController () 6899d0faf2SGreg Clayton { 6999d0faf2SGreg Clayton static UserSettingsControllerSP g_settings_controller; 7099d0faf2SGreg Clayton return g_settings_controller; 7199d0faf2SGreg Clayton } 7299d0faf2SGreg Clayton 732f88aadfSCaroline Tice int 742f88aadfSCaroline Tice Debugger::TestDebuggerRefCount () 752f88aadfSCaroline Tice { 762f88aadfSCaroline Tice return g_shared_debugger_refcount; 772f88aadfSCaroline Tice } 782f88aadfSCaroline Tice 7930fdc8d8SChris Lattner void 8030fdc8d8SChris Lattner Debugger::Initialize () 8130fdc8d8SChris Lattner { 826611103cSGreg Clayton if (g_shared_debugger_refcount == 0) 8399d0faf2SGreg Clayton { 84dbe54508SGreg Clayton lldb_private::Initialize(); 8599d0faf2SGreg Clayton } 866611103cSGreg Clayton g_shared_debugger_refcount++; 8799d0faf2SGreg Clayton 8830fdc8d8SChris Lattner } 8930fdc8d8SChris Lattner 9030fdc8d8SChris Lattner void 9130fdc8d8SChris Lattner Debugger::Terminate () 9230fdc8d8SChris Lattner { 936611103cSGreg Clayton if (g_shared_debugger_refcount > 0) 946611103cSGreg Clayton { 9530fdc8d8SChris Lattner g_shared_debugger_refcount--; 9630fdc8d8SChris Lattner if (g_shared_debugger_refcount == 0) 9730fdc8d8SChris Lattner { 98dbe54508SGreg Clayton lldb_private::WillTerminate(); 99dbe54508SGreg Clayton lldb_private::Terminate(); 1006760a517SCaroline Tice 10199d0faf2SGreg Clayton // Clear our master list of debugger objects 10299d0faf2SGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 10399d0faf2SGreg Clayton GetDebuggerList().clear(); 10430fdc8d8SChris Lattner } 1056760a517SCaroline Tice } 1066760a517SCaroline Tice } 10730fdc8d8SChris Lattner 10820bd37f7SCaroline Tice void 10920bd37f7SCaroline Tice Debugger::SettingsInitialize () 11020bd37f7SCaroline Tice { 11120bd37f7SCaroline Tice static bool g_initialized = false; 11220bd37f7SCaroline Tice 11320bd37f7SCaroline Tice if (!g_initialized) 11420bd37f7SCaroline Tice { 11520bd37f7SCaroline Tice g_initialized = true; 11620bd37f7SCaroline Tice UserSettingsControllerSP &usc = GetSettingsController(); 11720bd37f7SCaroline Tice usc.reset (new SettingsController); 11820bd37f7SCaroline Tice UserSettingsController::InitializeSettingsController (usc, 11920bd37f7SCaroline Tice SettingsController::global_settings_table, 12020bd37f7SCaroline Tice SettingsController::instance_settings_table); 12120bd37f7SCaroline Tice // Now call SettingsInitialize for each settings 'child' of Debugger 12220bd37f7SCaroline Tice Target::SettingsInitialize (); 12320bd37f7SCaroline Tice } 12420bd37f7SCaroline Tice } 12520bd37f7SCaroline Tice 12620bd37f7SCaroline Tice void 12720bd37f7SCaroline Tice Debugger::SettingsTerminate () 12820bd37f7SCaroline Tice { 12920bd37f7SCaroline Tice 13020bd37f7SCaroline Tice // Must call SettingsTerminate() for each settings 'child' of Debugger, before terminating the Debugger's 13120bd37f7SCaroline Tice // Settings. 13220bd37f7SCaroline Tice 13320bd37f7SCaroline Tice Target::SettingsTerminate (); 13420bd37f7SCaroline Tice 13520bd37f7SCaroline Tice // Now terminate the Debugger Settings. 13620bd37f7SCaroline Tice 13720bd37f7SCaroline Tice UserSettingsControllerSP &usc = GetSettingsController(); 13820bd37f7SCaroline Tice UserSettingsController::FinalizeSettingsController (usc); 13920bd37f7SCaroline Tice usc.reset(); 14020bd37f7SCaroline Tice } 14120bd37f7SCaroline Tice 1426611103cSGreg Clayton DebuggerSP 1436611103cSGreg Clayton Debugger::CreateInstance () 1446611103cSGreg Clayton { 1456611103cSGreg Clayton DebuggerSP debugger_sp (new Debugger); 1466611103cSGreg Clayton // Scope for locker 1476611103cSGreg Clayton { 1486611103cSGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 1496611103cSGreg Clayton GetDebuggerList().push_back(debugger_sp); 1506611103cSGreg Clayton } 1516611103cSGreg Clayton return debugger_sp; 1526611103cSGreg Clayton } 1536611103cSGreg Clayton 154e02657b1SCaroline Tice void 1554d122c40SGreg Clayton Debugger::Destroy (DebuggerSP &debugger_sp) 156e02657b1SCaroline Tice { 157e02657b1SCaroline Tice if (debugger_sp.get() == NULL) 158e02657b1SCaroline Tice return; 159e02657b1SCaroline Tice 1608314c525SJim Ingham debugger_sp->Clear(); 1618314c525SJim Ingham 162e02657b1SCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 163e02657b1SCaroline Tice DebuggerList &debugger_list = GetDebuggerList (); 164e02657b1SCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 165e02657b1SCaroline Tice for (pos = debugger_list.begin (); pos != end; ++pos) 166e02657b1SCaroline Tice { 167e02657b1SCaroline Tice if ((*pos).get() == debugger_sp.get()) 168e02657b1SCaroline Tice { 169e02657b1SCaroline Tice debugger_list.erase (pos); 170e02657b1SCaroline Tice return; 171e02657b1SCaroline Tice } 172e02657b1SCaroline Tice } 173e02657b1SCaroline Tice } 174e02657b1SCaroline Tice 1754d122c40SGreg Clayton DebuggerSP 1766611103cSGreg Clayton Debugger::GetSP () 1776611103cSGreg Clayton { 1784d122c40SGreg Clayton // This object contains an instrusive ref count base class so we can 1794d122c40SGreg Clayton // easily make a shared pointer to this object 1804d122c40SGreg Clayton return DebuggerSP (this); 1816611103cSGreg Clayton } 1826611103cSGreg Clayton 1834d122c40SGreg Clayton DebuggerSP 1843df9a8dfSCaroline Tice Debugger::FindDebuggerWithInstanceName (const ConstString &instance_name) 1853df9a8dfSCaroline Tice { 1864d122c40SGreg Clayton DebuggerSP debugger_sp; 1873df9a8dfSCaroline Tice 1883df9a8dfSCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 1893df9a8dfSCaroline Tice DebuggerList &debugger_list = GetDebuggerList(); 1903df9a8dfSCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 1913df9a8dfSCaroline Tice 1923df9a8dfSCaroline Tice for (pos = debugger_list.begin(); pos != end; ++pos) 1933df9a8dfSCaroline Tice { 1943df9a8dfSCaroline Tice if ((*pos).get()->m_instance_name == instance_name) 1953df9a8dfSCaroline Tice { 1963df9a8dfSCaroline Tice debugger_sp = *pos; 1973df9a8dfSCaroline Tice break; 1983df9a8dfSCaroline Tice } 1993df9a8dfSCaroline Tice } 2003df9a8dfSCaroline Tice return debugger_sp; 2013df9a8dfSCaroline Tice } 2026611103cSGreg Clayton 2036611103cSGreg Clayton TargetSP 2046611103cSGreg Clayton Debugger::FindTargetWithProcessID (lldb::pid_t pid) 2056611103cSGreg Clayton { 2064d122c40SGreg Clayton TargetSP target_sp; 2076611103cSGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 2086611103cSGreg Clayton DebuggerList &debugger_list = GetDebuggerList(); 2096611103cSGreg Clayton DebuggerList::iterator pos, end = debugger_list.end(); 2106611103cSGreg Clayton for (pos = debugger_list.begin(); pos != end; ++pos) 2116611103cSGreg Clayton { 2126611103cSGreg Clayton target_sp = (*pos)->GetTargetList().FindTargetWithProcessID (pid); 2136611103cSGreg Clayton if (target_sp) 2146611103cSGreg Clayton break; 2156611103cSGreg Clayton } 2166611103cSGreg Clayton return target_sp; 2176611103cSGreg Clayton } 2186611103cSGreg Clayton 2196611103cSGreg Clayton 22030fdc8d8SChris Lattner Debugger::Debugger () : 221ebc1bb27SCaroline Tice UserID (g_unique_id++), 222dbe54508SGreg Clayton DebuggerInstanceSettings (*GetSettingsController()), 223d46c87a1SGreg Clayton m_input_comm("debugger.input"), 22430fdc8d8SChris Lattner m_input_file (), 22530fdc8d8SChris Lattner m_output_file (), 22630fdc8d8SChris Lattner m_error_file (), 22730fdc8d8SChris Lattner m_target_list (), 228ded470d3SGreg Clayton m_platform_list (), 22930fdc8d8SChris Lattner m_listener ("lldb.Debugger"), 230e37d605eSJim Ingham m_source_manager(*this), 231e37d605eSJim Ingham m_source_file_cache(), 2326611103cSGreg Clayton m_command_interpreter_ap (new CommandInterpreter (*this, eScriptLanguageDefault, false)), 233d5a0a01bSCaroline Tice m_input_reader_stack (), 2344957bf69SGreg Clayton m_input_reader_data () 23530fdc8d8SChris Lattner { 2366611103cSGreg Clayton m_command_interpreter_ap->Initialize (); 237ded470d3SGreg Clayton // Always add our default platform to the platform list 238ded470d3SGreg Clayton PlatformSP default_platform_sp (Platform::GetDefaultPlatform()); 239ded470d3SGreg Clayton assert (default_platform_sp.get()); 240ded470d3SGreg Clayton m_platform_list.Append (default_platform_sp, true); 24130fdc8d8SChris Lattner } 24230fdc8d8SChris Lattner 24330fdc8d8SChris Lattner Debugger::~Debugger () 24430fdc8d8SChris Lattner { 2458314c525SJim Ingham Clear(); 2468314c525SJim Ingham } 2478314c525SJim Ingham 2488314c525SJim Ingham void 2498314c525SJim Ingham Debugger::Clear() 2508314c525SJim Ingham { 2513d6086f6SCaroline Tice CleanUpInputReaders(); 2521ed54f50SGreg Clayton m_listener.Clear(); 2536611103cSGreg Clayton int num_targets = m_target_list.GetNumTargets(); 2546611103cSGreg Clayton for (int i = 0; i < num_targets; i++) 2556611103cSGreg Clayton { 2566611103cSGreg Clayton ProcessSP process_sp (m_target_list.GetTargetAtIndex (i)->GetProcessSP()); 2576611103cSGreg Clayton if (process_sp) 2588314c525SJim Ingham { 2598314c525SJim Ingham if (process_sp->AttachedToProcess()) 2608314c525SJim Ingham process_sp->Detach(); 2618314c525SJim Ingham else 2626611103cSGreg Clayton process_sp->Destroy(); 2636611103cSGreg Clayton } 26430fdc8d8SChris Lattner } 2658314c525SJim Ingham DisconnectInput(); 26630fdc8d8SChris Lattner 2678314c525SJim Ingham } 26830fdc8d8SChris Lattner 26930fdc8d8SChris Lattner bool 270fc3f027dSGreg Clayton Debugger::GetCloseInputOnEOF () const 271fc3f027dSGreg Clayton { 272fc3f027dSGreg Clayton return m_input_comm.GetCloseOnEOF(); 273fc3f027dSGreg Clayton } 274fc3f027dSGreg Clayton 275fc3f027dSGreg Clayton void 276fc3f027dSGreg Clayton Debugger::SetCloseInputOnEOF (bool b) 277fc3f027dSGreg Clayton { 278fc3f027dSGreg Clayton m_input_comm.SetCloseOnEOF(b); 279fc3f027dSGreg Clayton } 280fc3f027dSGreg Clayton 281fc3f027dSGreg Clayton bool 28230fdc8d8SChris Lattner Debugger::GetAsyncExecution () 28330fdc8d8SChris Lattner { 2846611103cSGreg Clayton return !m_command_interpreter_ap->GetSynchronous(); 28530fdc8d8SChris Lattner } 28630fdc8d8SChris Lattner 28730fdc8d8SChris Lattner void 28830fdc8d8SChris Lattner Debugger::SetAsyncExecution (bool async_execution) 28930fdc8d8SChris Lattner { 2906611103cSGreg Clayton m_command_interpreter_ap->SetSynchronous (!async_execution); 29130fdc8d8SChris Lattner } 29230fdc8d8SChris Lattner 29330fdc8d8SChris Lattner 29430fdc8d8SChris Lattner void 29530fdc8d8SChris Lattner Debugger::SetInputFileHandle (FILE *fh, bool tranfer_ownership) 29630fdc8d8SChris Lattner { 29751b1e2d2SGreg Clayton File &in_file = GetInputFile(); 29851b1e2d2SGreg Clayton in_file.SetStream (fh, tranfer_ownership); 29951b1e2d2SGreg Clayton if (in_file.IsValid() == false) 30051b1e2d2SGreg Clayton in_file.SetStream (stdin, true); 30130fdc8d8SChris Lattner 30230fdc8d8SChris Lattner // Disconnect from any old connection if we had one 30330fdc8d8SChris Lattner m_input_comm.Disconnect (); 30451b1e2d2SGreg Clayton m_input_comm.SetConnection (new ConnectionFileDescriptor (in_file.GetDescriptor(), true)); 30530fdc8d8SChris Lattner m_input_comm.SetReadThreadBytesReceivedCallback (Debugger::DispatchInputCallback, this); 30630fdc8d8SChris Lattner 30730fdc8d8SChris Lattner Error error; 30830fdc8d8SChris Lattner if (m_input_comm.StartReadThread (&error) == false) 30930fdc8d8SChris Lattner { 31051b1e2d2SGreg Clayton File &err_file = GetErrorFile(); 31151b1e2d2SGreg Clayton 31251b1e2d2SGreg Clayton err_file.Printf ("error: failed to main input read thread: %s", error.AsCString() ? error.AsCString() : "unkown error"); 31330fdc8d8SChris Lattner exit(1); 31430fdc8d8SChris Lattner } 31530fdc8d8SChris Lattner } 31630fdc8d8SChris Lattner 31730fdc8d8SChris Lattner void 31830fdc8d8SChris Lattner Debugger::SetOutputFileHandle (FILE *fh, bool tranfer_ownership) 31930fdc8d8SChris Lattner { 32051b1e2d2SGreg Clayton File &out_file = GetOutputFile(); 32151b1e2d2SGreg Clayton out_file.SetStream (fh, tranfer_ownership); 32251b1e2d2SGreg Clayton if (out_file.IsValid() == false) 32351b1e2d2SGreg Clayton out_file.SetStream (stdout, false); 3242f88aadfSCaroline Tice 3252f88aadfSCaroline Tice GetCommandInterpreter().GetScriptInterpreter()->ResetOutputFileHandle (fh); 32630fdc8d8SChris Lattner } 32730fdc8d8SChris Lattner 32830fdc8d8SChris Lattner void 32930fdc8d8SChris Lattner Debugger::SetErrorFileHandle (FILE *fh, bool tranfer_ownership) 33030fdc8d8SChris Lattner { 33151b1e2d2SGreg Clayton File &err_file = GetErrorFile(); 33251b1e2d2SGreg Clayton err_file.SetStream (fh, tranfer_ownership); 33351b1e2d2SGreg Clayton if (err_file.IsValid() == false) 33451b1e2d2SGreg Clayton err_file.SetStream (stderr, false); 33530fdc8d8SChris Lattner } 33630fdc8d8SChris Lattner 33730fdc8d8SChris Lattner ExecutionContext 3382976d00aSJim Ingham Debugger::GetSelectedExecutionContext () 33930fdc8d8SChris Lattner { 34030fdc8d8SChris Lattner ExecutionContext exe_ctx; 341c14ee32dSGreg Clayton TargetSP target_sp(GetSelectedTarget()); 342c14ee32dSGreg Clayton exe_ctx.SetTargetSP (target_sp); 34330fdc8d8SChris Lattner 34430fdc8d8SChris Lattner if (target_sp) 34530fdc8d8SChris Lattner { 346c14ee32dSGreg Clayton ProcessSP process_sp (target_sp->GetProcessSP()); 347c14ee32dSGreg Clayton exe_ctx.SetProcessSP (process_sp); 348c14ee32dSGreg Clayton if (process_sp && process_sp->IsRunning() == false) 34930fdc8d8SChris Lattner { 350c14ee32dSGreg Clayton ThreadSP thread_sp (process_sp->GetThreadList().GetSelectedThread()); 351c14ee32dSGreg Clayton if (thread_sp) 35230fdc8d8SChris Lattner { 353c14ee32dSGreg Clayton exe_ctx.SetThreadSP (thread_sp); 354c14ee32dSGreg Clayton exe_ctx.SetFrameSP (thread_sp->GetSelectedFrame()); 355c14ee32dSGreg Clayton if (exe_ctx.GetFramePtr() == NULL) 356c14ee32dSGreg Clayton exe_ctx.SetFrameSP (thread_sp->GetStackFrameAtIndex (0)); 35730fdc8d8SChris Lattner } 35830fdc8d8SChris Lattner } 35930fdc8d8SChris Lattner } 36030fdc8d8SChris Lattner return exe_ctx; 36130fdc8d8SChris Lattner 36230fdc8d8SChris Lattner } 36330fdc8d8SChris Lattner 364b44880caSCaroline Tice InputReaderSP 365b44880caSCaroline Tice Debugger::GetCurrentInputReader () 366b44880caSCaroline Tice { 367b44880caSCaroline Tice InputReaderSP reader_sp; 368b44880caSCaroline Tice 369d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 370b44880caSCaroline Tice { 371b44880caSCaroline Tice // Clear any finished readers from the stack 372b44880caSCaroline Tice while (CheckIfTopInputReaderIsDone()) ; 373b44880caSCaroline Tice 374d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 375d5a0a01bSCaroline Tice reader_sp = m_input_reader_stack.Top(); 376b44880caSCaroline Tice } 377b44880caSCaroline Tice 378b44880caSCaroline Tice return reader_sp; 379b44880caSCaroline Tice } 380b44880caSCaroline Tice 38130fdc8d8SChris Lattner void 38230fdc8d8SChris Lattner Debugger::DispatchInputCallback (void *baton, const void *bytes, size_t bytes_len) 38330fdc8d8SChris Lattner { 384efed6131SCaroline Tice if (bytes_len > 0) 38530fdc8d8SChris Lattner ((Debugger *)baton)->DispatchInput ((char *)bytes, bytes_len); 386efed6131SCaroline Tice else 387efed6131SCaroline Tice ((Debugger *)baton)->DispatchInputEndOfFile (); 38830fdc8d8SChris Lattner } 38930fdc8d8SChris Lattner 39030fdc8d8SChris Lattner 39130fdc8d8SChris Lattner void 39230fdc8d8SChris Lattner Debugger::DispatchInput (const char *bytes, size_t bytes_len) 39330fdc8d8SChris Lattner { 394efed6131SCaroline Tice if (bytes == NULL || bytes_len == 0) 395efed6131SCaroline Tice return; 39630fdc8d8SChris Lattner 39730fdc8d8SChris Lattner WriteToDefaultReader (bytes, bytes_len); 39830fdc8d8SChris Lattner } 39930fdc8d8SChris Lattner 40030fdc8d8SChris Lattner void 401efed6131SCaroline Tice Debugger::DispatchInputInterrupt () 402efed6131SCaroline Tice { 403efed6131SCaroline Tice m_input_reader_data.clear(); 404efed6131SCaroline Tice 405b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 406efed6131SCaroline Tice if (reader_sp) 407b44880caSCaroline Tice { 408efed6131SCaroline Tice reader_sp->Notify (eInputReaderInterrupt); 409efed6131SCaroline Tice 410b44880caSCaroline Tice // If notifying the reader of the interrupt finished the reader, we should pop it off the stack. 411efed6131SCaroline Tice while (CheckIfTopInputReaderIsDone ()) ; 412efed6131SCaroline Tice } 413efed6131SCaroline Tice } 414efed6131SCaroline Tice 415efed6131SCaroline Tice void 416efed6131SCaroline Tice Debugger::DispatchInputEndOfFile () 417efed6131SCaroline Tice { 418efed6131SCaroline Tice m_input_reader_data.clear(); 419efed6131SCaroline Tice 420b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 421efed6131SCaroline Tice if (reader_sp) 422b44880caSCaroline Tice { 423efed6131SCaroline Tice reader_sp->Notify (eInputReaderEndOfFile); 424efed6131SCaroline Tice 425b44880caSCaroline Tice // If notifying the reader of the end-of-file finished the reader, we should pop it off the stack. 426efed6131SCaroline Tice while (CheckIfTopInputReaderIsDone ()) ; 427efed6131SCaroline Tice } 428efed6131SCaroline Tice } 429efed6131SCaroline Tice 430efed6131SCaroline Tice void 4313d6086f6SCaroline Tice Debugger::CleanUpInputReaders () 4323d6086f6SCaroline Tice { 4333d6086f6SCaroline Tice m_input_reader_data.clear(); 4343d6086f6SCaroline Tice 435b44880caSCaroline Tice // The bottom input reader should be the main debugger input reader. We do not want to close that one here. 436d5a0a01bSCaroline Tice while (m_input_reader_stack.GetSize() > 1) 4373d6086f6SCaroline Tice { 438b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 4393d6086f6SCaroline Tice if (reader_sp) 4403d6086f6SCaroline Tice { 4413d6086f6SCaroline Tice reader_sp->Notify (eInputReaderEndOfFile); 4423d6086f6SCaroline Tice reader_sp->SetIsDone (true); 4433d6086f6SCaroline Tice } 4443d6086f6SCaroline Tice } 4453d6086f6SCaroline Tice } 4463d6086f6SCaroline Tice 4473d6086f6SCaroline Tice void 448969ed3d1SCaroline Tice Debugger::NotifyTopInputReader (InputReaderAction notification) 449969ed3d1SCaroline Tice { 450969ed3d1SCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader()); 451969ed3d1SCaroline Tice if (reader_sp) 452969ed3d1SCaroline Tice { 453969ed3d1SCaroline Tice reader_sp->Notify (notification); 454969ed3d1SCaroline Tice 455969ed3d1SCaroline Tice // Flush out any input readers that are done. 456969ed3d1SCaroline Tice while (CheckIfTopInputReaderIsDone ()) 457969ed3d1SCaroline Tice /* Do nothing. */; 458969ed3d1SCaroline Tice } 459969ed3d1SCaroline Tice } 460969ed3d1SCaroline Tice 4619088b068SCaroline Tice bool 4624d122c40SGreg Clayton Debugger::InputReaderIsTopReader (const InputReaderSP& reader_sp) 4639088b068SCaroline Tice { 4649088b068SCaroline Tice InputReaderSP top_reader_sp (GetCurrentInputReader()); 4659088b068SCaroline Tice 466d61c10bcSCaroline Tice return (reader_sp.get() == top_reader_sp.get()); 4679088b068SCaroline Tice } 4689088b068SCaroline Tice 4699088b068SCaroline Tice 470969ed3d1SCaroline Tice void 47130fdc8d8SChris Lattner Debugger::WriteToDefaultReader (const char *bytes, size_t bytes_len) 47230fdc8d8SChris Lattner { 47330fdc8d8SChris Lattner if (bytes && bytes_len) 47430fdc8d8SChris Lattner m_input_reader_data.append (bytes, bytes_len); 47530fdc8d8SChris Lattner 47630fdc8d8SChris Lattner if (m_input_reader_data.empty()) 47730fdc8d8SChris Lattner return; 47830fdc8d8SChris Lattner 479d5a0a01bSCaroline Tice while (!m_input_reader_stack.IsEmpty() && !m_input_reader_data.empty()) 48030fdc8d8SChris Lattner { 48130fdc8d8SChris Lattner // Get the input reader from the top of the stack 482b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 48330fdc8d8SChris Lattner if (!reader_sp) 48430fdc8d8SChris Lattner break; 48530fdc8d8SChris Lattner 486471b31ceSGreg Clayton size_t bytes_handled = reader_sp->HandleRawBytes (m_input_reader_data.c_str(), 48730fdc8d8SChris Lattner m_input_reader_data.size()); 48830fdc8d8SChris Lattner if (bytes_handled) 48930fdc8d8SChris Lattner { 49030fdc8d8SChris Lattner m_input_reader_data.erase (0, bytes_handled); 49130fdc8d8SChris Lattner } 49230fdc8d8SChris Lattner else 49330fdc8d8SChris Lattner { 49430fdc8d8SChris Lattner // No bytes were handled, we might not have reached our 49530fdc8d8SChris Lattner // granularity, just return and wait for more data 49630fdc8d8SChris Lattner break; 49730fdc8d8SChris Lattner } 49830fdc8d8SChris Lattner } 49930fdc8d8SChris Lattner 500b44880caSCaroline Tice // Flush out any input readers that are done. 50130fdc8d8SChris Lattner while (CheckIfTopInputReaderIsDone ()) 50230fdc8d8SChris Lattner /* Do nothing. */; 50330fdc8d8SChris Lattner 50430fdc8d8SChris Lattner } 50530fdc8d8SChris Lattner 50630fdc8d8SChris Lattner void 50730fdc8d8SChris Lattner Debugger::PushInputReader (const InputReaderSP& reader_sp) 50830fdc8d8SChris Lattner { 50930fdc8d8SChris Lattner if (!reader_sp) 51030fdc8d8SChris Lattner return; 511b44880caSCaroline Tice 51230fdc8d8SChris Lattner // Deactivate the old top reader 513b44880caSCaroline Tice InputReaderSP top_reader_sp (GetCurrentInputReader ()); 514b44880caSCaroline Tice 51530fdc8d8SChris Lattner if (top_reader_sp) 51630fdc8d8SChris Lattner top_reader_sp->Notify (eInputReaderDeactivate); 517b44880caSCaroline Tice 518d5a0a01bSCaroline Tice m_input_reader_stack.Push (reader_sp); 51930fdc8d8SChris Lattner reader_sp->Notify (eInputReaderActivate); 52030fdc8d8SChris Lattner ActivateInputReader (reader_sp); 52130fdc8d8SChris Lattner } 52230fdc8d8SChris Lattner 52330fdc8d8SChris Lattner bool 5244d122c40SGreg Clayton Debugger::PopInputReader (const InputReaderSP& pop_reader_sp) 52530fdc8d8SChris Lattner { 52630fdc8d8SChris Lattner bool result = false; 52730fdc8d8SChris Lattner 52830fdc8d8SChris Lattner // The reader on the stop of the stack is done, so let the next 52930fdc8d8SChris Lattner // read on the stack referesh its prompt and if there is one... 530d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 53130fdc8d8SChris Lattner { 532b44880caSCaroline Tice // Cannot call GetCurrentInputReader here, as that would cause an infinite loop. 533d5a0a01bSCaroline Tice InputReaderSP reader_sp(m_input_reader_stack.Top()); 53430fdc8d8SChris Lattner 53530fdc8d8SChris Lattner if (!pop_reader_sp || pop_reader_sp.get() == reader_sp.get()) 53630fdc8d8SChris Lattner { 537d5a0a01bSCaroline Tice m_input_reader_stack.Pop (); 53830fdc8d8SChris Lattner reader_sp->Notify (eInputReaderDeactivate); 53930fdc8d8SChris Lattner reader_sp->Notify (eInputReaderDone); 54030fdc8d8SChris Lattner result = true; 54130fdc8d8SChris Lattner 542d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 54330fdc8d8SChris Lattner { 544d5a0a01bSCaroline Tice reader_sp = m_input_reader_stack.Top(); 54530fdc8d8SChris Lattner if (reader_sp) 54630fdc8d8SChris Lattner { 54730fdc8d8SChris Lattner ActivateInputReader (reader_sp); 54830fdc8d8SChris Lattner reader_sp->Notify (eInputReaderReactivate); 54930fdc8d8SChris Lattner } 55030fdc8d8SChris Lattner } 55130fdc8d8SChris Lattner } 55230fdc8d8SChris Lattner } 55330fdc8d8SChris Lattner return result; 55430fdc8d8SChris Lattner } 55530fdc8d8SChris Lattner 55630fdc8d8SChris Lattner bool 55730fdc8d8SChris Lattner Debugger::CheckIfTopInputReaderIsDone () 55830fdc8d8SChris Lattner { 55930fdc8d8SChris Lattner bool result = false; 560d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 56130fdc8d8SChris Lattner { 562b44880caSCaroline Tice // Cannot call GetCurrentInputReader here, as that would cause an infinite loop. 563d5a0a01bSCaroline Tice InputReaderSP reader_sp(m_input_reader_stack.Top()); 56430fdc8d8SChris Lattner 56530fdc8d8SChris Lattner if (reader_sp && reader_sp->IsDone()) 56630fdc8d8SChris Lattner { 56730fdc8d8SChris Lattner result = true; 56830fdc8d8SChris Lattner PopInputReader (reader_sp); 56930fdc8d8SChris Lattner } 57030fdc8d8SChris Lattner } 57130fdc8d8SChris Lattner return result; 57230fdc8d8SChris Lattner } 57330fdc8d8SChris Lattner 57430fdc8d8SChris Lattner void 57530fdc8d8SChris Lattner Debugger::ActivateInputReader (const InputReaderSP &reader_sp) 57630fdc8d8SChris Lattner { 57751b1e2d2SGreg Clayton int input_fd = m_input_file.GetFile().GetDescriptor(); 57830fdc8d8SChris Lattner 57951b1e2d2SGreg Clayton if (input_fd >= 0) 58030fdc8d8SChris Lattner { 58151b1e2d2SGreg Clayton Terminal tty(input_fd); 582a3406614SGreg Clayton 583a3406614SGreg Clayton tty.SetEcho(reader_sp->GetEcho()); 58430fdc8d8SChris Lattner 58530fdc8d8SChris Lattner switch (reader_sp->GetGranularity()) 58630fdc8d8SChris Lattner { 58730fdc8d8SChris Lattner case eInputReaderGranularityByte: 58830fdc8d8SChris Lattner case eInputReaderGranularityWord: 589a3406614SGreg Clayton tty.SetCanonical (false); 59030fdc8d8SChris Lattner break; 59130fdc8d8SChris Lattner 59230fdc8d8SChris Lattner case eInputReaderGranularityLine: 59330fdc8d8SChris Lattner case eInputReaderGranularityAll: 594a3406614SGreg Clayton tty.SetCanonical (true); 59530fdc8d8SChris Lattner break; 59630fdc8d8SChris Lattner 59730fdc8d8SChris Lattner default: 59830fdc8d8SChris Lattner break; 59930fdc8d8SChris Lattner } 60030fdc8d8SChris Lattner } 60130fdc8d8SChris Lattner } 6026611103cSGreg Clayton 6035b52f0c7SJim Ingham StreamSP 6045b52f0c7SJim Ingham Debugger::GetAsyncOutputStream () 6055b52f0c7SJim Ingham { 6065b52f0c7SJim Ingham return StreamSP (new StreamAsynchronousIO (GetCommandInterpreter(), 6075b52f0c7SJim Ingham CommandInterpreter::eBroadcastBitAsynchronousOutputData)); 6085b52f0c7SJim Ingham } 6095b52f0c7SJim Ingham 6105b52f0c7SJim Ingham StreamSP 6115b52f0c7SJim Ingham Debugger::GetAsyncErrorStream () 6125b52f0c7SJim Ingham { 6135b52f0c7SJim Ingham return StreamSP (new StreamAsynchronousIO (GetCommandInterpreter(), 6145b52f0c7SJim Ingham CommandInterpreter::eBroadcastBitAsynchronousErrorData)); 6155b52f0c7SJim Ingham } 6165b52f0c7SJim Ingham 617ebc1bb27SCaroline Tice DebuggerSP 618ebc1bb27SCaroline Tice Debugger::FindDebuggerWithID (lldb::user_id_t id) 619ebc1bb27SCaroline Tice { 6204d122c40SGreg Clayton DebuggerSP debugger_sp; 621ebc1bb27SCaroline Tice 622ebc1bb27SCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 623ebc1bb27SCaroline Tice DebuggerList &debugger_list = GetDebuggerList(); 624ebc1bb27SCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 625ebc1bb27SCaroline Tice for (pos = debugger_list.begin(); pos != end; ++pos) 626ebc1bb27SCaroline Tice { 627ebc1bb27SCaroline Tice if ((*pos).get()->GetID() == id) 628ebc1bb27SCaroline Tice { 629ebc1bb27SCaroline Tice debugger_sp = *pos; 630ebc1bb27SCaroline Tice break; 631ebc1bb27SCaroline Tice } 632ebc1bb27SCaroline Tice } 633ebc1bb27SCaroline Tice return debugger_sp; 634ebc1bb27SCaroline Tice } 6353df9a8dfSCaroline Tice 6361b654882SGreg Clayton static void 6371b654882SGreg Clayton TestPromptFormats (StackFrame *frame) 6381b654882SGreg Clayton { 6391b654882SGreg Clayton if (frame == NULL) 6401b654882SGreg Clayton return; 6411b654882SGreg Clayton 6421b654882SGreg Clayton StreamString s; 6431b654882SGreg Clayton const char *prompt_format = 6441b654882SGreg Clayton "{addr = '${addr}'\n}" 6451b654882SGreg Clayton "{process.id = '${process.id}'\n}" 6461b654882SGreg Clayton "{process.name = '${process.name}'\n}" 6471b654882SGreg Clayton "{process.file.basename = '${process.file.basename}'\n}" 6481b654882SGreg Clayton "{process.file.fullpath = '${process.file.fullpath}'\n}" 6491b654882SGreg Clayton "{thread.id = '${thread.id}'\n}" 6501b654882SGreg Clayton "{thread.index = '${thread.index}'\n}" 6511b654882SGreg Clayton "{thread.name = '${thread.name}'\n}" 6521b654882SGreg Clayton "{thread.queue = '${thread.queue}'\n}" 6531b654882SGreg Clayton "{thread.stop-reason = '${thread.stop-reason}'\n}" 6541b654882SGreg Clayton "{target.arch = '${target.arch}'\n}" 6551b654882SGreg Clayton "{module.file.basename = '${module.file.basename}'\n}" 6561b654882SGreg Clayton "{module.file.fullpath = '${module.file.fullpath}'\n}" 6571b654882SGreg Clayton "{file.basename = '${file.basename}'\n}" 6581b654882SGreg Clayton "{file.fullpath = '${file.fullpath}'\n}" 6591b654882SGreg Clayton "{frame.index = '${frame.index}'\n}" 6601b654882SGreg Clayton "{frame.pc = '${frame.pc}'\n}" 6611b654882SGreg Clayton "{frame.sp = '${frame.sp}'\n}" 6621b654882SGreg Clayton "{frame.fp = '${frame.fp}'\n}" 6631b654882SGreg Clayton "{frame.flags = '${frame.flags}'\n}" 6641b654882SGreg Clayton "{frame.reg.rdi = '${frame.reg.rdi}'\n}" 6651b654882SGreg Clayton "{frame.reg.rip = '${frame.reg.rip}'\n}" 6661b654882SGreg Clayton "{frame.reg.rsp = '${frame.reg.rsp}'\n}" 6671b654882SGreg Clayton "{frame.reg.rbp = '${frame.reg.rbp}'\n}" 6681b654882SGreg Clayton "{frame.reg.rflags = '${frame.reg.rflags}'\n}" 6691b654882SGreg Clayton "{frame.reg.xmm0 = '${frame.reg.xmm0}'\n}" 6701b654882SGreg Clayton "{frame.reg.carp = '${frame.reg.carp}'\n}" 6711b654882SGreg Clayton "{function.id = '${function.id}'\n}" 6721b654882SGreg Clayton "{function.name = '${function.name}'\n}" 6731b654882SGreg Clayton "{function.addr-offset = '${function.addr-offset}'\n}" 6741b654882SGreg Clayton "{function.line-offset = '${function.line-offset}'\n}" 6751b654882SGreg Clayton "{function.pc-offset = '${function.pc-offset}'\n}" 6761b654882SGreg Clayton "{line.file.basename = '${line.file.basename}'\n}" 6771b654882SGreg Clayton "{line.file.fullpath = '${line.file.fullpath}'\n}" 6781b654882SGreg Clayton "{line.number = '${line.number}'\n}" 6791b654882SGreg Clayton "{line.start-addr = '${line.start-addr}'\n}" 6801b654882SGreg Clayton "{line.end-addr = '${line.end-addr}'\n}" 6811b654882SGreg Clayton ; 6821b654882SGreg Clayton 6831b654882SGreg Clayton SymbolContext sc (frame->GetSymbolContext(eSymbolContextEverything)); 6841b654882SGreg Clayton ExecutionContext exe_ctx; 6850603aa9dSGreg Clayton frame->CalculateExecutionContext(exe_ctx); 6861b654882SGreg Clayton const char *end = NULL; 6871b654882SGreg Clayton if (Debugger::FormatPrompt (prompt_format, &sc, &exe_ctx, &sc.line_entry.range.GetBaseAddress(), s, &end)) 6881b654882SGreg Clayton { 6891b654882SGreg Clayton printf("%s\n", s.GetData()); 6901b654882SGreg Clayton } 6911b654882SGreg Clayton else 6921b654882SGreg Clayton { 6931b654882SGreg Clayton printf ("error: at '%s'\n", end); 6941b654882SGreg Clayton printf ("what we got: %s\n", s.GetData()); 6951b654882SGreg Clayton } 6961b654882SGreg Clayton } 6971b654882SGreg Clayton 6989fc1944eSEnrico Granata static bool 6999fc1944eSEnrico Granata ScanFormatDescriptor (const char* var_name_begin, 7009fc1944eSEnrico Granata const char* var_name_end, 7019fc1944eSEnrico Granata const char** var_name_final, 7029fc1944eSEnrico Granata const char** percent_position, 7034d122c40SGreg Clayton Format* custom_format, 7049fc1944eSEnrico Granata ValueObject::ValueObjectRepresentationStyle* val_obj_display) 7059fc1944eSEnrico Granata { 706e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 7079fc1944eSEnrico Granata *percent_position = ::strchr(var_name_begin,'%'); 7089fc1944eSEnrico Granata if (!*percent_position || *percent_position > var_name_end) 709e992a089SEnrico Granata { 710e992a089SEnrico Granata if (log) 711e992a089SEnrico Granata log->Printf("no format descriptor in string, skipping"); 7129fc1944eSEnrico Granata *var_name_final = var_name_end; 713e992a089SEnrico Granata } 7149fc1944eSEnrico Granata else 7159fc1944eSEnrico Granata { 7169fc1944eSEnrico Granata *var_name_final = *percent_position; 7179fc1944eSEnrico Granata char* format_name = new char[var_name_end-*var_name_final]; format_name[var_name_end-*var_name_final-1] = '\0'; 7189fc1944eSEnrico Granata memcpy(format_name, *var_name_final+1, var_name_end-*var_name_final-1); 719e992a089SEnrico Granata if (log) 720e992a089SEnrico Granata log->Printf("parsing %s as a format descriptor", format_name); 7219fc1944eSEnrico Granata if ( !FormatManager::GetFormatFromCString(format_name, 7229fc1944eSEnrico Granata true, 7239fc1944eSEnrico Granata *custom_format) ) 7249fc1944eSEnrico Granata { 725e992a089SEnrico Granata if (log) 726e992a089SEnrico Granata log->Printf("%s is an unknown format", format_name); 7279fc1944eSEnrico Granata // if this is an @ sign, print ObjC description 7289fc1944eSEnrico Granata if (*format_name == '@') 7299fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayLanguageSpecific; 7309fc1944eSEnrico Granata // if this is a V, print the value using the default format 731e992a089SEnrico Granata else if (*format_name == 'V') 7329fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayValue; 733d55546b2SEnrico Granata // if this is an L, print the location of the value 734e992a089SEnrico Granata else if (*format_name == 'L') 735f2bbf717SEnrico Granata *val_obj_display = ValueObject::eDisplayLocation; 736d55546b2SEnrico Granata // if this is an S, print the summary after all 737e992a089SEnrico Granata else if (*format_name == 'S') 738d55546b2SEnrico Granata *val_obj_display = ValueObject::eDisplaySummary; 7395dfd49ccSEnrico Granata else if (*format_name == '#') 7405dfd49ccSEnrico Granata *val_obj_display = ValueObject::eDisplayChildrenCount; 741d64d0bc0SEnrico Granata else if (*format_name == 'T') 742d64d0bc0SEnrico Granata *val_obj_display = ValueObject::eDisplayType; 743e992a089SEnrico Granata else if (log) 744e992a089SEnrico Granata log->Printf("%s is an error, leaving the previous value alone", format_name); 7459fc1944eSEnrico Granata } 7469fc1944eSEnrico Granata // a good custom format tells us to print the value using it 7479fc1944eSEnrico Granata else 748e992a089SEnrico Granata { 749e992a089SEnrico Granata if (log) 750e992a089SEnrico Granata log->Printf("will display value for this VO"); 7519fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayValue; 752e992a089SEnrico Granata } 7539fc1944eSEnrico Granata delete format_name; 7549fc1944eSEnrico Granata } 755e992a089SEnrico Granata if (log) 756e992a089SEnrico Granata log->Printf("final format description outcome: custom_format = %d, val_obj_display = %d", 757e992a089SEnrico Granata *custom_format, 758e992a089SEnrico Granata *val_obj_display); 7599fc1944eSEnrico Granata return true; 7609fc1944eSEnrico Granata } 7619fc1944eSEnrico Granata 7629fc1944eSEnrico Granata static bool 7639fc1944eSEnrico Granata ScanBracketedRange (const char* var_name_begin, 7649fc1944eSEnrico Granata const char* var_name_end, 7659fc1944eSEnrico Granata const char* var_name_final, 7669fc1944eSEnrico Granata const char** open_bracket_position, 7679fc1944eSEnrico Granata const char** separator_position, 7689fc1944eSEnrico Granata const char** close_bracket_position, 7699fc1944eSEnrico Granata const char** var_name_final_if_array_range, 7709fc1944eSEnrico Granata int64_t* index_lower, 7719fc1944eSEnrico Granata int64_t* index_higher) 7729fc1944eSEnrico Granata { 773e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 7749fc1944eSEnrico Granata *open_bracket_position = ::strchr(var_name_begin,'['); 7759fc1944eSEnrico Granata if (*open_bracket_position && *open_bracket_position < var_name_final) 7769fc1944eSEnrico Granata { 7779fc1944eSEnrico Granata *separator_position = ::strchr(*open_bracket_position,'-'); // might be NULL if this is a simple var[N] bitfield 7789fc1944eSEnrico Granata *close_bracket_position = ::strchr(*open_bracket_position,']'); 7799fc1944eSEnrico Granata // as usual, we assume that [] will come before % 7809fc1944eSEnrico Granata //printf("trying to expand a []\n"); 7819fc1944eSEnrico Granata *var_name_final_if_array_range = *open_bracket_position; 7829fc1944eSEnrico Granata if (*close_bracket_position - *open_bracket_position == 1) 7839fc1944eSEnrico Granata { 784e992a089SEnrico Granata if (log) 785e992a089SEnrico Granata log->Printf("[] detected.. going from 0 to end of data"); 7869fc1944eSEnrico Granata *index_lower = 0; 7879fc1944eSEnrico Granata } 7889fc1944eSEnrico Granata else if (*separator_position == NULL || *separator_position > var_name_end) 7899fc1944eSEnrico Granata { 7909fc1944eSEnrico Granata char *end = NULL; 7919fc1944eSEnrico Granata *index_lower = ::strtoul (*open_bracket_position+1, &end, 0); 7929fc1944eSEnrico Granata *index_higher = *index_lower; 793e992a089SEnrico Granata if (log) 794fd54b368SJason Molenda log->Printf("[%lld] detected, high index is same", *index_lower); 7959fc1944eSEnrico Granata } 7969fc1944eSEnrico Granata else if (*close_bracket_position && *close_bracket_position < var_name_end) 7979fc1944eSEnrico Granata { 7989fc1944eSEnrico Granata char *end = NULL; 7999fc1944eSEnrico Granata *index_lower = ::strtoul (*open_bracket_position+1, &end, 0); 8009fc1944eSEnrico Granata *index_higher = ::strtoul (*separator_position+1, &end, 0); 801e992a089SEnrico Granata if (log) 802fd54b368SJason Molenda log->Printf("[%lld-%lld] detected", *index_lower, *index_higher); 8039fc1944eSEnrico Granata } 8049fc1944eSEnrico Granata else 805e992a089SEnrico Granata { 806e992a089SEnrico Granata if (log) 807e992a089SEnrico Granata log->Printf("expression is erroneous, cannot extract indices out of it"); 8089fc1944eSEnrico Granata return false; 809e992a089SEnrico Granata } 8109fc1944eSEnrico Granata if (*index_lower > *index_higher && *index_higher > 0) 8119fc1944eSEnrico Granata { 812e992a089SEnrico Granata if (log) 813e992a089SEnrico Granata log->Printf("swapping indices"); 8149fc1944eSEnrico Granata int temp = *index_lower; 8159fc1944eSEnrico Granata *index_lower = *index_higher; 8169fc1944eSEnrico Granata *index_higher = temp; 8179fc1944eSEnrico Granata } 8189fc1944eSEnrico Granata } 819e992a089SEnrico Granata else if (log) 820e992a089SEnrico Granata log->Printf("no bracketed range, skipping entirely"); 8219fc1944eSEnrico Granata return true; 8229fc1944eSEnrico Granata } 8239fc1944eSEnrico Granata 8249fc1944eSEnrico Granata 8259fc1944eSEnrico Granata static ValueObjectSP 826c482a192SEnrico Granata ExpandExpressionPath (ValueObject* valobj, 8279fc1944eSEnrico Granata StackFrame* frame, 8289fc1944eSEnrico Granata bool* do_deref_pointer, 8299fc1944eSEnrico Granata const char* var_name_begin, 8309fc1944eSEnrico Granata const char* var_name_final, 8319fc1944eSEnrico Granata Error& error) 8329fc1944eSEnrico Granata { 833e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 8349fc1944eSEnrico Granata StreamString sstring; 8359fc1944eSEnrico Granata VariableSP var_sp; 8369fc1944eSEnrico Granata 8379fc1944eSEnrico Granata if (*do_deref_pointer) 838e992a089SEnrico Granata { 839e992a089SEnrico Granata if (log) 840e992a089SEnrico Granata log->Printf("been told to deref_pointer by caller"); 8419fc1944eSEnrico Granata sstring.PutChar('*'); 842e992a089SEnrico Granata } 843c482a192SEnrico Granata else if (valobj->IsDereferenceOfParent() && ClangASTContext::IsPointerType(valobj->GetParent()->GetClangType()) && !valobj->IsArrayItemForPointer()) 8449fc1944eSEnrico Granata { 845e992a089SEnrico Granata if (log) 846e992a089SEnrico Granata log->Printf("decided to deref_pointer myself"); 8479fc1944eSEnrico Granata sstring.PutChar('*'); 8489fc1944eSEnrico Granata *do_deref_pointer = true; 8499fc1944eSEnrico Granata } 8509fc1944eSEnrico Granata 851c482a192SEnrico Granata valobj->GetExpressionPath(sstring, true, ValueObject::eHonorPointers); 852e992a089SEnrico Granata if (log) 853e992a089SEnrico Granata log->Printf("expression path to expand in phase 0: %s",sstring.GetData()); 8549fc1944eSEnrico Granata sstring.PutRawBytes(var_name_begin+3, var_name_final-var_name_begin-3); 855e992a089SEnrico Granata if (log) 856e992a089SEnrico Granata log->Printf("expression path to expand in phase 1: %s",sstring.GetData()); 8579fc1944eSEnrico Granata std::string name = std::string(sstring.GetData()); 8589fc1944eSEnrico Granata ValueObjectSP target = frame->GetValueForVariableExpressionPath (name.c_str(), 8599fc1944eSEnrico Granata eNoDynamicValues, 8609fc1944eSEnrico Granata 0, 8619fc1944eSEnrico Granata var_sp, 8629fc1944eSEnrico Granata error); 8639fc1944eSEnrico Granata return target; 8649fc1944eSEnrico Granata } 8659fc1944eSEnrico Granata 8669fc1944eSEnrico Granata static ValueObjectSP 867c482a192SEnrico Granata ExpandIndexedExpression (ValueObject* valobj, 8689fc1944eSEnrico Granata uint32_t index, 8699fc1944eSEnrico Granata StackFrame* frame, 870fc7a7f3bSEnrico Granata bool deref_pointer) 8719fc1944eSEnrico Granata { 872e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 873fc7a7f3bSEnrico Granata const char* ptr_deref_format = "[%d]"; 874fc7a7f3bSEnrico Granata std::auto_ptr<char> ptr_deref_buffer(new char[10]); 875fc7a7f3bSEnrico Granata ::sprintf(ptr_deref_buffer.get(), ptr_deref_format, index); 876e992a089SEnrico Granata if (log) 877e992a089SEnrico Granata log->Printf("name to deref: %s",ptr_deref_buffer.get()); 878fc7a7f3bSEnrico Granata const char* first_unparsed; 879fc7a7f3bSEnrico Granata ValueObject::GetValueForExpressionPathOptions options; 880fc7a7f3bSEnrico Granata ValueObject::ExpressionPathEndResultType final_value_type; 881fc7a7f3bSEnrico Granata ValueObject::ExpressionPathScanEndReason reason_to_stop; 882fc7a7f3bSEnrico Granata ValueObject::ExpressionPathAftermath what_next = (deref_pointer ? ValueObject::eDereference : ValueObject::eNothing); 883c482a192SEnrico Granata ValueObjectSP item = valobj->GetValueForExpressionPath (ptr_deref_buffer.get(), 884fc7a7f3bSEnrico Granata &first_unparsed, 885fc7a7f3bSEnrico Granata &reason_to_stop, 886fc7a7f3bSEnrico Granata &final_value_type, 887fc7a7f3bSEnrico Granata options, 888fc7a7f3bSEnrico Granata &what_next); 889fc7a7f3bSEnrico Granata if (!item) 890fc7a7f3bSEnrico Granata { 891e992a089SEnrico Granata if (log) 892e992a089SEnrico Granata log->Printf("ERROR: unparsed portion = %s, why stopping = %d," 893e992a089SEnrico Granata " final_value_type %d", 894fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 895fc7a7f3bSEnrico Granata } 8969fc1944eSEnrico Granata else 8979fc1944eSEnrico Granata { 898e992a089SEnrico Granata if (log) 899e992a089SEnrico Granata log->Printf("ALL RIGHT: unparsed portion = %s, why stopping = %d," 900e992a089SEnrico Granata " final_value_type %d", 901fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 9029fc1944eSEnrico Granata } 9039fc1944eSEnrico Granata return item; 9049fc1944eSEnrico Granata } 9059fc1944eSEnrico Granata 9061b654882SGreg Clayton bool 9071b654882SGreg Clayton Debugger::FormatPrompt 9081b654882SGreg Clayton ( 9091b654882SGreg Clayton const char *format, 9101b654882SGreg Clayton const SymbolContext *sc, 9111b654882SGreg Clayton const ExecutionContext *exe_ctx, 9121b654882SGreg Clayton const Address *addr, 9131b654882SGreg Clayton Stream &s, 9144becb37eSEnrico Granata const char **end, 915c482a192SEnrico Granata ValueObject* valobj 9161b654882SGreg Clayton ) 9171b654882SGreg Clayton { 918c482a192SEnrico Granata ValueObject* realvalobj = NULL; // makes it super-easy to parse pointers 9191b654882SGreg Clayton bool success = true; 9201b654882SGreg Clayton const char *p; 921e992a089SEnrico Granata LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_TYPES)); 9221b654882SGreg Clayton for (p = format; *p != '\0'; ++p) 9231b654882SGreg Clayton { 924c482a192SEnrico Granata if (realvalobj) 9254becb37eSEnrico Granata { 926c482a192SEnrico Granata valobj = realvalobj; 927c482a192SEnrico Granata realvalobj = NULL; 9284becb37eSEnrico Granata } 9291b654882SGreg Clayton size_t non_special_chars = ::strcspn (p, "${}\\"); 9301b654882SGreg Clayton if (non_special_chars > 0) 9311b654882SGreg Clayton { 9321b654882SGreg Clayton if (success) 9331b654882SGreg Clayton s.Write (p, non_special_chars); 9341b654882SGreg Clayton p += non_special_chars; 9351b654882SGreg Clayton } 9361b654882SGreg Clayton 9371b654882SGreg Clayton if (*p == '\0') 9381b654882SGreg Clayton { 9391b654882SGreg Clayton break; 9401b654882SGreg Clayton } 9411b654882SGreg Clayton else if (*p == '{') 9421b654882SGreg Clayton { 9431b654882SGreg Clayton // Start a new scope that must have everything it needs if it is to 9441b654882SGreg Clayton // to make it into the final output stream "s". If you want to make 9451b654882SGreg Clayton // a format that only prints out the function or symbol name if there 9461b654882SGreg Clayton // is one in the symbol context you can use: 9471b654882SGreg Clayton // "{function =${function.name}}" 9481b654882SGreg Clayton // The first '{' starts a new scope that end with the matching '}' at 9491b654882SGreg Clayton // the end of the string. The contents "function =${function.name}" 9501b654882SGreg Clayton // will then be evaluated and only be output if there is a function 9511b654882SGreg Clayton // or symbol with a valid name. 9521b654882SGreg Clayton StreamString sub_strm; 9531b654882SGreg Clayton 9541b654882SGreg Clayton ++p; // Skip the '{' 9551b654882SGreg Clayton 956c482a192SEnrico Granata if (FormatPrompt (p, sc, exe_ctx, addr, sub_strm, &p, valobj)) 9571b654882SGreg Clayton { 9581b654882SGreg Clayton // The stream had all it needed 9591b654882SGreg Clayton s.Write(sub_strm.GetData(), sub_strm.GetSize()); 9601b654882SGreg Clayton } 9611b654882SGreg Clayton if (*p != '}') 9621b654882SGreg Clayton { 9631b654882SGreg Clayton success = false; 9641b654882SGreg Clayton break; 9651b654882SGreg Clayton } 9661b654882SGreg Clayton } 9671b654882SGreg Clayton else if (*p == '}') 9681b654882SGreg Clayton { 9691b654882SGreg Clayton // End of a enclosing scope 9701b654882SGreg Clayton break; 9711b654882SGreg Clayton } 9721b654882SGreg Clayton else if (*p == '$') 9731b654882SGreg Clayton { 9741b654882SGreg Clayton // We have a prompt variable to print 9751b654882SGreg Clayton ++p; 9761b654882SGreg Clayton if (*p == '{') 9771b654882SGreg Clayton { 9781b654882SGreg Clayton ++p; 9791b654882SGreg Clayton const char *var_name_begin = p; 9801b654882SGreg Clayton const char *var_name_end = ::strchr (p, '}'); 9811b654882SGreg Clayton 9821b654882SGreg Clayton if (var_name_end && var_name_begin < var_name_end) 9831b654882SGreg Clayton { 9841b654882SGreg Clayton // if we have already failed to parse, skip this variable 9851b654882SGreg Clayton if (success) 9861b654882SGreg Clayton { 9871b654882SGreg Clayton const char *cstr = NULL; 9881b654882SGreg Clayton Address format_addr; 9891b654882SGreg Clayton bool calculate_format_addr_function_offset = false; 9901b654882SGreg Clayton // Set reg_kind and reg_num to invalid values 9911b654882SGreg Clayton RegisterKind reg_kind = kNumRegisterKinds; 9921b654882SGreg Clayton uint32_t reg_num = LLDB_INVALID_REGNUM; 9931b654882SGreg Clayton FileSpec format_file_spec; 994e0d378b3SGreg Clayton const RegisterInfo *reg_info = NULL; 9951b654882SGreg Clayton RegisterContext *reg_ctx = NULL; 9969fc1944eSEnrico Granata bool do_deref_pointer = false; 997e992a089SEnrico Granata ValueObject::ExpressionPathScanEndReason reason_to_stop = ValueObject::eEndOfString; 998e992a089SEnrico Granata ValueObject::ExpressionPathEndResultType final_value_type = ValueObject::ePlain; 9991b654882SGreg Clayton 10001b654882SGreg Clayton // Each variable must set success to true below... 10011b654882SGreg Clayton bool var_success = false; 10021b654882SGreg Clayton switch (var_name_begin[0]) 10031b654882SGreg Clayton { 10044becb37eSEnrico Granata case '*': 10056f3533fbSEnrico Granata case 'v': 10066f3533fbSEnrico Granata case 's': 10074becb37eSEnrico Granata { 1008c482a192SEnrico Granata if (!valobj) 100934132754SGreg Clayton break; 10106f3533fbSEnrico Granata 1011c3e320a7SEnrico Granata if (log) 1012c3e320a7SEnrico Granata log->Printf("initial string: %s",var_name_begin); 1013c3e320a7SEnrico Granata 10146f3533fbSEnrico Granata // check for *var and *svar 10156f3533fbSEnrico Granata if (*var_name_begin == '*') 10166f3533fbSEnrico Granata { 10179fc1944eSEnrico Granata do_deref_pointer = true; 10189fc1944eSEnrico Granata var_name_begin++; 10199fc1944eSEnrico Granata } 1020c3e320a7SEnrico Granata 1021c3e320a7SEnrico Granata if (log) 1022c3e320a7SEnrico Granata log->Printf("initial string: %s",var_name_begin); 1023c3e320a7SEnrico Granata 10246f3533fbSEnrico Granata if (*var_name_begin == 's') 10254becb37eSEnrico Granata { 10264d122c40SGreg Clayton valobj = valobj->GetSyntheticValue(eUseSyntheticFilter).get(); 10276f3533fbSEnrico Granata var_name_begin++; 10286f3533fbSEnrico Granata } 10296f3533fbSEnrico Granata 1030c3e320a7SEnrico Granata if (log) 1031c3e320a7SEnrico Granata log->Printf("initial string: %s",var_name_begin); 1032c3e320a7SEnrico Granata 10336f3533fbSEnrico Granata // should be a 'v' by now 10346f3533fbSEnrico Granata if (*var_name_begin != 'v') 10356f3533fbSEnrico Granata break; 10366f3533fbSEnrico Granata 1037c3e320a7SEnrico Granata if (log) 1038c3e320a7SEnrico Granata log->Printf("initial string: %s",var_name_begin); 1039c3e320a7SEnrico Granata 1040fc7a7f3bSEnrico Granata ValueObject::ExpressionPathAftermath what_next = (do_deref_pointer ? 1041fc7a7f3bSEnrico Granata ValueObject::eDereference : ValueObject::eNothing); 1042fc7a7f3bSEnrico Granata ValueObject::GetValueForExpressionPathOptions options; 10438c9d3560SEnrico Granata options.DontCheckDotVsArrowSyntax().DoAllowBitfieldSyntax().DoAllowFragileIVar().DoAllowSyntheticChildren(); 10440a3958e0SEnrico Granata ValueObject::ValueObjectRepresentationStyle val_obj_display = ValueObject::eDisplaySummary; 104534132754SGreg Clayton ValueObject* target = NULL; 10464d122c40SGreg Clayton Format custom_format = eFormatInvalid; 104734132754SGreg Clayton const char* var_name_final = NULL; 10489fc1944eSEnrico Granata const char* var_name_final_if_array_range = NULL; 104934132754SGreg Clayton const char* close_bracket_position = NULL; 105034132754SGreg Clayton int64_t index_lower = -1; 105134132754SGreg Clayton int64_t index_higher = -1; 10529fc1944eSEnrico Granata bool is_array_range = false; 1053fc7a7f3bSEnrico Granata const char* first_unparsed; 105485933ed4SEnrico Granata bool was_plain_var = false; 105585933ed4SEnrico Granata bool was_var_format = false; 1056fc7a7f3bSEnrico Granata 1057c482a192SEnrico Granata if (!valobj) break; 1058c482a192SEnrico Granata // simplest case ${var}, just print valobj's value 10599fc1944eSEnrico Granata if (::strncmp (var_name_begin, "var}", strlen("var}")) == 0) 10600a3958e0SEnrico Granata { 106185933ed4SEnrico Granata was_plain_var = true; 1062c482a192SEnrico Granata target = valobj; 10630a3958e0SEnrico Granata val_obj_display = ValueObject::eDisplayValue; 10640a3958e0SEnrico Granata } 10659fc1944eSEnrico Granata else if (::strncmp(var_name_begin,"var%",strlen("var%")) == 0) 10669fc1944eSEnrico Granata { 106785933ed4SEnrico Granata was_var_format = true; 10689fc1944eSEnrico Granata // this is a variable with some custom format applied to it 10699fc1944eSEnrico Granata const char* percent_position; 1070c482a192SEnrico Granata target = valobj; 10710a3958e0SEnrico Granata val_obj_display = ValueObject::eDisplayValue; 10729fc1944eSEnrico Granata ScanFormatDescriptor (var_name_begin, 10739fc1944eSEnrico Granata var_name_end, 10749fc1944eSEnrico Granata &var_name_final, 10759fc1944eSEnrico Granata &percent_position, 10769fc1944eSEnrico Granata &custom_format, 10779fc1944eSEnrico Granata &val_obj_display); 10780a3958e0SEnrico Granata } 10799fc1944eSEnrico Granata // this is ${var.something} or multiple .something nested 10809fc1944eSEnrico Granata else if (::strncmp (var_name_begin, "var", strlen("var")) == 0) 10819fc1944eSEnrico Granata { 10829fc1944eSEnrico Granata 10839fc1944eSEnrico Granata const char* percent_position; 10849fc1944eSEnrico Granata ScanFormatDescriptor (var_name_begin, 10859fc1944eSEnrico Granata var_name_end, 10869fc1944eSEnrico Granata &var_name_final, 10879fc1944eSEnrico Granata &percent_position, 10889fc1944eSEnrico Granata &custom_format, 10899fc1944eSEnrico Granata &val_obj_display); 10909fc1944eSEnrico Granata 10919fc1944eSEnrico Granata const char* open_bracket_position; 10929fc1944eSEnrico Granata const char* separator_position; 10939fc1944eSEnrico Granata ScanBracketedRange (var_name_begin, 10949fc1944eSEnrico Granata var_name_end, 10959fc1944eSEnrico Granata var_name_final, 10969fc1944eSEnrico Granata &open_bracket_position, 10979fc1944eSEnrico Granata &separator_position, 10989fc1944eSEnrico Granata &close_bracket_position, 10999fc1944eSEnrico Granata &var_name_final_if_array_range, 11009fc1944eSEnrico Granata &index_lower, 11019fc1944eSEnrico Granata &index_higher); 11029fc1944eSEnrico Granata 11039fc1944eSEnrico Granata Error error; 11049fc1944eSEnrico Granata 1105fc7a7f3bSEnrico Granata std::auto_ptr<char> expr_path(new char[var_name_final-var_name_begin-1]); 1106fc7a7f3bSEnrico Granata ::memset(expr_path.get(), 0, var_name_final-var_name_begin-1); 1107fc7a7f3bSEnrico Granata memcpy(expr_path.get(), var_name_begin+3,var_name_final-var_name_begin-3); 1108fc7a7f3bSEnrico Granata 1109e992a089SEnrico Granata if (log) 1110e992a089SEnrico Granata log->Printf("symbol to expand: %s",expr_path.get()); 1111fc7a7f3bSEnrico Granata 1112c482a192SEnrico Granata target = valobj->GetValueForExpressionPath(expr_path.get(), 1113fc7a7f3bSEnrico Granata &first_unparsed, 1114fc7a7f3bSEnrico Granata &reason_to_stop, 1115fc7a7f3bSEnrico Granata &final_value_type, 1116fc7a7f3bSEnrico Granata options, 1117fc7a7f3bSEnrico Granata &what_next).get(); 1118fc7a7f3bSEnrico Granata 1119fc7a7f3bSEnrico Granata if (!target) 11209fc1944eSEnrico Granata { 1121e992a089SEnrico Granata if (log) 1122e992a089SEnrico Granata log->Printf("ERROR: unparsed portion = %s, why stopping = %d," 1123e992a089SEnrico Granata " final_value_type %d", 1124fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 1125fc7a7f3bSEnrico Granata break; 11260a3958e0SEnrico Granata } 1127a7187d00SEnrico Granata else 1128fc7a7f3bSEnrico Granata { 1129e992a089SEnrico Granata if (log) 1130e992a089SEnrico Granata log->Printf("ALL RIGHT: unparsed portion = %s, why stopping = %d," 1131e992a089SEnrico Granata " final_value_type %d", 1132fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 1133a7187d00SEnrico Granata } 11340a3958e0SEnrico Granata } 11350a3958e0SEnrico Granata else 11360a3958e0SEnrico Granata break; 11379fc1944eSEnrico Granata 1138fc7a7f3bSEnrico Granata is_array_range = (final_value_type == ValueObject::eBoundedRange || 1139fc7a7f3bSEnrico Granata final_value_type == ValueObject::eUnboundedRange); 1140fc7a7f3bSEnrico Granata 1141fc7a7f3bSEnrico Granata do_deref_pointer = (what_next == ValueObject::eDereference); 1142fc7a7f3bSEnrico Granata 1143a7187d00SEnrico Granata if (do_deref_pointer && !is_array_range) 11440a3958e0SEnrico Granata { 11459fc1944eSEnrico Granata // I have not deref-ed yet, let's do it 11469fc1944eSEnrico Granata // this happens when we are not going through GetValueForVariableExpressionPath 11479fc1944eSEnrico Granata // to get to the target ValueObject 11489fc1944eSEnrico Granata Error error; 11499fc1944eSEnrico Granata target = target->Dereference(error).get(); 1150dc940730SEnrico Granata if (error.Fail()) 1151dc940730SEnrico Granata { 1152dc940730SEnrico Granata if (log) 1153dc940730SEnrico Granata log->Printf("ERROR: %s\n", error.AsCString("unknown")); \ 1154dc940730SEnrico Granata break; 1155dc940730SEnrico Granata } 11569fc1944eSEnrico Granata do_deref_pointer = false; 11570a3958e0SEnrico Granata } 11580a3958e0SEnrico Granata 115985933ed4SEnrico Granata // TODO use flags for these 1160f4efecd9SEnrico Granata bool is_array = ClangASTContext::IsArrayType(target->GetClangType()); 1161f4efecd9SEnrico Granata bool is_pointer = ClangASTContext::IsPointerType(target->GetClangType()); 116285933ed4SEnrico Granata bool is_aggregate = ClangASTContext::IsAggregateType(target->GetClangType()); 1163f4efecd9SEnrico Granata 1164f4efecd9SEnrico Granata if ((is_array || is_pointer) && (!is_array_range) && val_obj_display == ValueObject::eDisplayValue) // this should be wrong, but there are some exceptions 1165f4efecd9SEnrico Granata { 116685933ed4SEnrico Granata StreamString str_temp; 1167e992a089SEnrico Granata if (log) 1168e992a089SEnrico Granata log->Printf("I am into array || pointer && !range"); 1169d64d0bc0SEnrico Granata 1170d64d0bc0SEnrico Granata if (target->HasSpecialCasesForPrintableRepresentation(val_obj_display, 1171d64d0bc0SEnrico Granata custom_format)) 1172d64d0bc0SEnrico Granata { 1173f4efecd9SEnrico Granata // try to use the special cases 117485933ed4SEnrico Granata var_success = target->DumpPrintableRepresentation(str_temp, 117585933ed4SEnrico Granata val_obj_display, 117685933ed4SEnrico Granata custom_format); 1177e992a089SEnrico Granata if (log) 1178e992a089SEnrico Granata log->Printf("special cases did%s match", var_success ? "" : "n't"); 1179d64d0bc0SEnrico Granata 1180d64d0bc0SEnrico Granata // should not happen 118185933ed4SEnrico Granata if (!var_success) 118285933ed4SEnrico Granata s << "<invalid usage of pointer value as object>"; 118385933ed4SEnrico Granata else 118485933ed4SEnrico Granata s << str_temp.GetData(); 1185d64d0bc0SEnrico Granata var_success = true; 1186d64d0bc0SEnrico Granata break; 1187d64d0bc0SEnrico Granata } 1188d64d0bc0SEnrico Granata else 1189d64d0bc0SEnrico Granata { 119088da35f8SEnrico Granata if (was_plain_var) // if ${var} 1191d64d0bc0SEnrico Granata { 1192d64d0bc0SEnrico Granata s << target->GetTypeName() << " @ " << target->GetLocationAsCString(); 1193d64d0bc0SEnrico Granata } 119488da35f8SEnrico Granata else if (is_pointer) // if pointer, value is the address stored 119588da35f8SEnrico Granata { 119688da35f8SEnrico Granata var_success = target->GetPrintableRepresentation(s, 119788da35f8SEnrico Granata val_obj_display, 119888da35f8SEnrico Granata custom_format); 119988da35f8SEnrico Granata } 1200d64d0bc0SEnrico Granata else 1201d64d0bc0SEnrico Granata { 1202d64d0bc0SEnrico Granata s << "<invalid usage of pointer value as object>"; 1203d64d0bc0SEnrico Granata } 1204d64d0bc0SEnrico Granata var_success = true; 1205d64d0bc0SEnrico Granata break; 1206d64d0bc0SEnrico Granata } 1207d64d0bc0SEnrico Granata } 1208d64d0bc0SEnrico Granata 1209d64d0bc0SEnrico Granata // if directly trying to print ${var}, and this is an aggregate, display a nice 1210d64d0bc0SEnrico Granata // type @ location message 1211d64d0bc0SEnrico Granata if (is_aggregate && was_plain_var) 1212d64d0bc0SEnrico Granata { 1213d64d0bc0SEnrico Granata s << target->GetTypeName() << " @ " << target->GetLocationAsCString(); 1214d64d0bc0SEnrico Granata var_success = true; 121585933ed4SEnrico Granata break; 121685933ed4SEnrico Granata } 121785933ed4SEnrico Granata 1218d64d0bc0SEnrico Granata // if directly trying to print ${var%V}, and this is an aggregate, do not let the user do it 1219d64d0bc0SEnrico Granata if (is_aggregate && ((was_var_format && val_obj_display == ValueObject::eDisplayValue))) 122085933ed4SEnrico Granata { 122185933ed4SEnrico Granata s << "<invalid use of aggregate type>"; 122285933ed4SEnrico Granata var_success = true; 1223f4efecd9SEnrico Granata break; 1224f4efecd9SEnrico Granata } 1225f4efecd9SEnrico Granata 12269fc1944eSEnrico Granata if (!is_array_range) 1227e992a089SEnrico Granata { 1228e992a089SEnrico Granata if (log) 1229e992a089SEnrico Granata log->Printf("dumping ordinary printable output"); 12309fc1944eSEnrico Granata var_success = target->DumpPrintableRepresentation(s,val_obj_display, custom_format); 1231e992a089SEnrico Granata } 12329fc1944eSEnrico Granata else 12339fc1944eSEnrico Granata { 1234e992a089SEnrico Granata if (log) 1235e992a089SEnrico Granata log->Printf("checking if I can handle as array"); 12369fc1944eSEnrico Granata if (!is_array && !is_pointer) 12379fc1944eSEnrico Granata break; 1238e992a089SEnrico Granata if (log) 1239e992a089SEnrico Granata log->Printf("handle as array"); 1240fc7a7f3bSEnrico Granata const char* special_directions = NULL; 1241fc7a7f3bSEnrico Granata StreamString special_directions_writer; 12420a3958e0SEnrico Granata if (close_bracket_position && (var_name_end-close_bracket_position > 1)) 12430a3958e0SEnrico Granata { 1244fc7a7f3bSEnrico Granata ConstString additional_data; 1245fc7a7f3bSEnrico Granata additional_data.SetCStringWithLength(close_bracket_position+1, var_name_end-close_bracket_position-1); 1246fc7a7f3bSEnrico Granata special_directions_writer.Printf("${%svar%s}", 1247fc7a7f3bSEnrico Granata do_deref_pointer ? "*" : "", 1248fc7a7f3bSEnrico Granata additional_data.GetCString()); 1249fc7a7f3bSEnrico Granata special_directions = special_directions_writer.GetData(); 12500a3958e0SEnrico Granata } 12510a3958e0SEnrico Granata 12520a3958e0SEnrico Granata // let us display items index_lower thru index_higher of this array 12530a3958e0SEnrico Granata s.PutChar('['); 12540a3958e0SEnrico Granata var_success = true; 12550a3958e0SEnrico Granata 12569fc1944eSEnrico Granata if (index_higher < 0) 1257c482a192SEnrico Granata index_higher = valobj->GetNumChildren() - 1; 12580a3958e0SEnrico Granata 125922c55d18SEnrico Granata uint32_t max_num_children = target->GetUpdatePoint().GetTargetSP()->GetMaximumNumberOfChildrenToDisplay(); 126022c55d18SEnrico Granata 12610a3958e0SEnrico Granata for (;index_lower<=index_higher;index_lower++) 12620a3958e0SEnrico Granata { 1263fc7a7f3bSEnrico Granata ValueObject* item = ExpandIndexedExpression (target, 12649fc1944eSEnrico Granata index_lower, 1265c14ee32dSGreg Clayton exe_ctx->GetFramePtr(), 1266fc7a7f3bSEnrico Granata false).get(); 12670a3958e0SEnrico Granata 1268fc7a7f3bSEnrico Granata if (!item) 1269fc7a7f3bSEnrico Granata { 1270e992a089SEnrico Granata if (log) 1271fd54b368SJason Molenda log->Printf("ERROR in getting child item at index %lld", index_lower); 1272fc7a7f3bSEnrico Granata } 1273fc7a7f3bSEnrico Granata else 1274fc7a7f3bSEnrico Granata { 1275e992a089SEnrico Granata if (log) 1276e992a089SEnrico Granata log->Printf("special_directions for child item: %s",special_directions); 1277fc7a7f3bSEnrico Granata } 1278fc7a7f3bSEnrico Granata 12790a3958e0SEnrico Granata if (!special_directions) 12809fc1944eSEnrico Granata var_success &= item->DumpPrintableRepresentation(s,val_obj_display, custom_format); 12810a3958e0SEnrico Granata else 12820a3958e0SEnrico Granata var_success &= FormatPrompt(special_directions, sc, exe_ctx, addr, s, NULL, item); 12830a3958e0SEnrico Granata 128422c55d18SEnrico Granata if (--max_num_children == 0) 128522c55d18SEnrico Granata { 128622c55d18SEnrico Granata s.PutCString(", ..."); 128722c55d18SEnrico Granata break; 128822c55d18SEnrico Granata } 128922c55d18SEnrico Granata 12900a3958e0SEnrico Granata if (index_lower < index_higher) 12910a3958e0SEnrico Granata s.PutChar(','); 12920a3958e0SEnrico Granata } 12930a3958e0SEnrico Granata s.PutChar(']'); 12944becb37eSEnrico Granata } 12954becb37eSEnrico Granata } 129634132754SGreg Clayton break; 12971b654882SGreg Clayton case 'a': 12981b654882SGreg Clayton if (::strncmp (var_name_begin, "addr}", strlen("addr}")) == 0) 12991b654882SGreg Clayton { 13001b654882SGreg Clayton if (addr && addr->IsValid()) 13011b654882SGreg Clayton { 13021b654882SGreg Clayton var_success = true; 13031b654882SGreg Clayton format_addr = *addr; 13041b654882SGreg Clayton } 13051b654882SGreg Clayton } 13065a31471eSGreg Clayton else if (::strncmp (var_name_begin, "ansi.", strlen("ansi.")) == 0) 13075a31471eSGreg Clayton { 13085a31471eSGreg Clayton var_success = true; 13095a31471eSGreg Clayton var_name_begin += strlen("ansi."); // Skip the "ansi." 13105a31471eSGreg Clayton if (::strncmp (var_name_begin, "fg.", strlen("fg.")) == 0) 13115a31471eSGreg Clayton { 13125a31471eSGreg Clayton var_name_begin += strlen("fg."); // Skip the "fg." 13135a31471eSGreg Clayton if (::strncmp (var_name_begin, "black}", strlen("black}")) == 0) 13145a31471eSGreg Clayton { 13155a31471eSGreg Clayton s.Printf ("%s%s%s", 13165a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13175a31471eSGreg Clayton lldb_utility::ansi::k_fg_black, 13185a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13195a31471eSGreg Clayton } 13205a31471eSGreg Clayton else if (::strncmp (var_name_begin, "red}", strlen("red}")) == 0) 13215a31471eSGreg Clayton { 13225a31471eSGreg Clayton s.Printf ("%s%s%s", 13235a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13245a31471eSGreg Clayton lldb_utility::ansi::k_fg_red, 13255a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13265a31471eSGreg Clayton } 13275a31471eSGreg Clayton else if (::strncmp (var_name_begin, "green}", strlen("green}")) == 0) 13285a31471eSGreg Clayton { 13295a31471eSGreg Clayton s.Printf ("%s%s%s", 13305a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13315a31471eSGreg Clayton lldb_utility::ansi::k_fg_green, 13325a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13335a31471eSGreg Clayton } 13345a31471eSGreg Clayton else if (::strncmp (var_name_begin, "yellow}", strlen("yellow}")) == 0) 13355a31471eSGreg Clayton { 13365a31471eSGreg Clayton s.Printf ("%s%s%s", 13375a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13385a31471eSGreg Clayton lldb_utility::ansi::k_fg_yellow, 13395a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13405a31471eSGreg Clayton } 13415a31471eSGreg Clayton else if (::strncmp (var_name_begin, "blue}", strlen("blue}")) == 0) 13425a31471eSGreg Clayton { 13435a31471eSGreg Clayton s.Printf ("%s%s%s", 13445a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13455a31471eSGreg Clayton lldb_utility::ansi::k_fg_blue, 13465a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13475a31471eSGreg Clayton } 13485a31471eSGreg Clayton else if (::strncmp (var_name_begin, "purple}", strlen("purple}")) == 0) 13495a31471eSGreg Clayton { 13505a31471eSGreg Clayton s.Printf ("%s%s%s", 13515a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13525a31471eSGreg Clayton lldb_utility::ansi::k_fg_purple, 13535a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13545a31471eSGreg Clayton } 13555a31471eSGreg Clayton else if (::strncmp (var_name_begin, "cyan}", strlen("cyan}")) == 0) 13565a31471eSGreg Clayton { 13575a31471eSGreg Clayton s.Printf ("%s%s%s", 13585a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13595a31471eSGreg Clayton lldb_utility::ansi::k_fg_cyan, 13605a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13615a31471eSGreg Clayton } 13625a31471eSGreg Clayton else if (::strncmp (var_name_begin, "white}", strlen("white}")) == 0) 13635a31471eSGreg Clayton { 13645a31471eSGreg Clayton s.Printf ("%s%s%s", 13655a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13665a31471eSGreg Clayton lldb_utility::ansi::k_fg_white, 13675a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13685a31471eSGreg Clayton } 13695a31471eSGreg Clayton else 13705a31471eSGreg Clayton { 13715a31471eSGreg Clayton var_success = false; 13725a31471eSGreg Clayton } 13735a31471eSGreg Clayton } 13745a31471eSGreg Clayton else if (::strncmp (var_name_begin, "bg.", strlen("bg.")) == 0) 13755a31471eSGreg Clayton { 13765a31471eSGreg Clayton var_name_begin += strlen("bg."); // Skip the "bg." 13775a31471eSGreg Clayton if (::strncmp (var_name_begin, "black}", strlen("black}")) == 0) 13785a31471eSGreg Clayton { 13795a31471eSGreg Clayton s.Printf ("%s%s%s", 13805a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13815a31471eSGreg Clayton lldb_utility::ansi::k_bg_black, 13825a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13835a31471eSGreg Clayton } 13845a31471eSGreg Clayton else if (::strncmp (var_name_begin, "red}", strlen("red}")) == 0) 13855a31471eSGreg Clayton { 13865a31471eSGreg Clayton s.Printf ("%s%s%s", 13875a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13885a31471eSGreg Clayton lldb_utility::ansi::k_bg_red, 13895a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13905a31471eSGreg Clayton } 13915a31471eSGreg Clayton else if (::strncmp (var_name_begin, "green}", strlen("green}")) == 0) 13925a31471eSGreg Clayton { 13935a31471eSGreg Clayton s.Printf ("%s%s%s", 13945a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 13955a31471eSGreg Clayton lldb_utility::ansi::k_bg_green, 13965a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 13975a31471eSGreg Clayton } 13985a31471eSGreg Clayton else if (::strncmp (var_name_begin, "yellow}", strlen("yellow}")) == 0) 13995a31471eSGreg Clayton { 14005a31471eSGreg Clayton s.Printf ("%s%s%s", 14015a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14025a31471eSGreg Clayton lldb_utility::ansi::k_bg_yellow, 14035a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14045a31471eSGreg Clayton } 14055a31471eSGreg Clayton else if (::strncmp (var_name_begin, "blue}", strlen("blue}")) == 0) 14065a31471eSGreg Clayton { 14075a31471eSGreg Clayton s.Printf ("%s%s%s", 14085a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14095a31471eSGreg Clayton lldb_utility::ansi::k_bg_blue, 14105a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14115a31471eSGreg Clayton } 14125a31471eSGreg Clayton else if (::strncmp (var_name_begin, "purple}", strlen("purple}")) == 0) 14135a31471eSGreg Clayton { 14145a31471eSGreg Clayton s.Printf ("%s%s%s", 14155a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14165a31471eSGreg Clayton lldb_utility::ansi::k_bg_purple, 14175a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14185a31471eSGreg Clayton } 14195a31471eSGreg Clayton else if (::strncmp (var_name_begin, "cyan}", strlen("cyan}")) == 0) 14205a31471eSGreg Clayton { 14215a31471eSGreg Clayton s.Printf ("%s%s%s", 14225a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14235a31471eSGreg Clayton lldb_utility::ansi::k_bg_cyan, 14245a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14255a31471eSGreg Clayton } 14265a31471eSGreg Clayton else if (::strncmp (var_name_begin, "white}", strlen("white}")) == 0) 14275a31471eSGreg Clayton { 14285a31471eSGreg Clayton s.Printf ("%s%s%s", 14295a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14305a31471eSGreg Clayton lldb_utility::ansi::k_bg_white, 14315a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14325a31471eSGreg Clayton } 14335a31471eSGreg Clayton else 14345a31471eSGreg Clayton { 14355a31471eSGreg Clayton var_success = false; 14365a31471eSGreg Clayton } 14375a31471eSGreg Clayton } 14385a31471eSGreg Clayton else if (::strncmp (var_name_begin, "normal}", strlen ("normal}")) == 0) 14395a31471eSGreg Clayton { 14405a31471eSGreg Clayton s.Printf ("%s%s%s", 14415a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14425a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_normal, 14435a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14445a31471eSGreg Clayton } 14455a31471eSGreg Clayton else if (::strncmp (var_name_begin, "bold}", strlen("bold}")) == 0) 14465a31471eSGreg Clayton { 14475a31471eSGreg Clayton s.Printf ("%s%s%s", 14485a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14495a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_bold, 14505a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14515a31471eSGreg Clayton } 14525a31471eSGreg Clayton else if (::strncmp (var_name_begin, "faint}", strlen("faint}")) == 0) 14535a31471eSGreg Clayton { 14545a31471eSGreg Clayton s.Printf ("%s%s%s", 14555a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14565a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_faint, 14575a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14585a31471eSGreg Clayton } 14595a31471eSGreg Clayton else if (::strncmp (var_name_begin, "italic}", strlen("italic}")) == 0) 14605a31471eSGreg Clayton { 14615a31471eSGreg Clayton s.Printf ("%s%s%s", 14625a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14635a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_italic, 14645a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14655a31471eSGreg Clayton } 14665a31471eSGreg Clayton else if (::strncmp (var_name_begin, "underline}", strlen("underline}")) == 0) 14675a31471eSGreg Clayton { 14685a31471eSGreg Clayton s.Printf ("%s%s%s", 14695a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14705a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_underline, 14715a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14725a31471eSGreg Clayton } 14735a31471eSGreg Clayton else if (::strncmp (var_name_begin, "slow-blink}", strlen("slow-blink}")) == 0) 14745a31471eSGreg Clayton { 14755a31471eSGreg Clayton s.Printf ("%s%s%s", 14765a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14775a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_slow_blink, 14785a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14795a31471eSGreg Clayton } 14805a31471eSGreg Clayton else if (::strncmp (var_name_begin, "fast-blink}", strlen("fast-blink}")) == 0) 14815a31471eSGreg Clayton { 14825a31471eSGreg Clayton s.Printf ("%s%s%s", 14835a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14845a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_fast_blink, 14855a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14865a31471eSGreg Clayton } 14875a31471eSGreg Clayton else if (::strncmp (var_name_begin, "negative}", strlen("negative}")) == 0) 14885a31471eSGreg Clayton { 14895a31471eSGreg Clayton s.Printf ("%s%s%s", 14905a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14915a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_negative, 14925a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 14935a31471eSGreg Clayton } 14945a31471eSGreg Clayton else if (::strncmp (var_name_begin, "conceal}", strlen("conceal}")) == 0) 14955a31471eSGreg Clayton { 14965a31471eSGreg Clayton s.Printf ("%s%s%s", 14975a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 14985a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_conceal, 14995a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15005a31471eSGreg Clayton 15015a31471eSGreg Clayton } 15025a31471eSGreg Clayton else if (::strncmp (var_name_begin, "crossed-out}", strlen("crossed-out}")) == 0) 15035a31471eSGreg Clayton { 15045a31471eSGreg Clayton s.Printf ("%s%s%s", 15055a31471eSGreg Clayton lldb_utility::ansi::k_escape_start, 15065a31471eSGreg Clayton lldb_utility::ansi::k_ctrl_crossed_out, 15075a31471eSGreg Clayton lldb_utility::ansi::k_escape_end); 15085a31471eSGreg Clayton } 15095a31471eSGreg Clayton else 15105a31471eSGreg Clayton { 15115a31471eSGreg Clayton var_success = false; 15125a31471eSGreg Clayton } 15135a31471eSGreg Clayton } 15141b654882SGreg Clayton break; 15151b654882SGreg Clayton 15161b654882SGreg Clayton case 'p': 15171b654882SGreg Clayton if (::strncmp (var_name_begin, "process.", strlen("process.")) == 0) 15181b654882SGreg Clayton { 1519c14ee32dSGreg Clayton if (exe_ctx) 1520c14ee32dSGreg Clayton { 1521c14ee32dSGreg Clayton Process *process = exe_ctx->GetProcessPtr(); 1522c14ee32dSGreg Clayton if (process) 15231b654882SGreg Clayton { 15241b654882SGreg Clayton var_name_begin += ::strlen ("process."); 15251b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 15261b654882SGreg Clayton { 152781c22f61SGreg Clayton s.Printf("%llu", process->GetID()); 15281b654882SGreg Clayton var_success = true; 15291b654882SGreg Clayton } 15301b654882SGreg Clayton else if ((::strncmp (var_name_begin, "name}", strlen("name}")) == 0) || 15311b654882SGreg Clayton (::strncmp (var_name_begin, "file.basename}", strlen("file.basename}")) == 0) || 15321b654882SGreg Clayton (::strncmp (var_name_begin, "file.fullpath}", strlen("file.fullpath}")) == 0)) 15331b654882SGreg Clayton { 1534c14ee32dSGreg Clayton Module *exe_module = process->GetTarget().GetExecutableModulePointer(); 1535aa149cbdSGreg Clayton if (exe_module) 15361b654882SGreg Clayton { 15371b654882SGreg Clayton if (var_name_begin[0] == 'n' || var_name_begin[5] == 'f') 15381b654882SGreg Clayton { 1539aa149cbdSGreg Clayton format_file_spec.GetFilename() = exe_module->GetFileSpec().GetFilename(); 15401b654882SGreg Clayton var_success = format_file_spec; 15411b654882SGreg Clayton } 15421b654882SGreg Clayton else 15431b654882SGreg Clayton { 1544aa149cbdSGreg Clayton format_file_spec = exe_module->GetFileSpec(); 15451b654882SGreg Clayton var_success = format_file_spec; 15461b654882SGreg Clayton } 15471b654882SGreg Clayton } 15481b654882SGreg Clayton } 15491b654882SGreg Clayton } 15501b654882SGreg Clayton } 1551c14ee32dSGreg Clayton } 15521b654882SGreg Clayton break; 15531b654882SGreg Clayton 15541b654882SGreg Clayton case 't': 15551b654882SGreg Clayton if (::strncmp (var_name_begin, "thread.", strlen("thread.")) == 0) 15561b654882SGreg Clayton { 1557c14ee32dSGreg Clayton if (exe_ctx) 1558c14ee32dSGreg Clayton { 1559c14ee32dSGreg Clayton Thread *thread = exe_ctx->GetThreadPtr(); 1560c14ee32dSGreg Clayton if (thread) 15611b654882SGreg Clayton { 15621b654882SGreg Clayton var_name_begin += ::strlen ("thread."); 15631b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 15641b654882SGreg Clayton { 156581c22f61SGreg Clayton s.Printf("0x%4.4llx", thread->GetID()); 15661b654882SGreg Clayton var_success = true; 15671b654882SGreg Clayton } 15681b654882SGreg Clayton else if (::strncmp (var_name_begin, "index}", strlen("index}")) == 0) 15691b654882SGreg Clayton { 1570c14ee32dSGreg Clayton s.Printf("%u", thread->GetIndexID()); 15711b654882SGreg Clayton var_success = true; 15721b654882SGreg Clayton } 15731b654882SGreg Clayton else if (::strncmp (var_name_begin, "name}", strlen("name}")) == 0) 15741b654882SGreg Clayton { 1575c14ee32dSGreg Clayton cstr = thread->GetName(); 15761b654882SGreg Clayton var_success = cstr && cstr[0]; 15771b654882SGreg Clayton if (var_success) 15781b654882SGreg Clayton s.PutCString(cstr); 15791b654882SGreg Clayton } 15801b654882SGreg Clayton else if (::strncmp (var_name_begin, "queue}", strlen("queue}")) == 0) 15811b654882SGreg Clayton { 1582c14ee32dSGreg Clayton cstr = thread->GetQueueName(); 15831b654882SGreg Clayton var_success = cstr && cstr[0]; 15841b654882SGreg Clayton if (var_success) 15851b654882SGreg Clayton s.PutCString(cstr); 15861b654882SGreg Clayton } 15871b654882SGreg Clayton else if (::strncmp (var_name_begin, "stop-reason}", strlen("stop-reason}")) == 0) 15881b654882SGreg Clayton { 1589c14ee32dSGreg Clayton StopInfoSP stop_info_sp = thread->GetStopInfo (); 1590b15bfc75SJim Ingham if (stop_info_sp) 15911b654882SGreg Clayton { 1592b15bfc75SJim Ingham cstr = stop_info_sp->GetDescription(); 15931b654882SGreg Clayton if (cstr && cstr[0]) 15941b654882SGreg Clayton { 15951b654882SGreg Clayton s.PutCString(cstr); 15961b654882SGreg Clayton var_success = true; 15971b654882SGreg Clayton } 15981b654882SGreg Clayton } 15991b654882SGreg Clayton } 16001b654882SGreg Clayton } 16011b654882SGreg Clayton } 1602c14ee32dSGreg Clayton } 16031b654882SGreg Clayton else if (::strncmp (var_name_begin, "target.", strlen("target.")) == 0) 16041b654882SGreg Clayton { 16050603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 16060603aa9dSGreg Clayton if (target) 16071b654882SGreg Clayton { 16081b654882SGreg Clayton var_name_begin += ::strlen ("target."); 16091b654882SGreg Clayton if (::strncmp (var_name_begin, "arch}", strlen("arch}")) == 0) 16101b654882SGreg Clayton { 16111b654882SGreg Clayton ArchSpec arch (target->GetArchitecture ()); 16121b654882SGreg Clayton if (arch.IsValid()) 16131b654882SGreg Clayton { 161464195a2cSGreg Clayton s.PutCString (arch.GetArchitectureName()); 16151b654882SGreg Clayton var_success = true; 16161b654882SGreg Clayton } 16171b654882SGreg Clayton } 16181b654882SGreg Clayton } 16191b654882SGreg Clayton } 16201b654882SGreg Clayton break; 16211b654882SGreg Clayton 16221b654882SGreg Clayton 16231b654882SGreg Clayton case 'm': 16241b654882SGreg Clayton if (::strncmp (var_name_begin, "module.", strlen("module.")) == 0) 16251b654882SGreg Clayton { 16260603aa9dSGreg Clayton if (sc && sc->module_sp.get()) 16271b654882SGreg Clayton { 16280603aa9dSGreg Clayton Module *module = sc->module_sp.get(); 16291b654882SGreg Clayton var_name_begin += ::strlen ("module."); 16301b654882SGreg Clayton 16311b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 16321b654882SGreg Clayton { 16331b654882SGreg Clayton if (module->GetFileSpec()) 16341b654882SGreg Clayton { 16351b654882SGreg Clayton var_name_begin += ::strlen ("file."); 16361b654882SGreg Clayton 16371b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 16381b654882SGreg Clayton { 16391b654882SGreg Clayton format_file_spec.GetFilename() = module->GetFileSpec().GetFilename(); 16401b654882SGreg Clayton var_success = format_file_spec; 16411b654882SGreg Clayton } 16421b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 16431b654882SGreg Clayton { 16441b654882SGreg Clayton format_file_spec = module->GetFileSpec(); 16451b654882SGreg Clayton var_success = format_file_spec; 16461b654882SGreg Clayton } 16471b654882SGreg Clayton } 16481b654882SGreg Clayton } 16491b654882SGreg Clayton } 16501b654882SGreg Clayton } 16511b654882SGreg Clayton break; 16521b654882SGreg Clayton 16531b654882SGreg Clayton 16541b654882SGreg Clayton case 'f': 16551b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 16561b654882SGreg Clayton { 16571b654882SGreg Clayton if (sc && sc->comp_unit != NULL) 16581b654882SGreg Clayton { 16591b654882SGreg Clayton var_name_begin += ::strlen ("file."); 16601b654882SGreg Clayton 16611b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 16621b654882SGreg Clayton { 16631b654882SGreg Clayton format_file_spec.GetFilename() = sc->comp_unit->GetFilename(); 16641b654882SGreg Clayton var_success = format_file_spec; 16651b654882SGreg Clayton } 16661b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 16671b654882SGreg Clayton { 16681b654882SGreg Clayton format_file_spec = *sc->comp_unit; 16691b654882SGreg Clayton var_success = format_file_spec; 16701b654882SGreg Clayton } 16711b654882SGreg Clayton } 16721b654882SGreg Clayton } 16731b654882SGreg Clayton else if (::strncmp (var_name_begin, "frame.", strlen("frame.")) == 0) 16741b654882SGreg Clayton { 1675c14ee32dSGreg Clayton if (exe_ctx) 1676c14ee32dSGreg Clayton { 1677c14ee32dSGreg Clayton StackFrame *frame = exe_ctx->GetFramePtr(); 1678c14ee32dSGreg Clayton if (frame) 16791b654882SGreg Clayton { 16801b654882SGreg Clayton var_name_begin += ::strlen ("frame."); 16811b654882SGreg Clayton if (::strncmp (var_name_begin, "index}", strlen("index}")) == 0) 16821b654882SGreg Clayton { 1683c14ee32dSGreg Clayton s.Printf("%u", frame->GetFrameIndex()); 16841b654882SGreg Clayton var_success = true; 16851b654882SGreg Clayton } 16861b654882SGreg Clayton else if (::strncmp (var_name_begin, "pc}", strlen("pc}")) == 0) 16871b654882SGreg Clayton { 16881b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 16891b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_PC; 16901b654882SGreg Clayton var_success = true; 16911b654882SGreg Clayton } 16921b654882SGreg Clayton else if (::strncmp (var_name_begin, "sp}", strlen("sp}")) == 0) 16931b654882SGreg Clayton { 16941b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 16951b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_SP; 16961b654882SGreg Clayton var_success = true; 16971b654882SGreg Clayton } 16981b654882SGreg Clayton else if (::strncmp (var_name_begin, "fp}", strlen("fp}")) == 0) 16991b654882SGreg Clayton { 17001b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 17011b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_FP; 17021b654882SGreg Clayton var_success = true; 17031b654882SGreg Clayton } 17041b654882SGreg Clayton else if (::strncmp (var_name_begin, "flags}", strlen("flags}")) == 0) 17051b654882SGreg Clayton { 17061b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 17071b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_FLAGS; 17081b654882SGreg Clayton var_success = true; 17091b654882SGreg Clayton } 17101b654882SGreg Clayton else if (::strncmp (var_name_begin, "reg.", strlen ("reg.")) == 0) 17111b654882SGreg Clayton { 1712c14ee32dSGreg Clayton reg_ctx = frame->GetRegisterContext().get(); 17131b654882SGreg Clayton if (reg_ctx) 17141b654882SGreg Clayton { 17151b654882SGreg Clayton var_name_begin += ::strlen ("reg."); 17161b654882SGreg Clayton if (var_name_begin < var_name_end) 17171b654882SGreg Clayton { 17181b654882SGreg Clayton std::string reg_name (var_name_begin, var_name_end); 17191b654882SGreg Clayton reg_info = reg_ctx->GetRegisterInfoByName (reg_name.c_str()); 17201b654882SGreg Clayton if (reg_info) 17211b654882SGreg Clayton var_success = true; 17221b654882SGreg Clayton } 17231b654882SGreg Clayton } 17241b654882SGreg Clayton } 17251b654882SGreg Clayton } 17261b654882SGreg Clayton } 1727c14ee32dSGreg Clayton } 17281b654882SGreg Clayton else if (::strncmp (var_name_begin, "function.", strlen("function.")) == 0) 17291b654882SGreg Clayton { 17301b654882SGreg Clayton if (sc && (sc->function != NULL || sc->symbol != NULL)) 17311b654882SGreg Clayton { 17321b654882SGreg Clayton var_name_begin += ::strlen ("function."); 17331b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 17341b654882SGreg Clayton { 17351b654882SGreg Clayton if (sc->function) 173681c22f61SGreg Clayton s.Printf("function{0x%8.8llx}", sc->function->GetID()); 17371b654882SGreg Clayton else 17381b654882SGreg Clayton s.Printf("symbol[%u]", sc->symbol->GetID()); 17391b654882SGreg Clayton 17401b654882SGreg Clayton var_success = true; 17411b654882SGreg Clayton } 17421b654882SGreg Clayton else if (::strncmp (var_name_begin, "name}", strlen("name}")) == 0) 17431b654882SGreg Clayton { 17441b654882SGreg Clayton if (sc->function) 17451b654882SGreg Clayton cstr = sc->function->GetName().AsCString (NULL); 17461b654882SGreg Clayton else if (sc->symbol) 17471b654882SGreg Clayton cstr = sc->symbol->GetName().AsCString (NULL); 17481b654882SGreg Clayton if (cstr) 17491b654882SGreg Clayton { 17501b654882SGreg Clayton s.PutCString(cstr); 17510d9c9934SGreg Clayton 17520d9c9934SGreg Clayton if (sc->block) 17530d9c9934SGreg Clayton { 17540d9c9934SGreg Clayton Block *inline_block = sc->block->GetContainingInlinedBlock (); 17550d9c9934SGreg Clayton if (inline_block) 17560d9c9934SGreg Clayton { 17570d9c9934SGreg Clayton const InlineFunctionInfo *inline_info = sc->block->GetInlinedFunctionInfo(); 17580d9c9934SGreg Clayton if (inline_info) 17590d9c9934SGreg Clayton { 17600d9c9934SGreg Clayton s.PutCString(" [inlined] "); 17610d9c9934SGreg Clayton inline_info->GetName().Dump(&s); 17620d9c9934SGreg Clayton } 17630d9c9934SGreg Clayton } 17640d9c9934SGreg Clayton } 17651b654882SGreg Clayton var_success = true; 17661b654882SGreg Clayton } 17671b654882SGreg Clayton } 17681b654882SGreg Clayton else if (::strncmp (var_name_begin, "addr-offset}", strlen("addr-offset}")) == 0) 17691b654882SGreg Clayton { 17701b654882SGreg Clayton var_success = addr != NULL; 17711b654882SGreg Clayton if (var_success) 17721b654882SGreg Clayton { 17731b654882SGreg Clayton format_addr = *addr; 17741b654882SGreg Clayton calculate_format_addr_function_offset = true; 17751b654882SGreg Clayton } 17761b654882SGreg Clayton } 17771b654882SGreg Clayton else if (::strncmp (var_name_begin, "line-offset}", strlen("line-offset}")) == 0) 17781b654882SGreg Clayton { 17791b654882SGreg Clayton var_success = sc->line_entry.range.GetBaseAddress().IsValid(); 17801b654882SGreg Clayton if (var_success) 17811b654882SGreg Clayton { 17821b654882SGreg Clayton format_addr = sc->line_entry.range.GetBaseAddress(); 17831b654882SGreg Clayton calculate_format_addr_function_offset = true; 17841b654882SGreg Clayton } 17851b654882SGreg Clayton } 17861b654882SGreg Clayton else if (::strncmp (var_name_begin, "pc-offset}", strlen("pc-offset}")) == 0) 17871b654882SGreg Clayton { 1788c14ee32dSGreg Clayton StackFrame *frame = exe_ctx->GetFramePtr(); 1789c14ee32dSGreg Clayton var_success = frame != NULL; 17901b654882SGreg Clayton if (var_success) 17911b654882SGreg Clayton { 1792c14ee32dSGreg Clayton format_addr = frame->GetFrameCodeAddress(); 17931b654882SGreg Clayton calculate_format_addr_function_offset = true; 17941b654882SGreg Clayton } 17951b654882SGreg Clayton } 17961b654882SGreg Clayton } 17971b654882SGreg Clayton } 17981b654882SGreg Clayton break; 17991b654882SGreg Clayton 18001b654882SGreg Clayton case 'l': 18011b654882SGreg Clayton if (::strncmp (var_name_begin, "line.", strlen("line.")) == 0) 18021b654882SGreg Clayton { 18031b654882SGreg Clayton if (sc && sc->line_entry.IsValid()) 18041b654882SGreg Clayton { 18051b654882SGreg Clayton var_name_begin += ::strlen ("line."); 18061b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 18071b654882SGreg Clayton { 18081b654882SGreg Clayton var_name_begin += ::strlen ("file."); 18091b654882SGreg Clayton 18101b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 18111b654882SGreg Clayton { 18121b654882SGreg Clayton format_file_spec.GetFilename() = sc->line_entry.file.GetFilename(); 18131b654882SGreg Clayton var_success = format_file_spec; 18141b654882SGreg Clayton } 18151b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 18161b654882SGreg Clayton { 18171b654882SGreg Clayton format_file_spec = sc->line_entry.file; 18181b654882SGreg Clayton var_success = format_file_spec; 18191b654882SGreg Clayton } 18201b654882SGreg Clayton } 18211b654882SGreg Clayton else if (::strncmp (var_name_begin, "number}", strlen("number}")) == 0) 18221b654882SGreg Clayton { 18231b654882SGreg Clayton var_success = true; 18241b654882SGreg Clayton s.Printf("%u", sc->line_entry.line); 18251b654882SGreg Clayton } 18261b654882SGreg Clayton else if ((::strncmp (var_name_begin, "start-addr}", strlen("start-addr}")) == 0) || 18271b654882SGreg Clayton (::strncmp (var_name_begin, "end-addr}", strlen("end-addr}")) == 0)) 18281b654882SGreg Clayton { 18291b654882SGreg Clayton var_success = sc && sc->line_entry.range.GetBaseAddress().IsValid(); 18301b654882SGreg Clayton if (var_success) 18311b654882SGreg Clayton { 18321b654882SGreg Clayton format_addr = sc->line_entry.range.GetBaseAddress(); 18331b654882SGreg Clayton if (var_name_begin[0] == 'e') 18341b654882SGreg Clayton format_addr.Slide (sc->line_entry.range.GetByteSize()); 18351b654882SGreg Clayton } 18361b654882SGreg Clayton } 18371b654882SGreg Clayton } 18381b654882SGreg Clayton } 18391b654882SGreg Clayton break; 18401b654882SGreg Clayton } 18411b654882SGreg Clayton 18421b654882SGreg Clayton if (var_success) 18431b654882SGreg Clayton { 18441b654882SGreg Clayton // If format addr is valid, then we need to print an address 18451b654882SGreg Clayton if (reg_num != LLDB_INVALID_REGNUM) 18461b654882SGreg Clayton { 1847c14ee32dSGreg Clayton StackFrame *frame = exe_ctx->GetFramePtr(); 18481b654882SGreg Clayton // We have a register value to display... 18491b654882SGreg Clayton if (reg_num == LLDB_REGNUM_GENERIC_PC && reg_kind == eRegisterKindGeneric) 18501b654882SGreg Clayton { 1851c14ee32dSGreg Clayton format_addr = frame->GetFrameCodeAddress(); 18521b654882SGreg Clayton } 18531b654882SGreg Clayton else 18541b654882SGreg Clayton { 18551b654882SGreg Clayton if (reg_ctx == NULL) 1856c14ee32dSGreg Clayton reg_ctx = frame->GetRegisterContext().get(); 18571b654882SGreg Clayton 18581b654882SGreg Clayton if (reg_ctx) 18591b654882SGreg Clayton { 18601b654882SGreg Clayton if (reg_kind != kNumRegisterKinds) 18611b654882SGreg Clayton reg_num = reg_ctx->ConvertRegisterKindToRegisterNumber(reg_kind, reg_num); 18621b654882SGreg Clayton reg_info = reg_ctx->GetRegisterInfoAtIndex (reg_num); 18631b654882SGreg Clayton var_success = reg_info != NULL; 18641b654882SGreg Clayton } 18651b654882SGreg Clayton } 18661b654882SGreg Clayton } 18671b654882SGreg Clayton 18681b654882SGreg Clayton if (reg_info != NULL) 18691b654882SGreg Clayton { 18707349bd90SGreg Clayton RegisterValue reg_value; 18717349bd90SGreg Clayton var_success = reg_ctx->ReadRegister (reg_info, reg_value); 18727349bd90SGreg Clayton if (var_success) 18731b654882SGreg Clayton { 18749a8fa916SGreg Clayton reg_value.Dump(&s, reg_info, false, false, eFormatDefault); 18751b654882SGreg Clayton } 18761b654882SGreg Clayton } 18771b654882SGreg Clayton 18781b654882SGreg Clayton if (format_file_spec) 18791b654882SGreg Clayton { 18801b654882SGreg Clayton s << format_file_spec; 18811b654882SGreg Clayton } 18821b654882SGreg Clayton 18831b654882SGreg Clayton // If format addr is valid, then we need to print an address 18841b654882SGreg Clayton if (format_addr.IsValid()) 18851b654882SGreg Clayton { 18860603aa9dSGreg Clayton var_success = false; 18870603aa9dSGreg Clayton 18881b654882SGreg Clayton if (calculate_format_addr_function_offset) 18891b654882SGreg Clayton { 18901b654882SGreg Clayton Address func_addr; 18910603aa9dSGreg Clayton 18920603aa9dSGreg Clayton if (sc) 18930603aa9dSGreg Clayton { 18941b654882SGreg Clayton if (sc->function) 18950d9c9934SGreg Clayton { 18961b654882SGreg Clayton func_addr = sc->function->GetAddressRange().GetBaseAddress(); 18970d9c9934SGreg Clayton if (sc->block) 18980d9c9934SGreg Clayton { 18990d9c9934SGreg Clayton // Check to make sure we aren't in an inline 19000d9c9934SGreg Clayton // function. If we are, use the inline block 19010d9c9934SGreg Clayton // range that contains "format_addr" since 19020d9c9934SGreg Clayton // blocks can be discontiguous. 19030d9c9934SGreg Clayton Block *inline_block = sc->block->GetContainingInlinedBlock (); 19040d9c9934SGreg Clayton AddressRange inline_range; 19050d9c9934SGreg Clayton if (inline_block && inline_block->GetRangeContainingAddress (format_addr, inline_range)) 19060d9c9934SGreg Clayton func_addr = inline_range.GetBaseAddress(); 19070d9c9934SGreg Clayton } 19080d9c9934SGreg Clayton } 19091b654882SGreg Clayton else if (sc->symbol && sc->symbol->GetAddressRangePtr()) 19101b654882SGreg Clayton func_addr = sc->symbol->GetAddressRangePtr()->GetBaseAddress(); 19110603aa9dSGreg Clayton } 19121b654882SGreg Clayton 19130603aa9dSGreg Clayton if (func_addr.IsValid()) 19141b654882SGreg Clayton { 19151b654882SGreg Clayton if (func_addr.GetSection() == format_addr.GetSection()) 19161b654882SGreg Clayton { 19171b654882SGreg Clayton addr_t func_file_addr = func_addr.GetFileAddress(); 19181b654882SGreg Clayton addr_t addr_file_addr = format_addr.GetFileAddress(); 19191b654882SGreg Clayton if (addr_file_addr > func_file_addr) 19201b654882SGreg Clayton s.Printf(" + %llu", addr_file_addr - func_file_addr); 19211b654882SGreg Clayton else if (addr_file_addr < func_file_addr) 19221b654882SGreg Clayton s.Printf(" - %llu", func_file_addr - addr_file_addr); 19230603aa9dSGreg Clayton var_success = true; 19241b654882SGreg Clayton } 19251b654882SGreg Clayton else 19260603aa9dSGreg Clayton { 19270603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 19280603aa9dSGreg Clayton if (target) 19290603aa9dSGreg Clayton { 19300603aa9dSGreg Clayton addr_t func_load_addr = func_addr.GetLoadAddress (target); 19310603aa9dSGreg Clayton addr_t addr_load_addr = format_addr.GetLoadAddress (target); 19320603aa9dSGreg Clayton if (addr_load_addr > func_load_addr) 19330603aa9dSGreg Clayton s.Printf(" + %llu", addr_load_addr - func_load_addr); 19340603aa9dSGreg Clayton else if (addr_load_addr < func_load_addr) 19350603aa9dSGreg Clayton s.Printf(" - %llu", func_load_addr - addr_load_addr); 19360603aa9dSGreg Clayton var_success = true; 19370603aa9dSGreg Clayton } 19380603aa9dSGreg Clayton } 19391b654882SGreg Clayton } 19401b654882SGreg Clayton } 19411b654882SGreg Clayton else 19421b654882SGreg Clayton { 19430603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 19441b654882SGreg Clayton addr_t vaddr = LLDB_INVALID_ADDRESS; 19450603aa9dSGreg Clayton if (exe_ctx && !target->GetSectionLoadList().IsEmpty()) 19460603aa9dSGreg Clayton vaddr = format_addr.GetLoadAddress (target); 19471b654882SGreg Clayton if (vaddr == LLDB_INVALID_ADDRESS) 19481b654882SGreg Clayton vaddr = format_addr.GetFileAddress (); 19491b654882SGreg Clayton 19501b654882SGreg Clayton if (vaddr != LLDB_INVALID_ADDRESS) 19510603aa9dSGreg Clayton { 1952514487e8SGreg Clayton int addr_width = target->GetArchitecture().GetAddressByteSize() * 2; 195335f1a0d5SGreg Clayton if (addr_width == 0) 195435f1a0d5SGreg Clayton addr_width = 16; 195535f1a0d5SGreg Clayton s.Printf("0x%*.*llx", addr_width, addr_width, vaddr); 19560603aa9dSGreg Clayton var_success = true; 19570603aa9dSGreg Clayton } 19581b654882SGreg Clayton } 19591b654882SGreg Clayton } 19601b654882SGreg Clayton } 19611b654882SGreg Clayton 19621b654882SGreg Clayton if (var_success == false) 19631b654882SGreg Clayton success = false; 19641b654882SGreg Clayton } 19651b654882SGreg Clayton p = var_name_end; 19661b654882SGreg Clayton } 19671b654882SGreg Clayton else 19681b654882SGreg Clayton break; 19691b654882SGreg Clayton } 19701b654882SGreg Clayton else 19711b654882SGreg Clayton { 19721b654882SGreg Clayton // We got a dollar sign with no '{' after it, it must just be a dollar sign 19731b654882SGreg Clayton s.PutChar(*p); 19741b654882SGreg Clayton } 19751b654882SGreg Clayton } 19761b654882SGreg Clayton else if (*p == '\\') 19771b654882SGreg Clayton { 19781b654882SGreg Clayton ++p; // skip the slash 19791b654882SGreg Clayton switch (*p) 19801b654882SGreg Clayton { 19811b654882SGreg Clayton case 'a': s.PutChar ('\a'); break; 19821b654882SGreg Clayton case 'b': s.PutChar ('\b'); break; 19831b654882SGreg Clayton case 'f': s.PutChar ('\f'); break; 19841b654882SGreg Clayton case 'n': s.PutChar ('\n'); break; 19851b654882SGreg Clayton case 'r': s.PutChar ('\r'); break; 19861b654882SGreg Clayton case 't': s.PutChar ('\t'); break; 19871b654882SGreg Clayton case 'v': s.PutChar ('\v'); break; 19881b654882SGreg Clayton case '\'': s.PutChar ('\''); break; 19891b654882SGreg Clayton case '\\': s.PutChar ('\\'); break; 19901b654882SGreg Clayton case '0': 19911b654882SGreg Clayton // 1 to 3 octal chars 19921b654882SGreg Clayton { 19930603aa9dSGreg Clayton // Make a string that can hold onto the initial zero char, 19940603aa9dSGreg Clayton // up to 3 octal digits, and a terminating NULL. 19950603aa9dSGreg Clayton char oct_str[5] = { 0, 0, 0, 0, 0 }; 19960603aa9dSGreg Clayton 19970603aa9dSGreg Clayton int i; 19980603aa9dSGreg Clayton for (i=0; (p[i] >= '0' && p[i] <= '7') && i<4; ++i) 19990603aa9dSGreg Clayton oct_str[i] = p[i]; 20000603aa9dSGreg Clayton 20010603aa9dSGreg Clayton // We don't want to consume the last octal character since 20020603aa9dSGreg Clayton // the main for loop will do this for us, so we advance p by 20030603aa9dSGreg Clayton // one less than i (even if i is zero) 20040603aa9dSGreg Clayton p += i - 1; 20050603aa9dSGreg Clayton unsigned long octal_value = ::strtoul (oct_str, NULL, 8); 20060603aa9dSGreg Clayton if (octal_value <= UINT8_MAX) 20071b654882SGreg Clayton { 20080603aa9dSGreg Clayton char octal_char = octal_value; 20090603aa9dSGreg Clayton s.Write (&octal_char, 1); 20101b654882SGreg Clayton } 20111b654882SGreg Clayton } 20121b654882SGreg Clayton break; 20131b654882SGreg Clayton 20141b654882SGreg Clayton case 'x': 20151b654882SGreg Clayton // hex number in the format 20160603aa9dSGreg Clayton if (isxdigit(p[1])) 20171b654882SGreg Clayton { 20180603aa9dSGreg Clayton ++p; // Skip the 'x' 20191b654882SGreg Clayton 20200603aa9dSGreg Clayton // Make a string that can hold onto two hex chars plus a 20210603aa9dSGreg Clayton // NULL terminator 20221b654882SGreg Clayton char hex_str[3] = { 0,0,0 }; 20231b654882SGreg Clayton hex_str[0] = *p; 20240603aa9dSGreg Clayton if (isxdigit(p[1])) 20250603aa9dSGreg Clayton { 20260603aa9dSGreg Clayton ++p; // Skip the first of the two hex chars 20271b654882SGreg Clayton hex_str[1] = *p; 20280603aa9dSGreg Clayton } 20290603aa9dSGreg Clayton 20301b654882SGreg Clayton unsigned long hex_value = strtoul (hex_str, NULL, 16); 20310603aa9dSGreg Clayton if (hex_value <= UINT8_MAX) 20321b654882SGreg Clayton s.PutChar (hex_value); 20331b654882SGreg Clayton } 20341b654882SGreg Clayton else 20351b654882SGreg Clayton { 20360603aa9dSGreg Clayton s.PutChar('x'); 20371b654882SGreg Clayton } 20381b654882SGreg Clayton break; 20391b654882SGreg Clayton 20401b654882SGreg Clayton default: 20410603aa9dSGreg Clayton // Just desensitize any other character by just printing what 20420603aa9dSGreg Clayton // came after the '\' 20430603aa9dSGreg Clayton s << *p; 20441b654882SGreg Clayton break; 20451b654882SGreg Clayton 20461b654882SGreg Clayton } 20471b654882SGreg Clayton 20481b654882SGreg Clayton } 20491b654882SGreg Clayton } 20501b654882SGreg Clayton if (end) 20511b654882SGreg Clayton *end = p; 20521b654882SGreg Clayton return success; 20531b654882SGreg Clayton } 20541b654882SGreg Clayton 20551b654882SGreg Clayton #pragma mark Debugger::SettingsController 20561b654882SGreg Clayton 20573df9a8dfSCaroline Tice //-------------------------------------------------- 20581b654882SGreg Clayton // class Debugger::SettingsController 20593df9a8dfSCaroline Tice //-------------------------------------------------- 20603df9a8dfSCaroline Tice 20611b654882SGreg Clayton Debugger::SettingsController::SettingsController () : 20624d122c40SGreg Clayton UserSettingsController ("", UserSettingsControllerSP()) 20633df9a8dfSCaroline Tice { 206491123da2SCaroline Tice m_default_settings.reset (new DebuggerInstanceSettings (*this, false, 206591123da2SCaroline Tice InstanceSettings::GetDefaultName().AsCString())); 20663df9a8dfSCaroline Tice } 20673df9a8dfSCaroline Tice 20681b654882SGreg Clayton Debugger::SettingsController::~SettingsController () 20693df9a8dfSCaroline Tice { 20703df9a8dfSCaroline Tice } 20713df9a8dfSCaroline Tice 20723df9a8dfSCaroline Tice 20734d122c40SGreg Clayton InstanceSettingsSP 20741b654882SGreg Clayton Debugger::SettingsController::CreateInstanceSettings (const char *instance_name) 20753df9a8dfSCaroline Tice { 2076dbe54508SGreg Clayton DebuggerInstanceSettings *new_settings = new DebuggerInstanceSettings (*GetSettingsController(), 207791123da2SCaroline Tice false, instance_name); 20784d122c40SGreg Clayton InstanceSettingsSP new_settings_sp (new_settings); 20793df9a8dfSCaroline Tice return new_settings_sp; 20803df9a8dfSCaroline Tice } 20813df9a8dfSCaroline Tice 20821b654882SGreg Clayton #pragma mark DebuggerInstanceSettings 20833df9a8dfSCaroline Tice //-------------------------------------------------- 20843df9a8dfSCaroline Tice // class DebuggerInstanceSettings 20853df9a8dfSCaroline Tice //-------------------------------------------------- 20863df9a8dfSCaroline Tice 2087a7015092SGreg Clayton DebuggerInstanceSettings::DebuggerInstanceSettings 2088a7015092SGreg Clayton ( 2089a7015092SGreg Clayton UserSettingsController &owner, 2090a7015092SGreg Clayton bool live_instance, 2091a7015092SGreg Clayton const char *name 2092a7015092SGreg Clayton ) : 209385851ddeSGreg Clayton InstanceSettings (owner, name ? name : InstanceSettings::InvalidName().AsCString(), live_instance), 2094a7015092SGreg Clayton m_term_width (80), 20953df9a8dfSCaroline Tice m_prompt (), 20960603aa9dSGreg Clayton m_frame_format (), 20970603aa9dSGreg Clayton m_thread_format (), 2098daccaa9eSCaroline Tice m_script_lang (), 20993bcdb29cSJim Ingham m_use_external_editor (false), 21003bcdb29cSJim Ingham m_auto_confirm_on (false) 21013df9a8dfSCaroline Tice { 2102f20e8239SCaroline Tice // CopyInstanceSettings is a pure virtual function in InstanceSettings; it therefore cannot be called 2103f20e8239SCaroline Tice // until the vtables for DebuggerInstanceSettings are properly set up, i.e. AFTER all the initializers. 2104f20e8239SCaroline Tice // For this reason it has to be called here, rather than in the initializer or in the parent constructor. 21059e41c15dSCaroline Tice // The same is true of CreateInstanceName(). 21069e41c15dSCaroline Tice 21079e41c15dSCaroline Tice if (GetInstanceName() == InstanceSettings::InvalidName()) 21089e41c15dSCaroline Tice { 21099e41c15dSCaroline Tice ChangeInstanceName (std::string (CreateInstanceName().AsCString())); 21109e41c15dSCaroline Tice m_owner.RegisterInstanceSettings (this); 21119e41c15dSCaroline Tice } 2112f20e8239SCaroline Tice 2113f20e8239SCaroline Tice if (live_instance) 21143df9a8dfSCaroline Tice { 21154d122c40SGreg Clayton const InstanceSettingsSP &pending_settings = m_owner.FindPendingSettings (m_instance_name); 21163df9a8dfSCaroline Tice CopyInstanceSettings (pending_settings, false); 21173df9a8dfSCaroline Tice } 21183df9a8dfSCaroline Tice } 21193df9a8dfSCaroline Tice 21203df9a8dfSCaroline Tice DebuggerInstanceSettings::DebuggerInstanceSettings (const DebuggerInstanceSettings &rhs) : 212199d0faf2SGreg Clayton InstanceSettings (*Debugger::GetSettingsController(), CreateInstanceName ().AsCString()), 21223df9a8dfSCaroline Tice m_prompt (rhs.m_prompt), 21230603aa9dSGreg Clayton m_frame_format (rhs.m_frame_format), 21240603aa9dSGreg Clayton m_thread_format (rhs.m_thread_format), 2125daccaa9eSCaroline Tice m_script_lang (rhs.m_script_lang), 21263bcdb29cSJim Ingham m_use_external_editor (rhs.m_use_external_editor), 21273bcdb29cSJim Ingham m_auto_confirm_on(rhs.m_auto_confirm_on) 21283df9a8dfSCaroline Tice { 21294d122c40SGreg Clayton const InstanceSettingsSP &pending_settings = m_owner.FindPendingSettings (m_instance_name); 21303df9a8dfSCaroline Tice CopyInstanceSettings (pending_settings, false); 21313df9a8dfSCaroline Tice m_owner.RemovePendingSettings (m_instance_name); 21323df9a8dfSCaroline Tice } 21333df9a8dfSCaroline Tice 21343df9a8dfSCaroline Tice DebuggerInstanceSettings::~DebuggerInstanceSettings () 21353df9a8dfSCaroline Tice { 21363df9a8dfSCaroline Tice } 21373df9a8dfSCaroline Tice 21383df9a8dfSCaroline Tice DebuggerInstanceSettings& 21393df9a8dfSCaroline Tice DebuggerInstanceSettings::operator= (const DebuggerInstanceSettings &rhs) 21403df9a8dfSCaroline Tice { 21413df9a8dfSCaroline Tice if (this != &rhs) 21423df9a8dfSCaroline Tice { 21431b654882SGreg Clayton m_term_width = rhs.m_term_width; 21443df9a8dfSCaroline Tice m_prompt = rhs.m_prompt; 21450603aa9dSGreg Clayton m_frame_format = rhs.m_frame_format; 21460603aa9dSGreg Clayton m_thread_format = rhs.m_thread_format; 21473df9a8dfSCaroline Tice m_script_lang = rhs.m_script_lang; 2148daccaa9eSCaroline Tice m_use_external_editor = rhs.m_use_external_editor; 21493bcdb29cSJim Ingham m_auto_confirm_on = rhs.m_auto_confirm_on; 21503df9a8dfSCaroline Tice } 21513df9a8dfSCaroline Tice 21523df9a8dfSCaroline Tice return *this; 21533df9a8dfSCaroline Tice } 21543df9a8dfSCaroline Tice 21551b654882SGreg Clayton bool 21561b654882SGreg Clayton DebuggerInstanceSettings::ValidTermWidthValue (const char *value, Error err) 21571b654882SGreg Clayton { 21581b654882SGreg Clayton bool valid = false; 21591b654882SGreg Clayton 21601b654882SGreg Clayton // Verify we have a value string. 21611b654882SGreg Clayton if (value == NULL || value[0] == '\0') 21621b654882SGreg Clayton { 2163*86edbf41SGreg Clayton err.SetErrorString ("missing value, can't set terminal width without a value"); 21641b654882SGreg Clayton } 21651b654882SGreg Clayton else 21661b654882SGreg Clayton { 21671b654882SGreg Clayton char *end = NULL; 21681b654882SGreg Clayton const uint32_t width = ::strtoul (value, &end, 0); 21691b654882SGreg Clayton 2170ea9fc181SJohnny Chen if (end && end[0] == '\0') 21711b654882SGreg Clayton { 2172433d7741SJohnny Chen if (width >= 10 && width <= 1024) 21731b654882SGreg Clayton valid = true; 21741b654882SGreg Clayton else 2175*86edbf41SGreg Clayton err.SetErrorString ("invalid term-width value; value must be between 10 and 1024"); 21761b654882SGreg Clayton } 21771b654882SGreg Clayton else 2178*86edbf41SGreg Clayton err.SetErrorStringWithFormat ("'%s' is not a valid unsigned integer string", value); 21791b654882SGreg Clayton } 21801b654882SGreg Clayton 21811b654882SGreg Clayton return valid; 21821b654882SGreg Clayton } 21831b654882SGreg Clayton 21841b654882SGreg Clayton 21853df9a8dfSCaroline Tice void 21863df9a8dfSCaroline Tice DebuggerInstanceSettings::UpdateInstanceSettingsVariable (const ConstString &var_name, 21873df9a8dfSCaroline Tice const char *index_value, 21883df9a8dfSCaroline Tice const char *value, 21893df9a8dfSCaroline Tice const ConstString &instance_name, 21903df9a8dfSCaroline Tice const SettingEntry &entry, 2191e0d378b3SGreg Clayton VarSetOperationType op, 21923df9a8dfSCaroline Tice Error &err, 21933df9a8dfSCaroline Tice bool pending) 21943df9a8dfSCaroline Tice { 21950603aa9dSGreg Clayton 21960603aa9dSGreg Clayton if (var_name == TermWidthVarName()) 21970603aa9dSGreg Clayton { 21980603aa9dSGreg Clayton if (ValidTermWidthValue (value, err)) 21990603aa9dSGreg Clayton { 22000603aa9dSGreg Clayton m_term_width = ::strtoul (value, NULL, 0); 22010603aa9dSGreg Clayton } 22020603aa9dSGreg Clayton } 22030603aa9dSGreg Clayton else if (var_name == PromptVarName()) 22043df9a8dfSCaroline Tice { 22053df9a8dfSCaroline Tice UserSettingsController::UpdateStringVariable (op, m_prompt, value, err); 22063df9a8dfSCaroline Tice if (!pending) 22073df9a8dfSCaroline Tice { 220849e2737eSCaroline Tice // 'instance_name' is actually (probably) in the form '[<instance_name>]'; if so, we need to 220949e2737eSCaroline Tice // strip off the brackets before passing it to BroadcastPromptChange. 221049e2737eSCaroline Tice 221149e2737eSCaroline Tice std::string tmp_instance_name (instance_name.AsCString()); 221249e2737eSCaroline Tice if ((tmp_instance_name[0] == '[') 221349e2737eSCaroline Tice && (tmp_instance_name[instance_name.GetLength() - 1] == ']')) 221449e2737eSCaroline Tice tmp_instance_name = tmp_instance_name.substr (1, instance_name.GetLength() - 2); 221549e2737eSCaroline Tice ConstString new_name (tmp_instance_name.c_str()); 221649e2737eSCaroline Tice 221749e2737eSCaroline Tice BroadcastPromptChange (new_name, m_prompt.c_str()); 22183df9a8dfSCaroline Tice } 22193df9a8dfSCaroline Tice } 22200603aa9dSGreg Clayton else if (var_name == GetFrameFormatName()) 22210603aa9dSGreg Clayton { 22220603aa9dSGreg Clayton UserSettingsController::UpdateStringVariable (op, m_frame_format, value, err); 22230603aa9dSGreg Clayton } 22240603aa9dSGreg Clayton else if (var_name == GetThreadFormatName()) 22250603aa9dSGreg Clayton { 22260603aa9dSGreg Clayton UserSettingsController::UpdateStringVariable (op, m_thread_format, value, err); 22270603aa9dSGreg Clayton } 22283df9a8dfSCaroline Tice else if (var_name == ScriptLangVarName()) 22293df9a8dfSCaroline Tice { 22303df9a8dfSCaroline Tice bool success; 22313df9a8dfSCaroline Tice m_script_lang = Args::StringToScriptLanguage (value, eScriptLanguageDefault, 22323df9a8dfSCaroline Tice &success); 22333df9a8dfSCaroline Tice } 2234daccaa9eSCaroline Tice else if (var_name == UseExternalEditorVarName ()) 2235daccaa9eSCaroline Tice { 2236385aa28cSGreg Clayton UserSettingsController::UpdateBooleanVariable (op, m_use_external_editor, value, false, err); 2237daccaa9eSCaroline Tice } 22383bcdb29cSJim Ingham else if (var_name == AutoConfirmName ()) 22393bcdb29cSJim Ingham { 2240385aa28cSGreg Clayton UserSettingsController::UpdateBooleanVariable (op, m_auto_confirm_on, value, false, err); 22413bcdb29cSJim Ingham } 22423df9a8dfSCaroline Tice } 22433df9a8dfSCaroline Tice 224412cecd74SCaroline Tice bool 22453df9a8dfSCaroline Tice DebuggerInstanceSettings::GetInstanceSettingsValue (const SettingEntry &entry, 22463df9a8dfSCaroline Tice const ConstString &var_name, 2247daccaa9eSCaroline Tice StringList &value, 224812cecd74SCaroline Tice Error *err) 22493df9a8dfSCaroline Tice { 22503df9a8dfSCaroline Tice if (var_name == PromptVarName()) 22513df9a8dfSCaroline Tice { 22520603aa9dSGreg Clayton value.AppendString (m_prompt.c_str(), m_prompt.size()); 22533df9a8dfSCaroline Tice 22543df9a8dfSCaroline Tice } 22553df9a8dfSCaroline Tice else if (var_name == ScriptLangVarName()) 22563df9a8dfSCaroline Tice { 22573df9a8dfSCaroline Tice value.AppendString (ScriptInterpreter::LanguageToString (m_script_lang).c_str()); 22583df9a8dfSCaroline Tice } 2259101c7c20SCaroline Tice else if (var_name == TermWidthVarName()) 2260101c7c20SCaroline Tice { 2261101c7c20SCaroline Tice StreamString width_str; 2262101c7c20SCaroline Tice width_str.Printf ("%d", m_term_width); 2263101c7c20SCaroline Tice value.AppendString (width_str.GetData()); 2264101c7c20SCaroline Tice } 22650603aa9dSGreg Clayton else if (var_name == GetFrameFormatName ()) 22660603aa9dSGreg Clayton { 22670603aa9dSGreg Clayton value.AppendString(m_frame_format.c_str(), m_frame_format.size()); 22680603aa9dSGreg Clayton } 22690603aa9dSGreg Clayton else if (var_name == GetThreadFormatName ()) 22700603aa9dSGreg Clayton { 22710603aa9dSGreg Clayton value.AppendString(m_thread_format.c_str(), m_thread_format.size()); 22720603aa9dSGreg Clayton } 2273daccaa9eSCaroline Tice else if (var_name == UseExternalEditorVarName()) 2274daccaa9eSCaroline Tice { 2275daccaa9eSCaroline Tice if (m_use_external_editor) 2276daccaa9eSCaroline Tice value.AppendString ("true"); 2277daccaa9eSCaroline Tice else 2278daccaa9eSCaroline Tice value.AppendString ("false"); 2279daccaa9eSCaroline Tice } 22803bcdb29cSJim Ingham else if (var_name == AutoConfirmName()) 22813bcdb29cSJim Ingham { 22823bcdb29cSJim Ingham if (m_auto_confirm_on) 22833bcdb29cSJim Ingham value.AppendString ("true"); 22843bcdb29cSJim Ingham else 22853bcdb29cSJim Ingham value.AppendString ("false"); 22863bcdb29cSJim Ingham } 2287daccaa9eSCaroline Tice else 228812cecd74SCaroline Tice { 228912cecd74SCaroline Tice if (err) 229012cecd74SCaroline Tice err->SetErrorStringWithFormat ("unrecognized variable name '%s'", var_name.AsCString()); 229112cecd74SCaroline Tice return false; 229212cecd74SCaroline Tice } 229312cecd74SCaroline Tice return true; 22943df9a8dfSCaroline Tice } 22953df9a8dfSCaroline Tice 22963df9a8dfSCaroline Tice void 22974d122c40SGreg Clayton DebuggerInstanceSettings::CopyInstanceSettings (const InstanceSettingsSP &new_settings, 22983df9a8dfSCaroline Tice bool pending) 22993df9a8dfSCaroline Tice { 23003df9a8dfSCaroline Tice if (new_settings.get() == NULL) 23013df9a8dfSCaroline Tice return; 23023df9a8dfSCaroline Tice 23033df9a8dfSCaroline Tice DebuggerInstanceSettings *new_debugger_settings = (DebuggerInstanceSettings *) new_settings.get(); 23043df9a8dfSCaroline Tice 23053df9a8dfSCaroline Tice m_prompt = new_debugger_settings->m_prompt; 23063df9a8dfSCaroline Tice if (!pending) 230749e2737eSCaroline Tice { 230849e2737eSCaroline Tice // 'instance_name' is actually (probably) in the form '[<instance_name>]'; if so, we need to 230949e2737eSCaroline Tice // strip off the brackets before passing it to BroadcastPromptChange. 231049e2737eSCaroline Tice 231149e2737eSCaroline Tice std::string tmp_instance_name (m_instance_name.AsCString()); 231249e2737eSCaroline Tice if ((tmp_instance_name[0] == '[') 231349e2737eSCaroline Tice && (tmp_instance_name[m_instance_name.GetLength() - 1] == ']')) 231449e2737eSCaroline Tice tmp_instance_name = tmp_instance_name.substr (1, m_instance_name.GetLength() - 2); 231549e2737eSCaroline Tice ConstString new_name (tmp_instance_name.c_str()); 231649e2737eSCaroline Tice 231749e2737eSCaroline Tice BroadcastPromptChange (new_name, m_prompt.c_str()); 231849e2737eSCaroline Tice } 23190603aa9dSGreg Clayton m_frame_format = new_debugger_settings->m_frame_format; 23200603aa9dSGreg Clayton m_thread_format = new_debugger_settings->m_thread_format; 2321daccaa9eSCaroline Tice m_term_width = new_debugger_settings->m_term_width; 23223df9a8dfSCaroline Tice m_script_lang = new_debugger_settings->m_script_lang; 2323daccaa9eSCaroline Tice m_use_external_editor = new_debugger_settings->m_use_external_editor; 23243bcdb29cSJim Ingham m_auto_confirm_on = new_debugger_settings->m_auto_confirm_on; 23253df9a8dfSCaroline Tice } 23263df9a8dfSCaroline Tice 23273df9a8dfSCaroline Tice 23283df9a8dfSCaroline Tice bool 23293df9a8dfSCaroline Tice DebuggerInstanceSettings::BroadcastPromptChange (const ConstString &instance_name, const char *new_prompt) 23303df9a8dfSCaroline Tice { 23313df9a8dfSCaroline Tice std::string tmp_prompt; 23323df9a8dfSCaroline Tice 23333df9a8dfSCaroline Tice if (new_prompt != NULL) 23343df9a8dfSCaroline Tice { 23353df9a8dfSCaroline Tice tmp_prompt = new_prompt ; 23363df9a8dfSCaroline Tice int len = tmp_prompt.size(); 23373df9a8dfSCaroline Tice if (len > 1 23383df9a8dfSCaroline Tice && (tmp_prompt[0] == '\'' || tmp_prompt[0] == '"') 23393df9a8dfSCaroline Tice && (tmp_prompt[len-1] == tmp_prompt[0])) 23403df9a8dfSCaroline Tice { 23413df9a8dfSCaroline Tice tmp_prompt = tmp_prompt.substr(1,len-2); 23423df9a8dfSCaroline Tice } 23433df9a8dfSCaroline Tice len = tmp_prompt.size(); 23443df9a8dfSCaroline Tice if (tmp_prompt[len-1] != ' ') 23453df9a8dfSCaroline Tice tmp_prompt.append(" "); 23463df9a8dfSCaroline Tice } 23473df9a8dfSCaroline Tice EventSP new_event_sp; 23483df9a8dfSCaroline Tice new_event_sp.reset (new Event(CommandInterpreter::eBroadcastBitResetPrompt, 23493df9a8dfSCaroline Tice new EventDataBytes (tmp_prompt.c_str()))); 23503df9a8dfSCaroline Tice 23513df9a8dfSCaroline Tice if (instance_name.GetLength() != 0) 23523df9a8dfSCaroline Tice { 23533df9a8dfSCaroline Tice // Set prompt for a particular instance. 23543df9a8dfSCaroline Tice Debugger *dbg = Debugger::FindDebuggerWithInstanceName (instance_name).get(); 23553df9a8dfSCaroline Tice if (dbg != NULL) 23563df9a8dfSCaroline Tice { 23573df9a8dfSCaroline Tice dbg->GetCommandInterpreter().BroadcastEvent (new_event_sp); 23583df9a8dfSCaroline Tice } 23593df9a8dfSCaroline Tice } 23603df9a8dfSCaroline Tice 23613df9a8dfSCaroline Tice return true; 23623df9a8dfSCaroline Tice } 23633df9a8dfSCaroline Tice 23643df9a8dfSCaroline Tice const ConstString 23653df9a8dfSCaroline Tice DebuggerInstanceSettings::CreateInstanceName () 23663df9a8dfSCaroline Tice { 23673df9a8dfSCaroline Tice static int instance_count = 1; 23683df9a8dfSCaroline Tice StreamString sstr; 23693df9a8dfSCaroline Tice 23703df9a8dfSCaroline Tice sstr.Printf ("debugger_%d", instance_count); 23713df9a8dfSCaroline Tice ++instance_count; 23723df9a8dfSCaroline Tice 23733df9a8dfSCaroline Tice const ConstString ret_val (sstr.GetData()); 23743df9a8dfSCaroline Tice 23753df9a8dfSCaroline Tice return ret_val; 23763df9a8dfSCaroline Tice } 23773df9a8dfSCaroline Tice 23783df9a8dfSCaroline Tice const ConstString & 23793df9a8dfSCaroline Tice DebuggerInstanceSettings::PromptVarName () 23803df9a8dfSCaroline Tice { 23813df9a8dfSCaroline Tice static ConstString prompt_var_name ("prompt"); 23823df9a8dfSCaroline Tice 23833df9a8dfSCaroline Tice return prompt_var_name; 23843df9a8dfSCaroline Tice } 23853df9a8dfSCaroline Tice 23863df9a8dfSCaroline Tice const ConstString & 23870603aa9dSGreg Clayton DebuggerInstanceSettings::GetFrameFormatName () 23880603aa9dSGreg Clayton { 23890603aa9dSGreg Clayton static ConstString prompt_var_name ("frame-format"); 23900603aa9dSGreg Clayton 23910603aa9dSGreg Clayton return prompt_var_name; 23920603aa9dSGreg Clayton } 23930603aa9dSGreg Clayton 23940603aa9dSGreg Clayton const ConstString & 23950603aa9dSGreg Clayton DebuggerInstanceSettings::GetThreadFormatName () 23960603aa9dSGreg Clayton { 23970603aa9dSGreg Clayton static ConstString prompt_var_name ("thread-format"); 23980603aa9dSGreg Clayton 23990603aa9dSGreg Clayton return prompt_var_name; 24000603aa9dSGreg Clayton } 24010603aa9dSGreg Clayton 24020603aa9dSGreg Clayton const ConstString & 24033df9a8dfSCaroline Tice DebuggerInstanceSettings::ScriptLangVarName () 24043df9a8dfSCaroline Tice { 24053df9a8dfSCaroline Tice static ConstString script_lang_var_name ("script-lang"); 24063df9a8dfSCaroline Tice 24073df9a8dfSCaroline Tice return script_lang_var_name; 24083df9a8dfSCaroline Tice } 24093df9a8dfSCaroline Tice 2410101c7c20SCaroline Tice const ConstString & 2411101c7c20SCaroline Tice DebuggerInstanceSettings::TermWidthVarName () 2412101c7c20SCaroline Tice { 2413101c7c20SCaroline Tice static ConstString term_width_var_name ("term-width"); 2414101c7c20SCaroline Tice 2415101c7c20SCaroline Tice return term_width_var_name; 2416101c7c20SCaroline Tice } 2417101c7c20SCaroline Tice 2418daccaa9eSCaroline Tice const ConstString & 2419daccaa9eSCaroline Tice DebuggerInstanceSettings::UseExternalEditorVarName () 2420daccaa9eSCaroline Tice { 2421daccaa9eSCaroline Tice static ConstString use_external_editor_var_name ("use-external-editor"); 2422daccaa9eSCaroline Tice 2423daccaa9eSCaroline Tice return use_external_editor_var_name; 2424daccaa9eSCaroline Tice } 2425daccaa9eSCaroline Tice 24263bcdb29cSJim Ingham const ConstString & 24273bcdb29cSJim Ingham DebuggerInstanceSettings::AutoConfirmName () 24283bcdb29cSJim Ingham { 24293bcdb29cSJim Ingham static ConstString use_external_editor_var_name ("auto-confirm"); 24303bcdb29cSJim Ingham 24313bcdb29cSJim Ingham return use_external_editor_var_name; 24323bcdb29cSJim Ingham } 24333bcdb29cSJim Ingham 24343df9a8dfSCaroline Tice //-------------------------------------------------- 24351b654882SGreg Clayton // SettingsController Variable Tables 24363df9a8dfSCaroline Tice //-------------------------------------------------- 24373df9a8dfSCaroline Tice 24383df9a8dfSCaroline Tice 24393df9a8dfSCaroline Tice SettingEntry 24401b654882SGreg Clayton Debugger::SettingsController::global_settings_table[] = 24413df9a8dfSCaroline Tice { 24423df9a8dfSCaroline Tice //{ "var-name", var-type, "default", enum-table, init'd, hidden, "help-text"}, 2443101c7c20SCaroline Tice // The Debugger level global table should always be empty; all Debugger settable variables should be instance 2444101c7c20SCaroline Tice // variables. 24453df9a8dfSCaroline Tice { NULL, eSetVarTypeNone, NULL, NULL, 0, 0, NULL } 24463df9a8dfSCaroline Tice }; 24473df9a8dfSCaroline Tice 2448bb562b13SGreg Clayton #define MODULE_WITH_FUNC "{ ${module.file.basename}{`${function.name}${function.pc-offset}}}" 24490603aa9dSGreg Clayton #define FILE_AND_LINE "{ at ${line.file.basename}:${line.number}}" 24503df9a8dfSCaroline Tice 24510603aa9dSGreg Clayton #define DEFAULT_THREAD_FORMAT "thread #${thread.index}: tid = ${thread.id}"\ 24520603aa9dSGreg Clayton "{, ${frame.pc}}"\ 24530603aa9dSGreg Clayton MODULE_WITH_FUNC\ 2454cf4b9078SGreg Clayton FILE_AND_LINE\ 24550603aa9dSGreg Clayton "{, stop reason = ${thread.stop-reason}}"\ 24560603aa9dSGreg Clayton "\\n" 24570603aa9dSGreg Clayton 2458315d2cabSGreg Clayton //#define DEFAULT_THREAD_FORMAT "thread #${thread.index}: tid = ${thread.id}"\ 2459315d2cabSGreg Clayton // "{, ${frame.pc}}"\ 2460315d2cabSGreg Clayton // MODULE_WITH_FUNC\ 2461315d2cabSGreg Clayton // FILE_AND_LINE\ 2462315d2cabSGreg Clayton // "{, stop reason = ${thread.stop-reason}}"\ 2463315d2cabSGreg Clayton // "{, name = ${thread.name}}"\ 2464315d2cabSGreg Clayton // "{, queue = ${thread.queue}}"\ 2465315d2cabSGreg Clayton // "\\n" 2466315d2cabSGreg Clayton 24670603aa9dSGreg Clayton #define DEFAULT_FRAME_FORMAT "frame #${frame.index}: ${frame.pc}"\ 24680603aa9dSGreg Clayton MODULE_WITH_FUNC\ 24690603aa9dSGreg Clayton FILE_AND_LINE\ 24700603aa9dSGreg Clayton "\\n" 24713df9a8dfSCaroline Tice 24723df9a8dfSCaroline Tice SettingEntry 24731b654882SGreg Clayton Debugger::SettingsController::instance_settings_table[] = 24743df9a8dfSCaroline Tice { 24750603aa9dSGreg Clayton // NAME Setting variable type Default Enum Init'd Hidden Help 24760603aa9dSGreg Clayton // ======================= ======================= ====================== ==== ====== ====== ====================== 24770603aa9dSGreg Clayton { "frame-format", eSetVarTypeString, DEFAULT_FRAME_FORMAT, NULL, false, false, "The default frame format string to use when displaying thread information." }, 24783bcdb29cSJim Ingham { "prompt", eSetVarTypeString, "(lldb) ", NULL, false, false, "The debugger command line prompt displayed for the user." }, 24793bcdb29cSJim Ingham { "script-lang", eSetVarTypeString, "python", NULL, false, false, "The script language to be used for evaluating user-written scripts." }, 24803bcdb29cSJim Ingham { "term-width", eSetVarTypeInt, "80" , NULL, false, false, "The maximum number of columns to use for displaying text." }, 24810603aa9dSGreg Clayton { "thread-format", eSetVarTypeString, DEFAULT_THREAD_FORMAT, NULL, false, false, "The default thread format string to use when displaying thread information." }, 248206e827ccSJim Ingham { "use-external-editor", eSetVarTypeBoolean, "false", NULL, false, false, "Whether to use an external editor or not." }, 248306e827ccSJim Ingham { "auto-confirm", eSetVarTypeBoolean, "false", NULL, false, false, "If true all confirmation prompts will receive their default reply." }, 24840603aa9dSGreg Clayton { NULL, eSetVarTypeNone, NULL, NULL, false, false, NULL } 24853df9a8dfSCaroline Tice }; 2486