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" 3430fdc8d8SChris Lattner 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 155e02657b1SCaroline Tice Debugger::Destroy (lldb::DebuggerSP &debugger_sp) 156e02657b1SCaroline Tice { 157e02657b1SCaroline Tice if (debugger_sp.get() == NULL) 158e02657b1SCaroline Tice return; 159e02657b1SCaroline Tice 160e02657b1SCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 161e02657b1SCaroline Tice DebuggerList &debugger_list = GetDebuggerList (); 162e02657b1SCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 163e02657b1SCaroline Tice for (pos = debugger_list.begin (); pos != end; ++pos) 164e02657b1SCaroline Tice { 165e02657b1SCaroline Tice if ((*pos).get() == debugger_sp.get()) 166e02657b1SCaroline Tice { 167e02657b1SCaroline Tice debugger_list.erase (pos); 168e02657b1SCaroline Tice return; 169e02657b1SCaroline Tice } 170e02657b1SCaroline Tice } 171e02657b1SCaroline Tice 172e02657b1SCaroline Tice } 173e02657b1SCaroline Tice 1746611103cSGreg Clayton lldb::DebuggerSP 1756611103cSGreg Clayton Debugger::GetSP () 1766611103cSGreg Clayton { 1776611103cSGreg Clayton lldb::DebuggerSP debugger_sp; 1786611103cSGreg Clayton 1796611103cSGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 1806611103cSGreg Clayton DebuggerList &debugger_list = GetDebuggerList(); 1816611103cSGreg Clayton DebuggerList::iterator pos, end = debugger_list.end(); 1826611103cSGreg Clayton for (pos = debugger_list.begin(); pos != end; ++pos) 1836611103cSGreg Clayton { 1846611103cSGreg Clayton if ((*pos).get() == this) 1856611103cSGreg Clayton { 1866611103cSGreg Clayton debugger_sp = *pos; 1876611103cSGreg Clayton break; 1886611103cSGreg Clayton } 1896611103cSGreg Clayton } 1906611103cSGreg Clayton return debugger_sp; 1916611103cSGreg Clayton } 1926611103cSGreg Clayton 1933df9a8dfSCaroline Tice lldb::DebuggerSP 1943df9a8dfSCaroline Tice Debugger::FindDebuggerWithInstanceName (const ConstString &instance_name) 1953df9a8dfSCaroline Tice { 1963df9a8dfSCaroline Tice lldb::DebuggerSP debugger_sp; 1973df9a8dfSCaroline Tice 1983df9a8dfSCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 1993df9a8dfSCaroline Tice DebuggerList &debugger_list = GetDebuggerList(); 2003df9a8dfSCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 2013df9a8dfSCaroline Tice 2023df9a8dfSCaroline Tice for (pos = debugger_list.begin(); pos != end; ++pos) 2033df9a8dfSCaroline Tice { 2043df9a8dfSCaroline Tice if ((*pos).get()->m_instance_name == instance_name) 2053df9a8dfSCaroline Tice { 2063df9a8dfSCaroline Tice debugger_sp = *pos; 2073df9a8dfSCaroline Tice break; 2083df9a8dfSCaroline Tice } 2093df9a8dfSCaroline Tice } 2103df9a8dfSCaroline Tice return debugger_sp; 2113df9a8dfSCaroline Tice } 2126611103cSGreg Clayton 2136611103cSGreg Clayton TargetSP 2146611103cSGreg Clayton Debugger::FindTargetWithProcessID (lldb::pid_t pid) 2156611103cSGreg Clayton { 2166611103cSGreg Clayton lldb::TargetSP target_sp; 2176611103cSGreg Clayton Mutex::Locker locker (GetDebuggerListMutex ()); 2186611103cSGreg Clayton DebuggerList &debugger_list = GetDebuggerList(); 2196611103cSGreg Clayton DebuggerList::iterator pos, end = debugger_list.end(); 2206611103cSGreg Clayton for (pos = debugger_list.begin(); pos != end; ++pos) 2216611103cSGreg Clayton { 2226611103cSGreg Clayton target_sp = (*pos)->GetTargetList().FindTargetWithProcessID (pid); 2236611103cSGreg Clayton if (target_sp) 2246611103cSGreg Clayton break; 2256611103cSGreg Clayton } 2266611103cSGreg Clayton return target_sp; 2276611103cSGreg Clayton } 2286611103cSGreg Clayton 2296611103cSGreg Clayton 23030fdc8d8SChris Lattner Debugger::Debugger () : 231ebc1bb27SCaroline Tice UserID (g_unique_id++), 232dbe54508SGreg Clayton DebuggerInstanceSettings (*GetSettingsController()), 233d46c87a1SGreg Clayton m_input_comm("debugger.input"), 23430fdc8d8SChris Lattner m_input_file (), 23530fdc8d8SChris Lattner m_output_file (), 23630fdc8d8SChris Lattner m_error_file (), 23730fdc8d8SChris Lattner m_target_list (), 238ded470d3SGreg Clayton m_platform_list (), 23930fdc8d8SChris Lattner m_listener ("lldb.Debugger"), 24030fdc8d8SChris Lattner m_source_manager (), 2416611103cSGreg Clayton m_command_interpreter_ap (new CommandInterpreter (*this, eScriptLanguageDefault, false)), 242d5a0a01bSCaroline Tice m_input_reader_stack (), 2434957bf69SGreg Clayton m_input_reader_data () 24430fdc8d8SChris Lattner { 2456611103cSGreg Clayton m_command_interpreter_ap->Initialize (); 246ded470d3SGreg Clayton // Always add our default platform to the platform list 247ded470d3SGreg Clayton PlatformSP default_platform_sp (Platform::GetDefaultPlatform()); 248ded470d3SGreg Clayton assert (default_platform_sp.get()); 249ded470d3SGreg Clayton m_platform_list.Append (default_platform_sp, true); 25030fdc8d8SChris Lattner } 25130fdc8d8SChris Lattner 25230fdc8d8SChris Lattner Debugger::~Debugger () 25330fdc8d8SChris Lattner { 2543d6086f6SCaroline Tice CleanUpInputReaders(); 2556611103cSGreg Clayton int num_targets = m_target_list.GetNumTargets(); 2566611103cSGreg Clayton for (int i = 0; i < num_targets; i++) 2576611103cSGreg Clayton { 2586611103cSGreg Clayton ProcessSP process_sp (m_target_list.GetTargetAtIndex (i)->GetProcessSP()); 2596611103cSGreg Clayton if (process_sp) 2606611103cSGreg Clayton process_sp->Destroy(); 2616611103cSGreg Clayton } 2626611103cSGreg Clayton DisconnectInput(); 26330fdc8d8SChris Lattner } 26430fdc8d8SChris Lattner 26530fdc8d8SChris Lattner 26630fdc8d8SChris Lattner bool 267fc3f027dSGreg Clayton Debugger::GetCloseInputOnEOF () const 268fc3f027dSGreg Clayton { 269fc3f027dSGreg Clayton return m_input_comm.GetCloseOnEOF(); 270fc3f027dSGreg Clayton } 271fc3f027dSGreg Clayton 272fc3f027dSGreg Clayton void 273fc3f027dSGreg Clayton Debugger::SetCloseInputOnEOF (bool b) 274fc3f027dSGreg Clayton { 275fc3f027dSGreg Clayton m_input_comm.SetCloseOnEOF(b); 276fc3f027dSGreg Clayton } 277fc3f027dSGreg Clayton 278fc3f027dSGreg Clayton bool 27930fdc8d8SChris Lattner Debugger::GetAsyncExecution () 28030fdc8d8SChris Lattner { 2816611103cSGreg Clayton return !m_command_interpreter_ap->GetSynchronous(); 28230fdc8d8SChris Lattner } 28330fdc8d8SChris Lattner 28430fdc8d8SChris Lattner void 28530fdc8d8SChris Lattner Debugger::SetAsyncExecution (bool async_execution) 28630fdc8d8SChris Lattner { 2876611103cSGreg Clayton m_command_interpreter_ap->SetSynchronous (!async_execution); 28830fdc8d8SChris Lattner } 28930fdc8d8SChris Lattner 29030fdc8d8SChris Lattner 29130fdc8d8SChris Lattner void 29230fdc8d8SChris Lattner Debugger::SetInputFileHandle (FILE *fh, bool tranfer_ownership) 29330fdc8d8SChris Lattner { 29451b1e2d2SGreg Clayton File &in_file = GetInputFile(); 29551b1e2d2SGreg Clayton in_file.SetStream (fh, tranfer_ownership); 29651b1e2d2SGreg Clayton if (in_file.IsValid() == false) 29751b1e2d2SGreg Clayton in_file.SetStream (stdin, true); 29830fdc8d8SChris Lattner 29930fdc8d8SChris Lattner // Disconnect from any old connection if we had one 30030fdc8d8SChris Lattner m_input_comm.Disconnect (); 30151b1e2d2SGreg Clayton m_input_comm.SetConnection (new ConnectionFileDescriptor (in_file.GetDescriptor(), true)); 30230fdc8d8SChris Lattner m_input_comm.SetReadThreadBytesReceivedCallback (Debugger::DispatchInputCallback, this); 30330fdc8d8SChris Lattner 30430fdc8d8SChris Lattner Error error; 30530fdc8d8SChris Lattner if (m_input_comm.StartReadThread (&error) == false) 30630fdc8d8SChris Lattner { 30751b1e2d2SGreg Clayton File &err_file = GetErrorFile(); 30851b1e2d2SGreg Clayton 30951b1e2d2SGreg Clayton err_file.Printf ("error: failed to main input read thread: %s", error.AsCString() ? error.AsCString() : "unkown error"); 31030fdc8d8SChris Lattner exit(1); 31130fdc8d8SChris Lattner } 31230fdc8d8SChris Lattner } 31330fdc8d8SChris Lattner 31430fdc8d8SChris Lattner void 31530fdc8d8SChris Lattner Debugger::SetOutputFileHandle (FILE *fh, bool tranfer_ownership) 31630fdc8d8SChris Lattner { 31751b1e2d2SGreg Clayton File &out_file = GetOutputFile(); 31851b1e2d2SGreg Clayton out_file.SetStream (fh, tranfer_ownership); 31951b1e2d2SGreg Clayton if (out_file.IsValid() == false) 32051b1e2d2SGreg Clayton out_file.SetStream (stdout, false); 3212f88aadfSCaroline Tice 3222f88aadfSCaroline Tice GetCommandInterpreter().GetScriptInterpreter()->ResetOutputFileHandle (fh); 32330fdc8d8SChris Lattner } 32430fdc8d8SChris Lattner 32530fdc8d8SChris Lattner void 32630fdc8d8SChris Lattner Debugger::SetErrorFileHandle (FILE *fh, bool tranfer_ownership) 32730fdc8d8SChris Lattner { 32851b1e2d2SGreg Clayton File &err_file = GetErrorFile(); 32951b1e2d2SGreg Clayton err_file.SetStream (fh, tranfer_ownership); 33051b1e2d2SGreg Clayton if (err_file.IsValid() == false) 33151b1e2d2SGreg Clayton err_file.SetStream (stderr, false); 33230fdc8d8SChris Lattner } 33330fdc8d8SChris Lattner 33430fdc8d8SChris Lattner ExecutionContext 3352976d00aSJim Ingham Debugger::GetSelectedExecutionContext () 33630fdc8d8SChris Lattner { 33730fdc8d8SChris Lattner ExecutionContext exe_ctx; 33830fdc8d8SChris Lattner exe_ctx.Clear(); 33930fdc8d8SChris Lattner 3402976d00aSJim Ingham lldb::TargetSP target_sp = GetSelectedTarget(); 34130fdc8d8SChris Lattner exe_ctx.target = target_sp.get(); 34230fdc8d8SChris Lattner 34330fdc8d8SChris Lattner if (target_sp) 34430fdc8d8SChris Lattner { 34530fdc8d8SChris Lattner exe_ctx.process = target_sp->GetProcessSP().get(); 34630fdc8d8SChris Lattner if (exe_ctx.process && exe_ctx.process->IsRunning() == false) 34730fdc8d8SChris Lattner { 3482976d00aSJim Ingham exe_ctx.thread = exe_ctx.process->GetThreadList().GetSelectedThread().get(); 34930fdc8d8SChris Lattner if (exe_ctx.thread == NULL) 35030fdc8d8SChris Lattner exe_ctx.thread = exe_ctx.process->GetThreadList().GetThreadAtIndex(0).get(); 35130fdc8d8SChris Lattner if (exe_ctx.thread) 35230fdc8d8SChris Lattner { 3532976d00aSJim Ingham exe_ctx.frame = exe_ctx.thread->GetSelectedFrame().get(); 35430fdc8d8SChris Lattner if (exe_ctx.frame == NULL) 35530fdc8d8SChris Lattner exe_ctx.frame = exe_ctx.thread->GetStackFrameAtIndex (0).get(); 35630fdc8d8SChris Lattner } 35730fdc8d8SChris Lattner } 35830fdc8d8SChris Lattner } 35930fdc8d8SChris Lattner return exe_ctx; 36030fdc8d8SChris Lattner 36130fdc8d8SChris Lattner } 36230fdc8d8SChris Lattner 363b44880caSCaroline Tice InputReaderSP 364b44880caSCaroline Tice Debugger::GetCurrentInputReader () 365b44880caSCaroline Tice { 366b44880caSCaroline Tice InputReaderSP reader_sp; 367b44880caSCaroline Tice 368d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 369b44880caSCaroline Tice { 370b44880caSCaroline Tice // Clear any finished readers from the stack 371b44880caSCaroline Tice while (CheckIfTopInputReaderIsDone()) ; 372b44880caSCaroline Tice 373d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 374d5a0a01bSCaroline Tice reader_sp = m_input_reader_stack.Top(); 375b44880caSCaroline Tice } 376b44880caSCaroline Tice 377b44880caSCaroline Tice return reader_sp; 378b44880caSCaroline Tice } 379b44880caSCaroline Tice 38030fdc8d8SChris Lattner void 38130fdc8d8SChris Lattner Debugger::DispatchInputCallback (void *baton, const void *bytes, size_t bytes_len) 38230fdc8d8SChris Lattner { 383efed6131SCaroline Tice if (bytes_len > 0) 38430fdc8d8SChris Lattner ((Debugger *)baton)->DispatchInput ((char *)bytes, bytes_len); 385efed6131SCaroline Tice else 386efed6131SCaroline Tice ((Debugger *)baton)->DispatchInputEndOfFile (); 38730fdc8d8SChris Lattner } 38830fdc8d8SChris Lattner 38930fdc8d8SChris Lattner 39030fdc8d8SChris Lattner void 39130fdc8d8SChris Lattner Debugger::DispatchInput (const char *bytes, size_t bytes_len) 39230fdc8d8SChris Lattner { 393efed6131SCaroline Tice if (bytes == NULL || bytes_len == 0) 394efed6131SCaroline Tice return; 39530fdc8d8SChris Lattner 39630fdc8d8SChris Lattner WriteToDefaultReader (bytes, bytes_len); 39730fdc8d8SChris Lattner } 39830fdc8d8SChris Lattner 39930fdc8d8SChris Lattner void 400efed6131SCaroline Tice Debugger::DispatchInputInterrupt () 401efed6131SCaroline Tice { 402efed6131SCaroline Tice m_input_reader_data.clear(); 403efed6131SCaroline Tice 404b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 405efed6131SCaroline Tice if (reader_sp) 406b44880caSCaroline Tice { 407efed6131SCaroline Tice reader_sp->Notify (eInputReaderInterrupt); 408efed6131SCaroline Tice 409b44880caSCaroline Tice // If notifying the reader of the interrupt finished the reader, we should pop it off the stack. 410efed6131SCaroline Tice while (CheckIfTopInputReaderIsDone ()) ; 411efed6131SCaroline Tice } 412efed6131SCaroline Tice } 413efed6131SCaroline Tice 414efed6131SCaroline Tice void 415efed6131SCaroline Tice Debugger::DispatchInputEndOfFile () 416efed6131SCaroline Tice { 417efed6131SCaroline Tice m_input_reader_data.clear(); 418efed6131SCaroline Tice 419b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 420efed6131SCaroline Tice if (reader_sp) 421b44880caSCaroline Tice { 422efed6131SCaroline Tice reader_sp->Notify (eInputReaderEndOfFile); 423efed6131SCaroline Tice 424b44880caSCaroline Tice // If notifying the reader of the end-of-file finished the reader, we should pop it off the stack. 425efed6131SCaroline Tice while (CheckIfTopInputReaderIsDone ()) ; 426efed6131SCaroline Tice } 427efed6131SCaroline Tice } 428efed6131SCaroline Tice 429efed6131SCaroline Tice void 4303d6086f6SCaroline Tice Debugger::CleanUpInputReaders () 4313d6086f6SCaroline Tice { 4323d6086f6SCaroline Tice m_input_reader_data.clear(); 4333d6086f6SCaroline Tice 434b44880caSCaroline Tice // The bottom input reader should be the main debugger input reader. We do not want to close that one here. 435d5a0a01bSCaroline Tice while (m_input_reader_stack.GetSize() > 1) 4363d6086f6SCaroline Tice { 437b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 4383d6086f6SCaroline Tice if (reader_sp) 4393d6086f6SCaroline Tice { 4403d6086f6SCaroline Tice reader_sp->Notify (eInputReaderEndOfFile); 4413d6086f6SCaroline Tice reader_sp->SetIsDone (true); 4423d6086f6SCaroline Tice } 4433d6086f6SCaroline Tice } 4443d6086f6SCaroline Tice } 4453d6086f6SCaroline Tice 4463d6086f6SCaroline Tice void 447969ed3d1SCaroline Tice Debugger::NotifyTopInputReader (InputReaderAction notification) 448969ed3d1SCaroline Tice { 449969ed3d1SCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader()); 450969ed3d1SCaroline Tice if (reader_sp) 451969ed3d1SCaroline Tice { 452969ed3d1SCaroline Tice reader_sp->Notify (notification); 453969ed3d1SCaroline Tice 454969ed3d1SCaroline Tice // Flush out any input readers that are done. 455969ed3d1SCaroline Tice while (CheckIfTopInputReaderIsDone ()) 456969ed3d1SCaroline Tice /* Do nothing. */; 457969ed3d1SCaroline Tice } 458969ed3d1SCaroline Tice } 459969ed3d1SCaroline Tice 4609088b068SCaroline Tice bool 4619088b068SCaroline Tice Debugger::InputReaderIsTopReader (const lldb::InputReaderSP& reader_sp) 4629088b068SCaroline Tice { 4639088b068SCaroline Tice InputReaderSP top_reader_sp (GetCurrentInputReader()); 4649088b068SCaroline Tice 465d61c10bcSCaroline Tice return (reader_sp.get() == top_reader_sp.get()); 4669088b068SCaroline Tice } 4679088b068SCaroline Tice 4689088b068SCaroline Tice 469969ed3d1SCaroline Tice void 47030fdc8d8SChris Lattner Debugger::WriteToDefaultReader (const char *bytes, size_t bytes_len) 47130fdc8d8SChris Lattner { 47230fdc8d8SChris Lattner if (bytes && bytes_len) 47330fdc8d8SChris Lattner m_input_reader_data.append (bytes, bytes_len); 47430fdc8d8SChris Lattner 47530fdc8d8SChris Lattner if (m_input_reader_data.empty()) 47630fdc8d8SChris Lattner return; 47730fdc8d8SChris Lattner 478d5a0a01bSCaroline Tice while (!m_input_reader_stack.IsEmpty() && !m_input_reader_data.empty()) 47930fdc8d8SChris Lattner { 48030fdc8d8SChris Lattner // Get the input reader from the top of the stack 481b44880caSCaroline Tice InputReaderSP reader_sp (GetCurrentInputReader ()); 48230fdc8d8SChris Lattner if (!reader_sp) 48330fdc8d8SChris Lattner break; 48430fdc8d8SChris Lattner 485471b31ceSGreg Clayton size_t bytes_handled = reader_sp->HandleRawBytes (m_input_reader_data.c_str(), 48630fdc8d8SChris Lattner m_input_reader_data.size()); 48730fdc8d8SChris Lattner if (bytes_handled) 48830fdc8d8SChris Lattner { 48930fdc8d8SChris Lattner m_input_reader_data.erase (0, bytes_handled); 49030fdc8d8SChris Lattner } 49130fdc8d8SChris Lattner else 49230fdc8d8SChris Lattner { 49330fdc8d8SChris Lattner // No bytes were handled, we might not have reached our 49430fdc8d8SChris Lattner // granularity, just return and wait for more data 49530fdc8d8SChris Lattner break; 49630fdc8d8SChris Lattner } 49730fdc8d8SChris Lattner } 49830fdc8d8SChris Lattner 499b44880caSCaroline Tice // Flush out any input readers that are done. 50030fdc8d8SChris Lattner while (CheckIfTopInputReaderIsDone ()) 50130fdc8d8SChris Lattner /* Do nothing. */; 50230fdc8d8SChris Lattner 50330fdc8d8SChris Lattner } 50430fdc8d8SChris Lattner 50530fdc8d8SChris Lattner void 50630fdc8d8SChris Lattner Debugger::PushInputReader (const InputReaderSP& reader_sp) 50730fdc8d8SChris Lattner { 50830fdc8d8SChris Lattner if (!reader_sp) 50930fdc8d8SChris Lattner return; 510b44880caSCaroline Tice 51130fdc8d8SChris Lattner // Deactivate the old top reader 512b44880caSCaroline Tice InputReaderSP top_reader_sp (GetCurrentInputReader ()); 513b44880caSCaroline Tice 51430fdc8d8SChris Lattner if (top_reader_sp) 51530fdc8d8SChris Lattner top_reader_sp->Notify (eInputReaderDeactivate); 516b44880caSCaroline Tice 517d5a0a01bSCaroline Tice m_input_reader_stack.Push (reader_sp); 51830fdc8d8SChris Lattner reader_sp->Notify (eInputReaderActivate); 51930fdc8d8SChris Lattner ActivateInputReader (reader_sp); 52030fdc8d8SChris Lattner } 52130fdc8d8SChris Lattner 52230fdc8d8SChris Lattner bool 52330fdc8d8SChris Lattner Debugger::PopInputReader (const lldb::InputReaderSP& pop_reader_sp) 52430fdc8d8SChris Lattner { 52530fdc8d8SChris Lattner bool result = false; 52630fdc8d8SChris Lattner 52730fdc8d8SChris Lattner // The reader on the stop of the stack is done, so let the next 52830fdc8d8SChris Lattner // read on the stack referesh its prompt and if there is one... 529d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 53030fdc8d8SChris Lattner { 531b44880caSCaroline Tice // Cannot call GetCurrentInputReader here, as that would cause an infinite loop. 532d5a0a01bSCaroline Tice InputReaderSP reader_sp(m_input_reader_stack.Top()); 53330fdc8d8SChris Lattner 53430fdc8d8SChris Lattner if (!pop_reader_sp || pop_reader_sp.get() == reader_sp.get()) 53530fdc8d8SChris Lattner { 536d5a0a01bSCaroline Tice m_input_reader_stack.Pop (); 53730fdc8d8SChris Lattner reader_sp->Notify (eInputReaderDeactivate); 53830fdc8d8SChris Lattner reader_sp->Notify (eInputReaderDone); 53930fdc8d8SChris Lattner result = true; 54030fdc8d8SChris Lattner 541d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 54230fdc8d8SChris Lattner { 543d5a0a01bSCaroline Tice reader_sp = m_input_reader_stack.Top(); 54430fdc8d8SChris Lattner if (reader_sp) 54530fdc8d8SChris Lattner { 54630fdc8d8SChris Lattner ActivateInputReader (reader_sp); 54730fdc8d8SChris Lattner reader_sp->Notify (eInputReaderReactivate); 54830fdc8d8SChris Lattner } 54930fdc8d8SChris Lattner } 55030fdc8d8SChris Lattner } 55130fdc8d8SChris Lattner } 55230fdc8d8SChris Lattner return result; 55330fdc8d8SChris Lattner } 55430fdc8d8SChris Lattner 55530fdc8d8SChris Lattner bool 55630fdc8d8SChris Lattner Debugger::CheckIfTopInputReaderIsDone () 55730fdc8d8SChris Lattner { 55830fdc8d8SChris Lattner bool result = false; 559d5a0a01bSCaroline Tice if (!m_input_reader_stack.IsEmpty()) 56030fdc8d8SChris Lattner { 561b44880caSCaroline Tice // Cannot call GetCurrentInputReader here, as that would cause an infinite loop. 562d5a0a01bSCaroline Tice InputReaderSP reader_sp(m_input_reader_stack.Top()); 56330fdc8d8SChris Lattner 56430fdc8d8SChris Lattner if (reader_sp && reader_sp->IsDone()) 56530fdc8d8SChris Lattner { 56630fdc8d8SChris Lattner result = true; 56730fdc8d8SChris Lattner PopInputReader (reader_sp); 56830fdc8d8SChris Lattner } 56930fdc8d8SChris Lattner } 57030fdc8d8SChris Lattner return result; 57130fdc8d8SChris Lattner } 57230fdc8d8SChris Lattner 57330fdc8d8SChris Lattner void 57430fdc8d8SChris Lattner Debugger::ActivateInputReader (const InputReaderSP &reader_sp) 57530fdc8d8SChris Lattner { 57651b1e2d2SGreg Clayton int input_fd = m_input_file.GetFile().GetDescriptor(); 57730fdc8d8SChris Lattner 57851b1e2d2SGreg Clayton if (input_fd >= 0) 57930fdc8d8SChris Lattner { 58051b1e2d2SGreg Clayton Terminal tty(input_fd); 581a3406614SGreg Clayton 582a3406614SGreg Clayton tty.SetEcho(reader_sp->GetEcho()); 58330fdc8d8SChris Lattner 58430fdc8d8SChris Lattner switch (reader_sp->GetGranularity()) 58530fdc8d8SChris Lattner { 58630fdc8d8SChris Lattner case eInputReaderGranularityByte: 58730fdc8d8SChris Lattner case eInputReaderGranularityWord: 588a3406614SGreg Clayton tty.SetCanonical (false); 58930fdc8d8SChris Lattner break; 59030fdc8d8SChris Lattner 59130fdc8d8SChris Lattner case eInputReaderGranularityLine: 59230fdc8d8SChris Lattner case eInputReaderGranularityAll: 593a3406614SGreg Clayton tty.SetCanonical (true); 59430fdc8d8SChris Lattner break; 59530fdc8d8SChris Lattner 59630fdc8d8SChris Lattner default: 59730fdc8d8SChris Lattner break; 59830fdc8d8SChris Lattner } 59930fdc8d8SChris Lattner } 60030fdc8d8SChris Lattner } 6016611103cSGreg Clayton 6025b52f0c7SJim Ingham StreamSP 6035b52f0c7SJim Ingham Debugger::GetAsyncOutputStream () 6045b52f0c7SJim Ingham { 6055b52f0c7SJim Ingham return StreamSP (new StreamAsynchronousIO (GetCommandInterpreter(), 6065b52f0c7SJim Ingham CommandInterpreter::eBroadcastBitAsynchronousOutputData)); 6075b52f0c7SJim Ingham } 6085b52f0c7SJim Ingham 6095b52f0c7SJim Ingham StreamSP 6105b52f0c7SJim Ingham Debugger::GetAsyncErrorStream () 6115b52f0c7SJim Ingham { 6125b52f0c7SJim Ingham return StreamSP (new StreamAsynchronousIO (GetCommandInterpreter(), 6135b52f0c7SJim Ingham CommandInterpreter::eBroadcastBitAsynchronousErrorData)); 6145b52f0c7SJim Ingham } 6155b52f0c7SJim Ingham 616ebc1bb27SCaroline Tice DebuggerSP 617ebc1bb27SCaroline Tice Debugger::FindDebuggerWithID (lldb::user_id_t id) 618ebc1bb27SCaroline Tice { 619ebc1bb27SCaroline Tice lldb::DebuggerSP debugger_sp; 620ebc1bb27SCaroline Tice 621ebc1bb27SCaroline Tice Mutex::Locker locker (GetDebuggerListMutex ()); 622ebc1bb27SCaroline Tice DebuggerList &debugger_list = GetDebuggerList(); 623ebc1bb27SCaroline Tice DebuggerList::iterator pos, end = debugger_list.end(); 624ebc1bb27SCaroline Tice for (pos = debugger_list.begin(); pos != end; ++pos) 625ebc1bb27SCaroline Tice { 626ebc1bb27SCaroline Tice if ((*pos).get()->GetID() == id) 627ebc1bb27SCaroline Tice { 628ebc1bb27SCaroline Tice debugger_sp = *pos; 629ebc1bb27SCaroline Tice break; 630ebc1bb27SCaroline Tice } 631ebc1bb27SCaroline Tice } 632ebc1bb27SCaroline Tice return debugger_sp; 633ebc1bb27SCaroline Tice } 6343df9a8dfSCaroline Tice 6351b654882SGreg Clayton static void 6361b654882SGreg Clayton TestPromptFormats (StackFrame *frame) 6371b654882SGreg Clayton { 6381b654882SGreg Clayton if (frame == NULL) 6391b654882SGreg Clayton return; 6401b654882SGreg Clayton 6411b654882SGreg Clayton StreamString s; 6421b654882SGreg Clayton const char *prompt_format = 6431b654882SGreg Clayton "{addr = '${addr}'\n}" 6441b654882SGreg Clayton "{process.id = '${process.id}'\n}" 6451b654882SGreg Clayton "{process.name = '${process.name}'\n}" 6461b654882SGreg Clayton "{process.file.basename = '${process.file.basename}'\n}" 6471b654882SGreg Clayton "{process.file.fullpath = '${process.file.fullpath}'\n}" 6481b654882SGreg Clayton "{thread.id = '${thread.id}'\n}" 6491b654882SGreg Clayton "{thread.index = '${thread.index}'\n}" 6501b654882SGreg Clayton "{thread.name = '${thread.name}'\n}" 6511b654882SGreg Clayton "{thread.queue = '${thread.queue}'\n}" 6521b654882SGreg Clayton "{thread.stop-reason = '${thread.stop-reason}'\n}" 6531b654882SGreg Clayton "{target.arch = '${target.arch}'\n}" 6541b654882SGreg Clayton "{module.file.basename = '${module.file.basename}'\n}" 6551b654882SGreg Clayton "{module.file.fullpath = '${module.file.fullpath}'\n}" 6561b654882SGreg Clayton "{file.basename = '${file.basename}'\n}" 6571b654882SGreg Clayton "{file.fullpath = '${file.fullpath}'\n}" 6581b654882SGreg Clayton "{frame.index = '${frame.index}'\n}" 6591b654882SGreg Clayton "{frame.pc = '${frame.pc}'\n}" 6601b654882SGreg Clayton "{frame.sp = '${frame.sp}'\n}" 6611b654882SGreg Clayton "{frame.fp = '${frame.fp}'\n}" 6621b654882SGreg Clayton "{frame.flags = '${frame.flags}'\n}" 6631b654882SGreg Clayton "{frame.reg.rdi = '${frame.reg.rdi}'\n}" 6641b654882SGreg Clayton "{frame.reg.rip = '${frame.reg.rip}'\n}" 6651b654882SGreg Clayton "{frame.reg.rsp = '${frame.reg.rsp}'\n}" 6661b654882SGreg Clayton "{frame.reg.rbp = '${frame.reg.rbp}'\n}" 6671b654882SGreg Clayton "{frame.reg.rflags = '${frame.reg.rflags}'\n}" 6681b654882SGreg Clayton "{frame.reg.xmm0 = '${frame.reg.xmm0}'\n}" 6691b654882SGreg Clayton "{frame.reg.carp = '${frame.reg.carp}'\n}" 6701b654882SGreg Clayton "{function.id = '${function.id}'\n}" 6711b654882SGreg Clayton "{function.name = '${function.name}'\n}" 6721b654882SGreg Clayton "{function.addr-offset = '${function.addr-offset}'\n}" 6731b654882SGreg Clayton "{function.line-offset = '${function.line-offset}'\n}" 6741b654882SGreg Clayton "{function.pc-offset = '${function.pc-offset}'\n}" 6751b654882SGreg Clayton "{line.file.basename = '${line.file.basename}'\n}" 6761b654882SGreg Clayton "{line.file.fullpath = '${line.file.fullpath}'\n}" 6771b654882SGreg Clayton "{line.number = '${line.number}'\n}" 6781b654882SGreg Clayton "{line.start-addr = '${line.start-addr}'\n}" 6791b654882SGreg Clayton "{line.end-addr = '${line.end-addr}'\n}" 6801b654882SGreg Clayton ; 6811b654882SGreg Clayton 6821b654882SGreg Clayton SymbolContext sc (frame->GetSymbolContext(eSymbolContextEverything)); 6831b654882SGreg Clayton ExecutionContext exe_ctx; 6840603aa9dSGreg Clayton frame->CalculateExecutionContext(exe_ctx); 6851b654882SGreg Clayton const char *end = NULL; 6861b654882SGreg Clayton if (Debugger::FormatPrompt (prompt_format, &sc, &exe_ctx, &sc.line_entry.range.GetBaseAddress(), s, &end)) 6871b654882SGreg Clayton { 6881b654882SGreg Clayton printf("%s\n", s.GetData()); 6891b654882SGreg Clayton } 6901b654882SGreg Clayton else 6911b654882SGreg Clayton { 6921b654882SGreg Clayton printf ("error: at '%s'\n", end); 6931b654882SGreg Clayton printf ("what we got: %s\n", s.GetData()); 6941b654882SGreg Clayton } 6951b654882SGreg Clayton } 6961b654882SGreg Clayton 697*f4efecd9SEnrico Granata // FIXME this should eventually be replaced by proper use of LLDB logging facilities 6989fc1944eSEnrico Granata //#define VERBOSE_FORMATPROMPT_OUTPUT 6999fc1944eSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 7009fc1944eSEnrico Granata #define IFERROR_PRINT_IT if (error.Fail()) \ 7019fc1944eSEnrico Granata { \ 7029fc1944eSEnrico Granata printf("ERROR: %s\n",error.AsCString("unknown")); \ 7039fc1944eSEnrico Granata break; \ 7049fc1944eSEnrico Granata } 7059fc1944eSEnrico Granata #else // IFERROR_PRINT_IT 7069fc1944eSEnrico Granata #define IFERROR_PRINT_IT if (error.Fail()) \ 7079fc1944eSEnrico Granata break; 7089fc1944eSEnrico Granata #endif // IFERROR_PRINT_IT 7099fc1944eSEnrico Granata 7109fc1944eSEnrico Granata static bool 7119fc1944eSEnrico Granata ScanFormatDescriptor(const char* var_name_begin, 7129fc1944eSEnrico Granata const char* var_name_end, 7139fc1944eSEnrico Granata const char** var_name_final, 7149fc1944eSEnrico Granata const char** percent_position, 7159fc1944eSEnrico Granata lldb::Format* custom_format, 7169fc1944eSEnrico Granata ValueObject::ValueObjectRepresentationStyle* val_obj_display) 7179fc1944eSEnrico Granata { 7189fc1944eSEnrico Granata *percent_position = ::strchr(var_name_begin,'%'); 7199fc1944eSEnrico Granata if (!*percent_position || *percent_position > var_name_end) 7209fc1944eSEnrico Granata *var_name_final = var_name_end; 7219fc1944eSEnrico Granata else 7229fc1944eSEnrico Granata { 7239fc1944eSEnrico Granata *var_name_final = *percent_position; 7249fc1944eSEnrico Granata char* format_name = new char[var_name_end-*var_name_final]; format_name[var_name_end-*var_name_final-1] = '\0'; 7259fc1944eSEnrico Granata memcpy(format_name, *var_name_final+1, var_name_end-*var_name_final-1); 7269fc1944eSEnrico Granata if ( !FormatManager::GetFormatFromCString(format_name, 7279fc1944eSEnrico Granata true, 7289fc1944eSEnrico Granata *custom_format) ) 7299fc1944eSEnrico Granata { 7309fc1944eSEnrico Granata // if this is an @ sign, print ObjC description 7319fc1944eSEnrico Granata if (*format_name == '@') 7329fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayLanguageSpecific; 7339fc1944eSEnrico Granata // if this is a V, print the value using the default format 7349fc1944eSEnrico Granata if (*format_name == 'V') 7359fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayValue; 7369fc1944eSEnrico Granata } 7379fc1944eSEnrico Granata // a good custom format tells us to print the value using it 7389fc1944eSEnrico Granata else 7399fc1944eSEnrico Granata *val_obj_display = ValueObject::eDisplayValue; 7409fc1944eSEnrico Granata delete format_name; 7419fc1944eSEnrico Granata } 7429fc1944eSEnrico Granata return true; 7439fc1944eSEnrico Granata } 7449fc1944eSEnrico Granata 7459fc1944eSEnrico Granata static bool 7469fc1944eSEnrico Granata ScanBracketedRange(const char* var_name_begin, 7479fc1944eSEnrico Granata const char* var_name_end, 7489fc1944eSEnrico Granata const char* var_name_final, 7499fc1944eSEnrico Granata const char** open_bracket_position, 7509fc1944eSEnrico Granata const char** separator_position, 7519fc1944eSEnrico Granata const char** close_bracket_position, 7529fc1944eSEnrico Granata const char** var_name_final_if_array_range, 7539fc1944eSEnrico Granata int64_t* index_lower, 7549fc1944eSEnrico Granata int64_t* index_higher) 7559fc1944eSEnrico Granata { 7569fc1944eSEnrico Granata *open_bracket_position = ::strchr(var_name_begin,'['); 7579fc1944eSEnrico Granata if (*open_bracket_position && *open_bracket_position < var_name_final) 7589fc1944eSEnrico Granata { 7599fc1944eSEnrico Granata *separator_position = ::strchr(*open_bracket_position,'-'); // might be NULL if this is a simple var[N] bitfield 7609fc1944eSEnrico Granata *close_bracket_position = ::strchr(*open_bracket_position,']'); 7619fc1944eSEnrico Granata // as usual, we assume that [] will come before % 7629fc1944eSEnrico Granata //printf("trying to expand a []\n"); 7639fc1944eSEnrico Granata *var_name_final_if_array_range = *open_bracket_position; 7649fc1944eSEnrico Granata if (*close_bracket_position - *open_bracket_position == 1) 7659fc1944eSEnrico Granata { 7669fc1944eSEnrico Granata *index_lower = 0; 7679fc1944eSEnrico Granata } 7689fc1944eSEnrico Granata else if (*separator_position == NULL || *separator_position > var_name_end) 7699fc1944eSEnrico Granata { 7709fc1944eSEnrico Granata char *end = NULL; 7719fc1944eSEnrico Granata *index_lower = ::strtoul (*open_bracket_position+1, &end, 0); 7729fc1944eSEnrico Granata *index_higher = *index_lower; 7739fc1944eSEnrico Granata //printf("got to read low=%d high same\n",bitfield_lower); 7749fc1944eSEnrico Granata } 7759fc1944eSEnrico Granata else if (*close_bracket_position && *close_bracket_position < var_name_end) 7769fc1944eSEnrico Granata { 7779fc1944eSEnrico Granata char *end = NULL; 7789fc1944eSEnrico Granata *index_lower = ::strtoul (*open_bracket_position+1, &end, 0); 7799fc1944eSEnrico Granata *index_higher = ::strtoul (*separator_position+1, &end, 0); 7809fc1944eSEnrico Granata //printf("got to read low=%d high=%d\n",bitfield_lower,bitfield_higher); 7819fc1944eSEnrico Granata } 7829fc1944eSEnrico Granata else 7839fc1944eSEnrico Granata return false; 7849fc1944eSEnrico Granata if (*index_lower > *index_higher && *index_higher > 0) 7859fc1944eSEnrico Granata { 7869fc1944eSEnrico Granata int temp = *index_lower; 7879fc1944eSEnrico Granata *index_lower = *index_higher; 7889fc1944eSEnrico Granata *index_higher = temp; 7899fc1944eSEnrico Granata } 7909fc1944eSEnrico Granata } 7919fc1944eSEnrico Granata return true; 7929fc1944eSEnrico Granata } 7939fc1944eSEnrico Granata 7949fc1944eSEnrico Granata 7959fc1944eSEnrico Granata static ValueObjectSP 7969fc1944eSEnrico Granata ExpandExpressionPath(ValueObject* vobj, 7979fc1944eSEnrico Granata StackFrame* frame, 7989fc1944eSEnrico Granata bool* do_deref_pointer, 7999fc1944eSEnrico Granata const char* var_name_begin, 8009fc1944eSEnrico Granata const char* var_name_final, 8019fc1944eSEnrico Granata Error& error) 8029fc1944eSEnrico Granata { 8039fc1944eSEnrico Granata 8049fc1944eSEnrico Granata StreamString sstring; 8059fc1944eSEnrico Granata VariableSP var_sp; 8069fc1944eSEnrico Granata 8079fc1944eSEnrico Granata if (*do_deref_pointer) 8089fc1944eSEnrico Granata sstring.PutChar('*'); 8099fc1944eSEnrico Granata else if (vobj->IsDereferenceOfParent() && ClangASTContext::IsPointerType(vobj->GetParent()->GetClangType()) && !vobj->IsArrayItemForPointer()) 8109fc1944eSEnrico Granata { 8119fc1944eSEnrico Granata sstring.PutChar('*'); 8129fc1944eSEnrico Granata *do_deref_pointer = true; 8139fc1944eSEnrico Granata } 8149fc1944eSEnrico Granata 8159fc1944eSEnrico Granata vobj->GetExpressionPath(sstring, true, ValueObject::eHonorPointers); 8169fc1944eSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 8179fc1944eSEnrico Granata printf("name to expand in phase 0: %s\n",sstring.GetData()); 8189fc1944eSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 8199fc1944eSEnrico Granata sstring.PutRawBytes(var_name_begin+3, var_name_final-var_name_begin-3); 8209fc1944eSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 8219fc1944eSEnrico Granata printf("name to expand in phase 1: %s\n",sstring.GetData()); 8229fc1944eSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 8239fc1944eSEnrico Granata std::string name = std::string(sstring.GetData()); 8249fc1944eSEnrico Granata ValueObjectSP target = frame->GetValueForVariableExpressionPath (name.c_str(), 8259fc1944eSEnrico Granata eNoDynamicValues, 8269fc1944eSEnrico Granata 0, 8279fc1944eSEnrico Granata var_sp, 8289fc1944eSEnrico Granata error); 8299fc1944eSEnrico Granata return target; 8309fc1944eSEnrico Granata } 8319fc1944eSEnrico Granata 8329fc1944eSEnrico Granata static ValueObjectSP 8339fc1944eSEnrico Granata ExpandIndexedExpression(ValueObject* vobj, 8349fc1944eSEnrico Granata uint32_t index, 8359fc1944eSEnrico Granata StackFrame* frame, 836fc7a7f3bSEnrico Granata bool deref_pointer) 8379fc1944eSEnrico Granata { 838fc7a7f3bSEnrico Granata const char* ptr_deref_format = "[%d]"; 839fc7a7f3bSEnrico Granata std::auto_ptr<char> ptr_deref_buffer(new char[10]); 840fc7a7f3bSEnrico Granata ::sprintf(ptr_deref_buffer.get(), ptr_deref_format, index); 841fc7a7f3bSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 842fc7a7f3bSEnrico Granata printf("name to deref: %s\n",ptr_deref_buffer.get()); 843fc7a7f3bSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 844fc7a7f3bSEnrico Granata const char* first_unparsed; 845fc7a7f3bSEnrico Granata ValueObject::GetValueForExpressionPathOptions options; 846fc7a7f3bSEnrico Granata ValueObject::ExpressionPathEndResultType final_value_type; 847fc7a7f3bSEnrico Granata ValueObject::ExpressionPathScanEndReason reason_to_stop; 848fc7a7f3bSEnrico Granata ValueObject::ExpressionPathAftermath what_next = (deref_pointer ? ValueObject::eDereference : ValueObject::eNothing); 849fc7a7f3bSEnrico Granata ValueObjectSP item = vobj->GetValueForExpressionPath (ptr_deref_buffer.get(), 850fc7a7f3bSEnrico Granata &first_unparsed, 851fc7a7f3bSEnrico Granata &reason_to_stop, 852fc7a7f3bSEnrico Granata &final_value_type, 853fc7a7f3bSEnrico Granata options, 854fc7a7f3bSEnrico Granata &what_next); 855fc7a7f3bSEnrico Granata if (!item) 856fc7a7f3bSEnrico Granata { 857fc7a7f3bSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 858fc7a7f3bSEnrico Granata printf("ERROR: unparsed portion = %s, why stopping = %d," 859fc7a7f3bSEnrico Granata " final_value_type %d\n", 860fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 861fc7a7f3bSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 862fc7a7f3bSEnrico Granata } 863fc7a7f3bSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 8649fc1944eSEnrico Granata else 8659fc1944eSEnrico Granata { 866fc7a7f3bSEnrico Granata printf("ALL RIGHT: unparsed portion = %s, why stopping = %d," 867fc7a7f3bSEnrico Granata " final_value_type %d\n", 868fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 8699fc1944eSEnrico Granata } 870fc7a7f3bSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 8719fc1944eSEnrico Granata return item; 8729fc1944eSEnrico Granata } 8739fc1944eSEnrico Granata 8741b654882SGreg Clayton bool 8751b654882SGreg Clayton Debugger::FormatPrompt 8761b654882SGreg Clayton ( 8771b654882SGreg Clayton const char *format, 8781b654882SGreg Clayton const SymbolContext *sc, 8791b654882SGreg Clayton const ExecutionContext *exe_ctx, 8801b654882SGreg Clayton const Address *addr, 8811b654882SGreg Clayton Stream &s, 8824becb37eSEnrico Granata const char **end, 8834becb37eSEnrico Granata ValueObject* vobj 8841b654882SGreg Clayton ) 8851b654882SGreg Clayton { 8864becb37eSEnrico Granata ValueObject* realvobj = NULL; // makes it super-easy to parse pointers 8871b654882SGreg Clayton bool success = true; 8881b654882SGreg Clayton const char *p; 8891b654882SGreg Clayton for (p = format; *p != '\0'; ++p) 8901b654882SGreg Clayton { 8914becb37eSEnrico Granata if (realvobj) 8924becb37eSEnrico Granata { 8934becb37eSEnrico Granata vobj = realvobj; 8944becb37eSEnrico Granata realvobj = NULL; 8954becb37eSEnrico Granata } 8961b654882SGreg Clayton size_t non_special_chars = ::strcspn (p, "${}\\"); 8971b654882SGreg Clayton if (non_special_chars > 0) 8981b654882SGreg Clayton { 8991b654882SGreg Clayton if (success) 9001b654882SGreg Clayton s.Write (p, non_special_chars); 9011b654882SGreg Clayton p += non_special_chars; 9021b654882SGreg Clayton } 9031b654882SGreg Clayton 9041b654882SGreg Clayton if (*p == '\0') 9051b654882SGreg Clayton { 9061b654882SGreg Clayton break; 9071b654882SGreg Clayton } 9081b654882SGreg Clayton else if (*p == '{') 9091b654882SGreg Clayton { 9101b654882SGreg Clayton // Start a new scope that must have everything it needs if it is to 9111b654882SGreg Clayton // to make it into the final output stream "s". If you want to make 9121b654882SGreg Clayton // a format that only prints out the function or symbol name if there 9131b654882SGreg Clayton // is one in the symbol context you can use: 9141b654882SGreg Clayton // "{function =${function.name}}" 9151b654882SGreg Clayton // The first '{' starts a new scope that end with the matching '}' at 9161b654882SGreg Clayton // the end of the string. The contents "function =${function.name}" 9171b654882SGreg Clayton // will then be evaluated and only be output if there is a function 9181b654882SGreg Clayton // or symbol with a valid name. 9191b654882SGreg Clayton StreamString sub_strm; 9201b654882SGreg Clayton 9211b654882SGreg Clayton ++p; // Skip the '{' 9221b654882SGreg Clayton 9234becb37eSEnrico Granata if (FormatPrompt (p, sc, exe_ctx, addr, sub_strm, &p, vobj)) 9241b654882SGreg Clayton { 9251b654882SGreg Clayton // The stream had all it needed 9261b654882SGreg Clayton s.Write(sub_strm.GetData(), sub_strm.GetSize()); 9271b654882SGreg Clayton } 9281b654882SGreg Clayton if (*p != '}') 9291b654882SGreg Clayton { 9301b654882SGreg Clayton success = false; 9311b654882SGreg Clayton break; 9321b654882SGreg Clayton } 9331b654882SGreg Clayton } 9341b654882SGreg Clayton else if (*p == '}') 9351b654882SGreg Clayton { 9361b654882SGreg Clayton // End of a enclosing scope 9371b654882SGreg Clayton break; 9381b654882SGreg Clayton } 9391b654882SGreg Clayton else if (*p == '$') 9401b654882SGreg Clayton { 9411b654882SGreg Clayton // We have a prompt variable to print 9421b654882SGreg Clayton ++p; 9431b654882SGreg Clayton if (*p == '{') 9441b654882SGreg Clayton { 9451b654882SGreg Clayton ++p; 9461b654882SGreg Clayton const char *var_name_begin = p; 9471b654882SGreg Clayton const char *var_name_end = ::strchr (p, '}'); 9481b654882SGreg Clayton 9491b654882SGreg Clayton if (var_name_end && var_name_begin < var_name_end) 9501b654882SGreg Clayton { 9511b654882SGreg Clayton // if we have already failed to parse, skip this variable 9521b654882SGreg Clayton if (success) 9531b654882SGreg Clayton { 9541b654882SGreg Clayton const char *cstr = NULL; 9551b654882SGreg Clayton Address format_addr; 9561b654882SGreg Clayton bool calculate_format_addr_function_offset = false; 9571b654882SGreg Clayton // Set reg_kind and reg_num to invalid values 9581b654882SGreg Clayton RegisterKind reg_kind = kNumRegisterKinds; 9591b654882SGreg Clayton uint32_t reg_num = LLDB_INVALID_REGNUM; 9601b654882SGreg Clayton FileSpec format_file_spec; 961e0d378b3SGreg Clayton const RegisterInfo *reg_info = NULL; 9621b654882SGreg Clayton RegisterContext *reg_ctx = NULL; 9639fc1944eSEnrico Granata bool do_deref_pointer = false; 964fc7a7f3bSEnrico Granata ValueObject::ExpressionPathScanEndReason reason_to_stop; 965fc7a7f3bSEnrico Granata ValueObject::ExpressionPathEndResultType final_value_type; 9661b654882SGreg Clayton 9671b654882SGreg Clayton // Each variable must set success to true below... 9681b654882SGreg Clayton bool var_success = false; 9691b654882SGreg Clayton switch (var_name_begin[0]) 9701b654882SGreg Clayton { 9714becb37eSEnrico Granata case '*': 9724becb37eSEnrico Granata { 97334132754SGreg Clayton if (!vobj) 97434132754SGreg Clayton break; 9759fc1944eSEnrico Granata do_deref_pointer = true; 9769fc1944eSEnrico Granata var_name_begin++; 9779fc1944eSEnrico Granata } 97834132754SGreg Clayton // Fall through... 97934132754SGreg Clayton 9809fc1944eSEnrico Granata case 'v': 9814becb37eSEnrico Granata { 982fc7a7f3bSEnrico Granata ValueObject::ExpressionPathAftermath what_next = (do_deref_pointer ? 983fc7a7f3bSEnrico Granata ValueObject::eDereference : ValueObject::eNothing); 984fc7a7f3bSEnrico Granata ValueObject::GetValueForExpressionPathOptions options; 985fc7a7f3bSEnrico Granata options.DontCheckDotVsArrowSyntax().DoAllowBitfieldSyntax().DoAllowFragileIVar(); 9860a3958e0SEnrico Granata ValueObject::ValueObjectRepresentationStyle val_obj_display = ValueObject::eDisplaySummary; 98734132754SGreg Clayton ValueObject* target = NULL; 9889fc1944eSEnrico Granata lldb::Format custom_format = eFormatInvalid; 98934132754SGreg Clayton const char* var_name_final = NULL; 9909fc1944eSEnrico Granata const char* var_name_final_if_array_range = NULL; 99134132754SGreg Clayton const char* close_bracket_position = NULL; 99234132754SGreg Clayton int64_t index_lower = -1; 99334132754SGreg Clayton int64_t index_higher = -1; 9949fc1944eSEnrico Granata bool is_array_range = false; 995fc7a7f3bSEnrico Granata const char* first_unparsed; 996fc7a7f3bSEnrico Granata 9979fc1944eSEnrico Granata if (!vobj) break; 9989fc1944eSEnrico Granata // simplest case ${var}, just print vobj's value 9999fc1944eSEnrico Granata if (::strncmp (var_name_begin, "var}", strlen("var}")) == 0) 10000a3958e0SEnrico Granata { 10019fc1944eSEnrico Granata target = vobj; 10020a3958e0SEnrico Granata val_obj_display = ValueObject::eDisplayValue; 10030a3958e0SEnrico Granata } 10049fc1944eSEnrico Granata else if (::strncmp(var_name_begin,"var%",strlen("var%")) == 0) 10059fc1944eSEnrico Granata { 10069fc1944eSEnrico Granata // this is a variable with some custom format applied to it 10079fc1944eSEnrico Granata const char* percent_position; 10089fc1944eSEnrico Granata target = vobj; 10090a3958e0SEnrico Granata val_obj_display = ValueObject::eDisplayValue; 10109fc1944eSEnrico Granata ScanFormatDescriptor (var_name_begin, 10119fc1944eSEnrico Granata var_name_end, 10129fc1944eSEnrico Granata &var_name_final, 10139fc1944eSEnrico Granata &percent_position, 10149fc1944eSEnrico Granata &custom_format, 10159fc1944eSEnrico Granata &val_obj_display); 10160a3958e0SEnrico Granata } 10179fc1944eSEnrico Granata // this is ${var.something} or multiple .something nested 10189fc1944eSEnrico Granata else if (::strncmp (var_name_begin, "var", strlen("var")) == 0) 10199fc1944eSEnrico Granata { 10209fc1944eSEnrico Granata 10219fc1944eSEnrico Granata const char* percent_position; 10229fc1944eSEnrico Granata ScanFormatDescriptor (var_name_begin, 10239fc1944eSEnrico Granata var_name_end, 10249fc1944eSEnrico Granata &var_name_final, 10259fc1944eSEnrico Granata &percent_position, 10269fc1944eSEnrico Granata &custom_format, 10279fc1944eSEnrico Granata &val_obj_display); 10289fc1944eSEnrico Granata 10299fc1944eSEnrico Granata const char* open_bracket_position; 10309fc1944eSEnrico Granata const char* separator_position; 10319fc1944eSEnrico Granata ScanBracketedRange (var_name_begin, 10329fc1944eSEnrico Granata var_name_end, 10339fc1944eSEnrico Granata var_name_final, 10349fc1944eSEnrico Granata &open_bracket_position, 10359fc1944eSEnrico Granata &separator_position, 10369fc1944eSEnrico Granata &close_bracket_position, 10379fc1944eSEnrico Granata &var_name_final_if_array_range, 10389fc1944eSEnrico Granata &index_lower, 10399fc1944eSEnrico Granata &index_higher); 10409fc1944eSEnrico Granata 10419fc1944eSEnrico Granata Error error; 10429fc1944eSEnrico Granata 1043fc7a7f3bSEnrico Granata std::auto_ptr<char> expr_path(new char[var_name_final-var_name_begin-1]); 1044fc7a7f3bSEnrico Granata ::memset(expr_path.get(), 0, var_name_final-var_name_begin-1); 1045fc7a7f3bSEnrico Granata memcpy(expr_path.get(), var_name_begin+3,var_name_final-var_name_begin-3); 1046fc7a7f3bSEnrico Granata 1047fc7a7f3bSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 1048fc7a7f3bSEnrico Granata printf("symbol to expand: %s\n",expr_path.get()); 1049fc7a7f3bSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 1050fc7a7f3bSEnrico Granata 1051fc7a7f3bSEnrico Granata target = vobj->GetValueForExpressionPath(expr_path.get(), 1052fc7a7f3bSEnrico Granata &first_unparsed, 1053fc7a7f3bSEnrico Granata &reason_to_stop, 1054fc7a7f3bSEnrico Granata &final_value_type, 1055fc7a7f3bSEnrico Granata options, 1056fc7a7f3bSEnrico Granata &what_next).get(); 1057fc7a7f3bSEnrico Granata 1058fc7a7f3bSEnrico Granata if (!target) 10599fc1944eSEnrico Granata { 10609fc1944eSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 1061fc7a7f3bSEnrico Granata printf("ERROR: unparsed portion = %s, why stopping = %d," 1062fc7a7f3bSEnrico Granata " final_value_type %d\n", 1063fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 10649fc1944eSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 1065fc7a7f3bSEnrico Granata break; 10660a3958e0SEnrico Granata } 1067fc7a7f3bSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 1068a7187d00SEnrico Granata else 1069fc7a7f3bSEnrico Granata { 1070fc7a7f3bSEnrico Granata printf("ALL RIGHT: unparsed portion = %s, why stopping = %d," 1071fc7a7f3bSEnrico Granata " final_value_type %d\n", 1072fc7a7f3bSEnrico Granata first_unparsed, reason_to_stop, final_value_type); 1073a7187d00SEnrico Granata } 1074fc7a7f3bSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 10750a3958e0SEnrico Granata } 10760a3958e0SEnrico Granata else 10770a3958e0SEnrico Granata break; 10789fc1944eSEnrico Granata 1079fc7a7f3bSEnrico Granata is_array_range = (final_value_type == ValueObject::eBoundedRange || 1080fc7a7f3bSEnrico Granata final_value_type == ValueObject::eUnboundedRange); 1081fc7a7f3bSEnrico Granata 1082fc7a7f3bSEnrico Granata do_deref_pointer = (what_next == ValueObject::eDereference); 1083fc7a7f3bSEnrico Granata 1084a7187d00SEnrico Granata if (do_deref_pointer && !is_array_range) 10850a3958e0SEnrico Granata { 10869fc1944eSEnrico Granata // I have not deref-ed yet, let's do it 10879fc1944eSEnrico Granata // this happens when we are not going through GetValueForVariableExpressionPath 10889fc1944eSEnrico Granata // to get to the target ValueObject 10899fc1944eSEnrico Granata Error error; 10909fc1944eSEnrico Granata target = target->Dereference(error).get(); 10919fc1944eSEnrico Granata IFERROR_PRINT_IT 10929fc1944eSEnrico Granata do_deref_pointer = false; 10930a3958e0SEnrico Granata } 10940a3958e0SEnrico Granata 1095*f4efecd9SEnrico Granata bool is_array = ClangASTContext::IsArrayType(target->GetClangType()); 1096*f4efecd9SEnrico Granata bool is_pointer = ClangASTContext::IsPointerType(target->GetClangType()); 1097*f4efecd9SEnrico Granata 1098*f4efecd9SEnrico Granata if ((is_array || is_pointer) && (!is_array_range) && val_obj_display == ValueObject::eDisplayValue) // this should be wrong, but there are some exceptions 1099*f4efecd9SEnrico Granata { 1100*f4efecd9SEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 1101*f4efecd9SEnrico Granata printf("I am into array || pointer && !range\n"); 1102*f4efecd9SEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 1103*f4efecd9SEnrico Granata // try to use the special cases 1104*f4efecd9SEnrico Granata var_success = target->DumpPrintableRepresentation(s,val_obj_display, custom_format); 1105*f4efecd9SEnrico Granata if (!var_success) 1106*f4efecd9SEnrico Granata s << "<invalid, please use [] operator>"; 1107*f4efecd9SEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 1108*f4efecd9SEnrico Granata printf("outcome was : %s\n", var_success ? "good" : "bad"); 1109*f4efecd9SEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 1110*f4efecd9SEnrico Granata break; 1111*f4efecd9SEnrico Granata } 1112*f4efecd9SEnrico Granata 11139fc1944eSEnrico Granata if (!is_array_range) 11149fc1944eSEnrico Granata var_success = target->DumpPrintableRepresentation(s,val_obj_display, custom_format); 11159fc1944eSEnrico Granata else 11169fc1944eSEnrico Granata { 11179fc1944eSEnrico Granata if (!is_array && !is_pointer) 11189fc1944eSEnrico Granata break; 11199fc1944eSEnrico Granata 1120fc7a7f3bSEnrico Granata const char* special_directions = NULL; 1121fc7a7f3bSEnrico Granata StreamString special_directions_writer; 11220a3958e0SEnrico Granata if (close_bracket_position && (var_name_end-close_bracket_position > 1)) 11230a3958e0SEnrico Granata { 1124fc7a7f3bSEnrico Granata ConstString additional_data; 1125fc7a7f3bSEnrico Granata additional_data.SetCStringWithLength(close_bracket_position+1, var_name_end-close_bracket_position-1); 1126fc7a7f3bSEnrico Granata special_directions_writer.Printf("${%svar%s}", 1127fc7a7f3bSEnrico Granata do_deref_pointer ? "*" : "", 1128fc7a7f3bSEnrico Granata additional_data.GetCString()); 1129fc7a7f3bSEnrico Granata special_directions = special_directions_writer.GetData(); 11300a3958e0SEnrico Granata } 11310a3958e0SEnrico Granata 11320a3958e0SEnrico Granata // let us display items index_lower thru index_higher of this array 11330a3958e0SEnrico Granata s.PutChar('['); 11340a3958e0SEnrico Granata var_success = true; 11350a3958e0SEnrico Granata 11369fc1944eSEnrico Granata if (index_higher < 0) 11379fc1944eSEnrico Granata index_higher = vobj->GetNumChildren() - 1; 11380a3958e0SEnrico Granata 11390a3958e0SEnrico Granata for (;index_lower<=index_higher;index_lower++) 11400a3958e0SEnrico Granata { 1141fc7a7f3bSEnrico Granata ValueObject* item = ExpandIndexedExpression(target, 11429fc1944eSEnrico Granata index_lower, 11439fc1944eSEnrico Granata exe_ctx->frame, 1144fc7a7f3bSEnrico Granata false).get(); 11450a3958e0SEnrico Granata 1146fc7a7f3bSEnrico Granata if (!item) 1147fc7a7f3bSEnrico Granata { 1148fc7a7f3bSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 1149fc7a7f3bSEnrico Granata printf("ERROR\n"); 1150fc7a7f3bSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 1151fc7a7f3bSEnrico Granata } 1152fc7a7f3bSEnrico Granata #ifdef VERBOSE_FORMATPROMPT_OUTPUT 1153fc7a7f3bSEnrico Granata else 1154fc7a7f3bSEnrico Granata { 1155fc7a7f3bSEnrico Granata printf("special_directions: %s\n",special_directions); 1156fc7a7f3bSEnrico Granata } 1157fc7a7f3bSEnrico Granata #endif //VERBOSE_FORMATPROMPT_OUTPUT 1158fc7a7f3bSEnrico Granata 11590a3958e0SEnrico Granata if (!special_directions) 11609fc1944eSEnrico Granata var_success &= item->DumpPrintableRepresentation(s,val_obj_display, custom_format); 11610a3958e0SEnrico Granata else 11620a3958e0SEnrico Granata var_success &= FormatPrompt(special_directions, sc, exe_ctx, addr, s, NULL, item); 11630a3958e0SEnrico Granata 11640a3958e0SEnrico Granata if (index_lower < index_higher) 11650a3958e0SEnrico Granata s.PutChar(','); 11660a3958e0SEnrico Granata } 11670a3958e0SEnrico Granata s.PutChar(']'); 11684becb37eSEnrico Granata } 11694becb37eSEnrico Granata } 117034132754SGreg Clayton break; 11711b654882SGreg Clayton case 'a': 11721b654882SGreg Clayton if (::strncmp (var_name_begin, "addr}", strlen("addr}")) == 0) 11731b654882SGreg Clayton { 11741b654882SGreg Clayton if (addr && addr->IsValid()) 11751b654882SGreg Clayton { 11761b654882SGreg Clayton var_success = true; 11771b654882SGreg Clayton format_addr = *addr; 11781b654882SGreg Clayton } 11791b654882SGreg Clayton } 11801b654882SGreg Clayton break; 11811b654882SGreg Clayton 11821b654882SGreg Clayton case 'p': 11831b654882SGreg Clayton if (::strncmp (var_name_begin, "process.", strlen("process.")) == 0) 11841b654882SGreg Clayton { 11851b654882SGreg Clayton if (exe_ctx && exe_ctx->process != NULL) 11861b654882SGreg Clayton { 11871b654882SGreg Clayton var_name_begin += ::strlen ("process."); 11881b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 11891b654882SGreg Clayton { 11901b654882SGreg Clayton s.Printf("%i", exe_ctx->process->GetID()); 11911b654882SGreg Clayton var_success = true; 11921b654882SGreg Clayton } 11931b654882SGreg Clayton else if ((::strncmp (var_name_begin, "name}", strlen("name}")) == 0) || 11941b654882SGreg Clayton (::strncmp (var_name_begin, "file.basename}", strlen("file.basename}")) == 0) || 11951b654882SGreg Clayton (::strncmp (var_name_begin, "file.fullpath}", strlen("file.fullpath}")) == 0)) 11961b654882SGreg Clayton { 11971b654882SGreg Clayton ModuleSP exe_module_sp (exe_ctx->process->GetTarget().GetExecutableModule()); 11981b654882SGreg Clayton if (exe_module_sp) 11991b654882SGreg Clayton { 12001b654882SGreg Clayton if (var_name_begin[0] == 'n' || var_name_begin[5] == 'f') 12011b654882SGreg Clayton { 12021b654882SGreg Clayton format_file_spec.GetFilename() = exe_module_sp->GetFileSpec().GetFilename(); 12031b654882SGreg Clayton var_success = format_file_spec; 12041b654882SGreg Clayton } 12051b654882SGreg Clayton else 12061b654882SGreg Clayton { 12071b654882SGreg Clayton format_file_spec = exe_module_sp->GetFileSpec(); 12081b654882SGreg Clayton var_success = format_file_spec; 12091b654882SGreg Clayton } 12101b654882SGreg Clayton } 12111b654882SGreg Clayton } 12121b654882SGreg Clayton } 12131b654882SGreg Clayton } 12141b654882SGreg Clayton break; 12151b654882SGreg Clayton 12161b654882SGreg Clayton case 't': 12171b654882SGreg Clayton if (::strncmp (var_name_begin, "thread.", strlen("thread.")) == 0) 12181b654882SGreg Clayton { 12191b654882SGreg Clayton if (exe_ctx && exe_ctx->thread) 12201b654882SGreg Clayton { 12211b654882SGreg Clayton var_name_begin += ::strlen ("thread."); 12221b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 12231b654882SGreg Clayton { 12241b654882SGreg Clayton s.Printf("0x%4.4x", exe_ctx->thread->GetID()); 12251b654882SGreg Clayton var_success = true; 12261b654882SGreg Clayton } 12271b654882SGreg Clayton else if (::strncmp (var_name_begin, "index}", strlen("index}")) == 0) 12281b654882SGreg Clayton { 12291b654882SGreg Clayton s.Printf("%u", exe_ctx->thread->GetIndexID()); 12301b654882SGreg Clayton var_success = true; 12311b654882SGreg Clayton } 12321b654882SGreg Clayton else if (::strncmp (var_name_begin, "name}", strlen("name}")) == 0) 12331b654882SGreg Clayton { 12341b654882SGreg Clayton cstr = exe_ctx->thread->GetName(); 12351b654882SGreg Clayton var_success = cstr && cstr[0]; 12361b654882SGreg Clayton if (var_success) 12371b654882SGreg Clayton s.PutCString(cstr); 12381b654882SGreg Clayton } 12391b654882SGreg Clayton else if (::strncmp (var_name_begin, "queue}", strlen("queue}")) == 0) 12401b654882SGreg Clayton { 12411b654882SGreg Clayton cstr = exe_ctx->thread->GetQueueName(); 12421b654882SGreg Clayton var_success = cstr && cstr[0]; 12431b654882SGreg Clayton if (var_success) 12441b654882SGreg Clayton s.PutCString(cstr); 12451b654882SGreg Clayton } 12461b654882SGreg Clayton else if (::strncmp (var_name_begin, "stop-reason}", strlen("stop-reason}")) == 0) 12471b654882SGreg Clayton { 1248b15bfc75SJim Ingham StopInfoSP stop_info_sp = exe_ctx->thread->GetStopInfo (); 1249b15bfc75SJim Ingham if (stop_info_sp) 12501b654882SGreg Clayton { 1251b15bfc75SJim Ingham cstr = stop_info_sp->GetDescription(); 12521b654882SGreg Clayton if (cstr && cstr[0]) 12531b654882SGreg Clayton { 12541b654882SGreg Clayton s.PutCString(cstr); 12551b654882SGreg Clayton var_success = true; 12561b654882SGreg Clayton } 12571b654882SGreg Clayton } 12581b654882SGreg Clayton } 12591b654882SGreg Clayton } 12601b654882SGreg Clayton } 12611b654882SGreg Clayton else if (::strncmp (var_name_begin, "target.", strlen("target.")) == 0) 12621b654882SGreg Clayton { 12630603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 12640603aa9dSGreg Clayton if (target) 12651b654882SGreg Clayton { 12661b654882SGreg Clayton var_name_begin += ::strlen ("target."); 12671b654882SGreg Clayton if (::strncmp (var_name_begin, "arch}", strlen("arch}")) == 0) 12681b654882SGreg Clayton { 12691b654882SGreg Clayton ArchSpec arch (target->GetArchitecture ()); 12701b654882SGreg Clayton if (arch.IsValid()) 12711b654882SGreg Clayton { 127264195a2cSGreg Clayton s.PutCString (arch.GetArchitectureName()); 12731b654882SGreg Clayton var_success = true; 12741b654882SGreg Clayton } 12751b654882SGreg Clayton } 12761b654882SGreg Clayton } 12771b654882SGreg Clayton } 12781b654882SGreg Clayton break; 12791b654882SGreg Clayton 12801b654882SGreg Clayton 12811b654882SGreg Clayton case 'm': 12821b654882SGreg Clayton if (::strncmp (var_name_begin, "module.", strlen("module.")) == 0) 12831b654882SGreg Clayton { 12840603aa9dSGreg Clayton if (sc && sc->module_sp.get()) 12851b654882SGreg Clayton { 12860603aa9dSGreg Clayton Module *module = sc->module_sp.get(); 12871b654882SGreg Clayton var_name_begin += ::strlen ("module."); 12881b654882SGreg Clayton 12891b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 12901b654882SGreg Clayton { 12911b654882SGreg Clayton if (module->GetFileSpec()) 12921b654882SGreg Clayton { 12931b654882SGreg Clayton var_name_begin += ::strlen ("file."); 12941b654882SGreg Clayton 12951b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 12961b654882SGreg Clayton { 12971b654882SGreg Clayton format_file_spec.GetFilename() = module->GetFileSpec().GetFilename(); 12981b654882SGreg Clayton var_success = format_file_spec; 12991b654882SGreg Clayton } 13001b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 13011b654882SGreg Clayton { 13021b654882SGreg Clayton format_file_spec = module->GetFileSpec(); 13031b654882SGreg Clayton var_success = format_file_spec; 13041b654882SGreg Clayton } 13051b654882SGreg Clayton } 13061b654882SGreg Clayton } 13071b654882SGreg Clayton } 13081b654882SGreg Clayton } 13091b654882SGreg Clayton break; 13101b654882SGreg Clayton 13111b654882SGreg Clayton 13121b654882SGreg Clayton case 'f': 13131b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 13141b654882SGreg Clayton { 13151b654882SGreg Clayton if (sc && sc->comp_unit != NULL) 13161b654882SGreg Clayton { 13171b654882SGreg Clayton var_name_begin += ::strlen ("file."); 13181b654882SGreg Clayton 13191b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 13201b654882SGreg Clayton { 13211b654882SGreg Clayton format_file_spec.GetFilename() = sc->comp_unit->GetFilename(); 13221b654882SGreg Clayton var_success = format_file_spec; 13231b654882SGreg Clayton } 13241b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 13251b654882SGreg Clayton { 13261b654882SGreg Clayton format_file_spec = *sc->comp_unit; 13271b654882SGreg Clayton var_success = format_file_spec; 13281b654882SGreg Clayton } 13291b654882SGreg Clayton } 13301b654882SGreg Clayton } 13311b654882SGreg Clayton else if (::strncmp (var_name_begin, "frame.", strlen("frame.")) == 0) 13321b654882SGreg Clayton { 13331b654882SGreg Clayton if (exe_ctx && exe_ctx->frame) 13341b654882SGreg Clayton { 13351b654882SGreg Clayton var_name_begin += ::strlen ("frame."); 13361b654882SGreg Clayton if (::strncmp (var_name_begin, "index}", strlen("index}")) == 0) 13371b654882SGreg Clayton { 13381b654882SGreg Clayton s.Printf("%u", exe_ctx->frame->GetFrameIndex()); 13391b654882SGreg Clayton var_success = true; 13401b654882SGreg Clayton } 13411b654882SGreg Clayton else if (::strncmp (var_name_begin, "pc}", strlen("pc}")) == 0) 13421b654882SGreg Clayton { 13431b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 13441b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_PC; 13451b654882SGreg Clayton var_success = true; 13461b654882SGreg Clayton } 13471b654882SGreg Clayton else if (::strncmp (var_name_begin, "sp}", strlen("sp}")) == 0) 13481b654882SGreg Clayton { 13491b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 13501b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_SP; 13511b654882SGreg Clayton var_success = true; 13521b654882SGreg Clayton } 13531b654882SGreg Clayton else if (::strncmp (var_name_begin, "fp}", strlen("fp}")) == 0) 13541b654882SGreg Clayton { 13551b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 13561b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_FP; 13571b654882SGreg Clayton var_success = true; 13581b654882SGreg Clayton } 13591b654882SGreg Clayton else if (::strncmp (var_name_begin, "flags}", strlen("flags}")) == 0) 13601b654882SGreg Clayton { 13611b654882SGreg Clayton reg_kind = eRegisterKindGeneric; 13621b654882SGreg Clayton reg_num = LLDB_REGNUM_GENERIC_FLAGS; 13631b654882SGreg Clayton var_success = true; 13641b654882SGreg Clayton } 13651b654882SGreg Clayton else if (::strncmp (var_name_begin, "reg.", strlen ("reg.")) == 0) 13661b654882SGreg Clayton { 13675ccbd294SGreg Clayton reg_ctx = exe_ctx->frame->GetRegisterContext().get(); 13681b654882SGreg Clayton if (reg_ctx) 13691b654882SGreg Clayton { 13701b654882SGreg Clayton var_name_begin += ::strlen ("reg."); 13711b654882SGreg Clayton if (var_name_begin < var_name_end) 13721b654882SGreg Clayton { 13731b654882SGreg Clayton std::string reg_name (var_name_begin, var_name_end); 13741b654882SGreg Clayton reg_info = reg_ctx->GetRegisterInfoByName (reg_name.c_str()); 13751b654882SGreg Clayton if (reg_info) 13761b654882SGreg Clayton var_success = true; 13771b654882SGreg Clayton } 13781b654882SGreg Clayton } 13791b654882SGreg Clayton } 13801b654882SGreg Clayton } 13811b654882SGreg Clayton } 13821b654882SGreg Clayton else if (::strncmp (var_name_begin, "function.", strlen("function.")) == 0) 13831b654882SGreg Clayton { 13841b654882SGreg Clayton if (sc && (sc->function != NULL || sc->symbol != NULL)) 13851b654882SGreg Clayton { 13861b654882SGreg Clayton var_name_begin += ::strlen ("function."); 13871b654882SGreg Clayton if (::strncmp (var_name_begin, "id}", strlen("id}")) == 0) 13881b654882SGreg Clayton { 13891b654882SGreg Clayton if (sc->function) 13901b654882SGreg Clayton s.Printf("function{0x%8.8x}", sc->function->GetID()); 13911b654882SGreg Clayton else 13921b654882SGreg Clayton s.Printf("symbol[%u]", sc->symbol->GetID()); 13931b654882SGreg Clayton 13941b654882SGreg Clayton var_success = true; 13951b654882SGreg Clayton } 13961b654882SGreg Clayton else if (::strncmp (var_name_begin, "name}", strlen("name}")) == 0) 13971b654882SGreg Clayton { 13981b654882SGreg Clayton if (sc->function) 13991b654882SGreg Clayton cstr = sc->function->GetName().AsCString (NULL); 14001b654882SGreg Clayton else if (sc->symbol) 14011b654882SGreg Clayton cstr = sc->symbol->GetName().AsCString (NULL); 14021b654882SGreg Clayton if (cstr) 14031b654882SGreg Clayton { 14041b654882SGreg Clayton s.PutCString(cstr); 14050d9c9934SGreg Clayton 14060d9c9934SGreg Clayton if (sc->block) 14070d9c9934SGreg Clayton { 14080d9c9934SGreg Clayton Block *inline_block = sc->block->GetContainingInlinedBlock (); 14090d9c9934SGreg Clayton if (inline_block) 14100d9c9934SGreg Clayton { 14110d9c9934SGreg Clayton const InlineFunctionInfo *inline_info = sc->block->GetInlinedFunctionInfo(); 14120d9c9934SGreg Clayton if (inline_info) 14130d9c9934SGreg Clayton { 14140d9c9934SGreg Clayton s.PutCString(" [inlined] "); 14150d9c9934SGreg Clayton inline_info->GetName().Dump(&s); 14160d9c9934SGreg Clayton } 14170d9c9934SGreg Clayton } 14180d9c9934SGreg Clayton } 14191b654882SGreg Clayton var_success = true; 14201b654882SGreg Clayton } 14211b654882SGreg Clayton } 14221b654882SGreg Clayton else if (::strncmp (var_name_begin, "addr-offset}", strlen("addr-offset}")) == 0) 14231b654882SGreg Clayton { 14241b654882SGreg Clayton var_success = addr != NULL; 14251b654882SGreg Clayton if (var_success) 14261b654882SGreg Clayton { 14271b654882SGreg Clayton format_addr = *addr; 14281b654882SGreg Clayton calculate_format_addr_function_offset = true; 14291b654882SGreg Clayton } 14301b654882SGreg Clayton } 14311b654882SGreg Clayton else if (::strncmp (var_name_begin, "line-offset}", strlen("line-offset}")) == 0) 14321b654882SGreg Clayton { 14331b654882SGreg Clayton var_success = sc->line_entry.range.GetBaseAddress().IsValid(); 14341b654882SGreg Clayton if (var_success) 14351b654882SGreg Clayton { 14361b654882SGreg Clayton format_addr = sc->line_entry.range.GetBaseAddress(); 14371b654882SGreg Clayton calculate_format_addr_function_offset = true; 14381b654882SGreg Clayton } 14391b654882SGreg Clayton } 14401b654882SGreg Clayton else if (::strncmp (var_name_begin, "pc-offset}", strlen("pc-offset}")) == 0) 14411b654882SGreg Clayton { 14421b654882SGreg Clayton var_success = exe_ctx->frame; 14431b654882SGreg Clayton if (var_success) 14441b654882SGreg Clayton { 14451b654882SGreg Clayton format_addr = exe_ctx->frame->GetFrameCodeAddress(); 14461b654882SGreg Clayton calculate_format_addr_function_offset = true; 14471b654882SGreg Clayton } 14481b654882SGreg Clayton } 14491b654882SGreg Clayton } 14501b654882SGreg Clayton } 14511b654882SGreg Clayton break; 14521b654882SGreg Clayton 14531b654882SGreg Clayton case 'l': 14541b654882SGreg Clayton if (::strncmp (var_name_begin, "line.", strlen("line.")) == 0) 14551b654882SGreg Clayton { 14561b654882SGreg Clayton if (sc && sc->line_entry.IsValid()) 14571b654882SGreg Clayton { 14581b654882SGreg Clayton var_name_begin += ::strlen ("line."); 14591b654882SGreg Clayton if (::strncmp (var_name_begin, "file.", strlen("file.")) == 0) 14601b654882SGreg Clayton { 14611b654882SGreg Clayton var_name_begin += ::strlen ("file."); 14621b654882SGreg Clayton 14631b654882SGreg Clayton if (::strncmp (var_name_begin, "basename}", strlen("basename}")) == 0) 14641b654882SGreg Clayton { 14651b654882SGreg Clayton format_file_spec.GetFilename() = sc->line_entry.file.GetFilename(); 14661b654882SGreg Clayton var_success = format_file_spec; 14671b654882SGreg Clayton } 14681b654882SGreg Clayton else if (::strncmp (var_name_begin, "fullpath}", strlen("fullpath}")) == 0) 14691b654882SGreg Clayton { 14701b654882SGreg Clayton format_file_spec = sc->line_entry.file; 14711b654882SGreg Clayton var_success = format_file_spec; 14721b654882SGreg Clayton } 14731b654882SGreg Clayton } 14741b654882SGreg Clayton else if (::strncmp (var_name_begin, "number}", strlen("number}")) == 0) 14751b654882SGreg Clayton { 14761b654882SGreg Clayton var_success = true; 14771b654882SGreg Clayton s.Printf("%u", sc->line_entry.line); 14781b654882SGreg Clayton } 14791b654882SGreg Clayton else if ((::strncmp (var_name_begin, "start-addr}", strlen("start-addr}")) == 0) || 14801b654882SGreg Clayton (::strncmp (var_name_begin, "end-addr}", strlen("end-addr}")) == 0)) 14811b654882SGreg Clayton { 14821b654882SGreg Clayton var_success = sc && sc->line_entry.range.GetBaseAddress().IsValid(); 14831b654882SGreg Clayton if (var_success) 14841b654882SGreg Clayton { 14851b654882SGreg Clayton format_addr = sc->line_entry.range.GetBaseAddress(); 14861b654882SGreg Clayton if (var_name_begin[0] == 'e') 14871b654882SGreg Clayton format_addr.Slide (sc->line_entry.range.GetByteSize()); 14881b654882SGreg Clayton } 14891b654882SGreg Clayton } 14901b654882SGreg Clayton } 14911b654882SGreg Clayton } 14921b654882SGreg Clayton break; 14931b654882SGreg Clayton } 14941b654882SGreg Clayton 14951b654882SGreg Clayton if (var_success) 14961b654882SGreg Clayton { 14971b654882SGreg Clayton // If format addr is valid, then we need to print an address 14981b654882SGreg Clayton if (reg_num != LLDB_INVALID_REGNUM) 14991b654882SGreg Clayton { 15001b654882SGreg Clayton // We have a register value to display... 15011b654882SGreg Clayton if (reg_num == LLDB_REGNUM_GENERIC_PC && reg_kind == eRegisterKindGeneric) 15021b654882SGreg Clayton { 15031b654882SGreg Clayton format_addr = exe_ctx->frame->GetFrameCodeAddress(); 15041b654882SGreg Clayton } 15051b654882SGreg Clayton else 15061b654882SGreg Clayton { 15071b654882SGreg Clayton if (reg_ctx == NULL) 15085ccbd294SGreg Clayton reg_ctx = exe_ctx->frame->GetRegisterContext().get(); 15091b654882SGreg Clayton 15101b654882SGreg Clayton if (reg_ctx) 15111b654882SGreg Clayton { 15121b654882SGreg Clayton if (reg_kind != kNumRegisterKinds) 15131b654882SGreg Clayton reg_num = reg_ctx->ConvertRegisterKindToRegisterNumber(reg_kind, reg_num); 15141b654882SGreg Clayton reg_info = reg_ctx->GetRegisterInfoAtIndex (reg_num); 15151b654882SGreg Clayton var_success = reg_info != NULL; 15161b654882SGreg Clayton } 15171b654882SGreg Clayton } 15181b654882SGreg Clayton } 15191b654882SGreg Clayton 15201b654882SGreg Clayton if (reg_info != NULL) 15211b654882SGreg Clayton { 15227349bd90SGreg Clayton RegisterValue reg_value; 15237349bd90SGreg Clayton var_success = reg_ctx->ReadRegister (reg_info, reg_value); 15247349bd90SGreg Clayton if (var_success) 15251b654882SGreg Clayton { 15269a8fa916SGreg Clayton reg_value.Dump(&s, reg_info, false, false, eFormatDefault); 15271b654882SGreg Clayton } 15281b654882SGreg Clayton } 15291b654882SGreg Clayton 15301b654882SGreg Clayton if (format_file_spec) 15311b654882SGreg Clayton { 15321b654882SGreg Clayton s << format_file_spec; 15331b654882SGreg Clayton } 15341b654882SGreg Clayton 15351b654882SGreg Clayton // If format addr is valid, then we need to print an address 15361b654882SGreg Clayton if (format_addr.IsValid()) 15371b654882SGreg Clayton { 15380603aa9dSGreg Clayton var_success = false; 15390603aa9dSGreg Clayton 15401b654882SGreg Clayton if (calculate_format_addr_function_offset) 15411b654882SGreg Clayton { 15421b654882SGreg Clayton Address func_addr; 15430603aa9dSGreg Clayton 15440603aa9dSGreg Clayton if (sc) 15450603aa9dSGreg Clayton { 15461b654882SGreg Clayton if (sc->function) 15470d9c9934SGreg Clayton { 15481b654882SGreg Clayton func_addr = sc->function->GetAddressRange().GetBaseAddress(); 15490d9c9934SGreg Clayton if (sc->block) 15500d9c9934SGreg Clayton { 15510d9c9934SGreg Clayton // Check to make sure we aren't in an inline 15520d9c9934SGreg Clayton // function. If we are, use the inline block 15530d9c9934SGreg Clayton // range that contains "format_addr" since 15540d9c9934SGreg Clayton // blocks can be discontiguous. 15550d9c9934SGreg Clayton Block *inline_block = sc->block->GetContainingInlinedBlock (); 15560d9c9934SGreg Clayton AddressRange inline_range; 15570d9c9934SGreg Clayton if (inline_block && inline_block->GetRangeContainingAddress (format_addr, inline_range)) 15580d9c9934SGreg Clayton func_addr = inline_range.GetBaseAddress(); 15590d9c9934SGreg Clayton } 15600d9c9934SGreg Clayton } 15611b654882SGreg Clayton else if (sc->symbol && sc->symbol->GetAddressRangePtr()) 15621b654882SGreg Clayton func_addr = sc->symbol->GetAddressRangePtr()->GetBaseAddress(); 15630603aa9dSGreg Clayton } 15641b654882SGreg Clayton 15650603aa9dSGreg Clayton if (func_addr.IsValid()) 15661b654882SGreg Clayton { 15671b654882SGreg Clayton if (func_addr.GetSection() == format_addr.GetSection()) 15681b654882SGreg Clayton { 15691b654882SGreg Clayton addr_t func_file_addr = func_addr.GetFileAddress(); 15701b654882SGreg Clayton addr_t addr_file_addr = format_addr.GetFileAddress(); 15711b654882SGreg Clayton if (addr_file_addr > func_file_addr) 15721b654882SGreg Clayton s.Printf(" + %llu", addr_file_addr - func_file_addr); 15731b654882SGreg Clayton else if (addr_file_addr < func_file_addr) 15741b654882SGreg Clayton s.Printf(" - %llu", func_file_addr - addr_file_addr); 15750603aa9dSGreg Clayton var_success = true; 15761b654882SGreg Clayton } 15771b654882SGreg Clayton else 15780603aa9dSGreg Clayton { 15790603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 15800603aa9dSGreg Clayton if (target) 15810603aa9dSGreg Clayton { 15820603aa9dSGreg Clayton addr_t func_load_addr = func_addr.GetLoadAddress (target); 15830603aa9dSGreg Clayton addr_t addr_load_addr = format_addr.GetLoadAddress (target); 15840603aa9dSGreg Clayton if (addr_load_addr > func_load_addr) 15850603aa9dSGreg Clayton s.Printf(" + %llu", addr_load_addr - func_load_addr); 15860603aa9dSGreg Clayton else if (addr_load_addr < func_load_addr) 15870603aa9dSGreg Clayton s.Printf(" - %llu", func_load_addr - addr_load_addr); 15880603aa9dSGreg Clayton var_success = true; 15890603aa9dSGreg Clayton } 15900603aa9dSGreg Clayton } 15911b654882SGreg Clayton } 15921b654882SGreg Clayton } 15931b654882SGreg Clayton else 15941b654882SGreg Clayton { 15950603aa9dSGreg Clayton Target *target = Target::GetTargetFromContexts (exe_ctx, sc); 15961b654882SGreg Clayton addr_t vaddr = LLDB_INVALID_ADDRESS; 15970603aa9dSGreg Clayton if (exe_ctx && !target->GetSectionLoadList().IsEmpty()) 15980603aa9dSGreg Clayton vaddr = format_addr.GetLoadAddress (target); 15991b654882SGreg Clayton if (vaddr == LLDB_INVALID_ADDRESS) 16001b654882SGreg Clayton vaddr = format_addr.GetFileAddress (); 16011b654882SGreg Clayton 16021b654882SGreg Clayton if (vaddr != LLDB_INVALID_ADDRESS) 16030603aa9dSGreg Clayton { 1604514487e8SGreg Clayton int addr_width = target->GetArchitecture().GetAddressByteSize() * 2; 160535f1a0d5SGreg Clayton if (addr_width == 0) 160635f1a0d5SGreg Clayton addr_width = 16; 160735f1a0d5SGreg Clayton s.Printf("0x%*.*llx", addr_width, addr_width, vaddr); 16080603aa9dSGreg Clayton var_success = true; 16090603aa9dSGreg Clayton } 16101b654882SGreg Clayton } 16111b654882SGreg Clayton } 16121b654882SGreg Clayton } 16131b654882SGreg Clayton 16141b654882SGreg Clayton if (var_success == false) 16151b654882SGreg Clayton success = false; 16161b654882SGreg Clayton } 16171b654882SGreg Clayton p = var_name_end; 16181b654882SGreg Clayton } 16191b654882SGreg Clayton else 16201b654882SGreg Clayton break; 16211b654882SGreg Clayton } 16221b654882SGreg Clayton else 16231b654882SGreg Clayton { 16241b654882SGreg Clayton // We got a dollar sign with no '{' after it, it must just be a dollar sign 16251b654882SGreg Clayton s.PutChar(*p); 16261b654882SGreg Clayton } 16271b654882SGreg Clayton } 16281b654882SGreg Clayton else if (*p == '\\') 16291b654882SGreg Clayton { 16301b654882SGreg Clayton ++p; // skip the slash 16311b654882SGreg Clayton switch (*p) 16321b654882SGreg Clayton { 16331b654882SGreg Clayton case 'a': s.PutChar ('\a'); break; 16341b654882SGreg Clayton case 'b': s.PutChar ('\b'); break; 16351b654882SGreg Clayton case 'f': s.PutChar ('\f'); break; 16361b654882SGreg Clayton case 'n': s.PutChar ('\n'); break; 16371b654882SGreg Clayton case 'r': s.PutChar ('\r'); break; 16381b654882SGreg Clayton case 't': s.PutChar ('\t'); break; 16391b654882SGreg Clayton case 'v': s.PutChar ('\v'); break; 16401b654882SGreg Clayton case '\'': s.PutChar ('\''); break; 16411b654882SGreg Clayton case '\\': s.PutChar ('\\'); break; 16421b654882SGreg Clayton case '0': 16431b654882SGreg Clayton // 1 to 3 octal chars 16441b654882SGreg Clayton { 16450603aa9dSGreg Clayton // Make a string that can hold onto the initial zero char, 16460603aa9dSGreg Clayton // up to 3 octal digits, and a terminating NULL. 16470603aa9dSGreg Clayton char oct_str[5] = { 0, 0, 0, 0, 0 }; 16480603aa9dSGreg Clayton 16490603aa9dSGreg Clayton int i; 16500603aa9dSGreg Clayton for (i=0; (p[i] >= '0' && p[i] <= '7') && i<4; ++i) 16510603aa9dSGreg Clayton oct_str[i] = p[i]; 16520603aa9dSGreg Clayton 16530603aa9dSGreg Clayton // We don't want to consume the last octal character since 16540603aa9dSGreg Clayton // the main for loop will do this for us, so we advance p by 16550603aa9dSGreg Clayton // one less than i (even if i is zero) 16560603aa9dSGreg Clayton p += i - 1; 16570603aa9dSGreg Clayton unsigned long octal_value = ::strtoul (oct_str, NULL, 8); 16580603aa9dSGreg Clayton if (octal_value <= UINT8_MAX) 16591b654882SGreg Clayton { 16600603aa9dSGreg Clayton char octal_char = octal_value; 16610603aa9dSGreg Clayton s.Write (&octal_char, 1); 16621b654882SGreg Clayton } 16631b654882SGreg Clayton } 16641b654882SGreg Clayton break; 16651b654882SGreg Clayton 16661b654882SGreg Clayton case 'x': 16671b654882SGreg Clayton // hex number in the format 16680603aa9dSGreg Clayton if (isxdigit(p[1])) 16691b654882SGreg Clayton { 16700603aa9dSGreg Clayton ++p; // Skip the 'x' 16711b654882SGreg Clayton 16720603aa9dSGreg Clayton // Make a string that can hold onto two hex chars plus a 16730603aa9dSGreg Clayton // NULL terminator 16741b654882SGreg Clayton char hex_str[3] = { 0,0,0 }; 16751b654882SGreg Clayton hex_str[0] = *p; 16760603aa9dSGreg Clayton if (isxdigit(p[1])) 16770603aa9dSGreg Clayton { 16780603aa9dSGreg Clayton ++p; // Skip the first of the two hex chars 16791b654882SGreg Clayton hex_str[1] = *p; 16800603aa9dSGreg Clayton } 16810603aa9dSGreg Clayton 16821b654882SGreg Clayton unsigned long hex_value = strtoul (hex_str, NULL, 16); 16830603aa9dSGreg Clayton if (hex_value <= UINT8_MAX) 16841b654882SGreg Clayton s.PutChar (hex_value); 16851b654882SGreg Clayton } 16861b654882SGreg Clayton else 16871b654882SGreg Clayton { 16880603aa9dSGreg Clayton s.PutChar('x'); 16891b654882SGreg Clayton } 16901b654882SGreg Clayton break; 16911b654882SGreg Clayton 16921b654882SGreg Clayton default: 16930603aa9dSGreg Clayton // Just desensitize any other character by just printing what 16940603aa9dSGreg Clayton // came after the '\' 16950603aa9dSGreg Clayton s << *p; 16961b654882SGreg Clayton break; 16971b654882SGreg Clayton 16981b654882SGreg Clayton } 16991b654882SGreg Clayton 17001b654882SGreg Clayton } 17011b654882SGreg Clayton } 17021b654882SGreg Clayton if (end) 17031b654882SGreg Clayton *end = p; 17041b654882SGreg Clayton return success; 17051b654882SGreg Clayton } 17061b654882SGreg Clayton 17074a33d318SGreg Clayton 17084a33d318SGreg Clayton static FormatManager& 17094a33d318SGreg Clayton GetFormatManager() { 17104a33d318SGreg Clayton static FormatManager g_format_manager; 17114a33d318SGreg Clayton return g_format_manager; 17124a33d318SGreg Clayton } 17134a33d318SGreg Clayton 17144a33d318SGreg Clayton bool 17154becb37eSEnrico Granata Debugger::ValueFormats::Get(ValueObject& vobj, ValueFormat::SharedPointer &entry) 17164a33d318SGreg Clayton { 17174becb37eSEnrico Granata return GetFormatManager().Value().Get(vobj,entry); 17184a33d318SGreg Clayton } 17194a33d318SGreg Clayton 17204a33d318SGreg Clayton void 17214becb37eSEnrico Granata Debugger::ValueFormats::Add(const ConstString &type, const ValueFormat::SharedPointer &entry) 17224a33d318SGreg Clayton { 17234becb37eSEnrico Granata GetFormatManager().Value().Add(type.AsCString(),entry); 17244a33d318SGreg Clayton } 17254a33d318SGreg Clayton 17264a33d318SGreg Clayton bool 17274becb37eSEnrico Granata Debugger::ValueFormats::Delete(const ConstString &type) 17284a33d318SGreg Clayton { 17294becb37eSEnrico Granata return GetFormatManager().Value().Delete(type.AsCString()); 17304a33d318SGreg Clayton } 17314a33d318SGreg Clayton 17324a33d318SGreg Clayton void 17334becb37eSEnrico Granata Debugger::ValueFormats::Clear() 17344a33d318SGreg Clayton { 17354becb37eSEnrico Granata GetFormatManager().Value().Clear(); 17364becb37eSEnrico Granata } 17374becb37eSEnrico Granata 17384becb37eSEnrico Granata void 17390a3958e0SEnrico Granata Debugger::ValueFormats::LoopThrough(ValueFormat::ValueCallback callback, void* callback_baton) 17404becb37eSEnrico Granata { 17414becb37eSEnrico Granata GetFormatManager().Value().LoopThrough(callback, callback_baton); 17424becb37eSEnrico Granata } 17434becb37eSEnrico Granata 17444becb37eSEnrico Granata uint32_t 17454becb37eSEnrico Granata Debugger::ValueFormats::GetCurrentRevision() 17464becb37eSEnrico Granata { 17474becb37eSEnrico Granata return GetFormatManager().GetCurrentRevision(); 17484becb37eSEnrico Granata } 17494becb37eSEnrico Granata 17500a3958e0SEnrico Granata uint32_t 17510a3958e0SEnrico Granata Debugger::ValueFormats::GetCount() 17520a3958e0SEnrico Granata { 17530a3958e0SEnrico Granata return GetFormatManager().Value().GetCount(); 17540a3958e0SEnrico Granata } 17554becb37eSEnrico Granata 17564becb37eSEnrico Granata bool 17574becb37eSEnrico Granata Debugger::SummaryFormats::Get(ValueObject& vobj, SummaryFormat::SharedPointer &entry) 17584becb37eSEnrico Granata { 17594becb37eSEnrico Granata return GetFormatManager().Summary().Get(vobj,entry); 17604becb37eSEnrico Granata } 17614becb37eSEnrico Granata 17624becb37eSEnrico Granata void 17634becb37eSEnrico Granata Debugger::SummaryFormats::Add(const ConstString &type, const SummaryFormat::SharedPointer &entry) 17644becb37eSEnrico Granata { 17654becb37eSEnrico Granata GetFormatManager().Summary().Add(type.AsCString(),entry); 17664becb37eSEnrico Granata } 17674becb37eSEnrico Granata 17684becb37eSEnrico Granata bool 17694becb37eSEnrico Granata Debugger::SummaryFormats::Delete(const ConstString &type) 17704becb37eSEnrico Granata { 17714becb37eSEnrico Granata return GetFormatManager().Summary().Delete(type.AsCString()); 17724becb37eSEnrico Granata } 17734becb37eSEnrico Granata 17744becb37eSEnrico Granata void 17754becb37eSEnrico Granata Debugger::SummaryFormats::Clear() 17764becb37eSEnrico Granata { 17774becb37eSEnrico Granata GetFormatManager().Summary().Clear(); 17784becb37eSEnrico Granata } 17794becb37eSEnrico Granata 17804becb37eSEnrico Granata void 17810a3958e0SEnrico Granata Debugger::SummaryFormats::LoopThrough(SummaryFormat::SummaryCallback callback, void* callback_baton) 17824becb37eSEnrico Granata { 17834becb37eSEnrico Granata GetFormatManager().Summary().LoopThrough(callback, callback_baton); 17844becb37eSEnrico Granata } 17854becb37eSEnrico Granata 17864becb37eSEnrico Granata uint32_t 17874becb37eSEnrico Granata Debugger::SummaryFormats::GetCurrentRevision() 17884becb37eSEnrico Granata { 17894becb37eSEnrico Granata return GetFormatManager().GetCurrentRevision(); 17904a33d318SGreg Clayton } 17914a33d318SGreg Clayton 17920a3958e0SEnrico Granata uint32_t 17930a3958e0SEnrico Granata Debugger::SummaryFormats::GetCount() 17940a3958e0SEnrico Granata { 17950a3958e0SEnrico Granata return GetFormatManager().Summary().GetCount(); 17960a3958e0SEnrico Granata } 17970a3958e0SEnrico Granata 17980a3958e0SEnrico Granata bool 17990a3958e0SEnrico Granata Debugger::RegexSummaryFormats::Get(ValueObject& vobj, SummaryFormat::SharedPointer &entry) 18000a3958e0SEnrico Granata { 18010a3958e0SEnrico Granata return GetFormatManager().RegexSummary().Get(vobj,entry); 18020a3958e0SEnrico Granata } 18030a3958e0SEnrico Granata 18040a3958e0SEnrico Granata void 18050a3958e0SEnrico Granata Debugger::RegexSummaryFormats::Add(const lldb::RegularExpressionSP &type, const SummaryFormat::SharedPointer &entry) 18060a3958e0SEnrico Granata { 18070a3958e0SEnrico Granata GetFormatManager().RegexSummary().Add(type,entry); 18080a3958e0SEnrico Granata } 18090a3958e0SEnrico Granata 18100a3958e0SEnrico Granata bool 18110a3958e0SEnrico Granata Debugger::RegexSummaryFormats::Delete(const ConstString &type) 18120a3958e0SEnrico Granata { 18130a3958e0SEnrico Granata return GetFormatManager().RegexSummary().Delete(type.AsCString()); 18140a3958e0SEnrico Granata } 18150a3958e0SEnrico Granata 18160a3958e0SEnrico Granata void 18170a3958e0SEnrico Granata Debugger::RegexSummaryFormats::Clear() 18180a3958e0SEnrico Granata { 18190a3958e0SEnrico Granata GetFormatManager().RegexSummary().Clear(); 18200a3958e0SEnrico Granata } 18210a3958e0SEnrico Granata 18220a3958e0SEnrico Granata void 18230a3958e0SEnrico Granata Debugger::RegexSummaryFormats::LoopThrough(SummaryFormat::RegexSummaryCallback callback, void* callback_baton) 18240a3958e0SEnrico Granata { 18250a3958e0SEnrico Granata GetFormatManager().RegexSummary().LoopThrough(callback, callback_baton); 18260a3958e0SEnrico Granata } 18270a3958e0SEnrico Granata 18280a3958e0SEnrico Granata uint32_t 18290a3958e0SEnrico Granata Debugger::RegexSummaryFormats::GetCurrentRevision() 18300a3958e0SEnrico Granata { 18310a3958e0SEnrico Granata return GetFormatManager().GetCurrentRevision(); 18320a3958e0SEnrico Granata } 18330a3958e0SEnrico Granata 18340a3958e0SEnrico Granata uint32_t 18350a3958e0SEnrico Granata Debugger::RegexSummaryFormats::GetCount() 18360a3958e0SEnrico Granata { 18370a3958e0SEnrico Granata return GetFormatManager().RegexSummary().GetCount(); 18380a3958e0SEnrico Granata } 18390a3958e0SEnrico Granata 1840f9fa6ee5SEnrico Granata bool 1841f9fa6ee5SEnrico Granata Debugger::NamedSummaryFormats::Get(const ConstString &type, SummaryFormat::SharedPointer &entry) 1842f9fa6ee5SEnrico Granata { 1843f9fa6ee5SEnrico Granata return GetFormatManager().NamedSummary().Get(type.AsCString(),entry); 1844f9fa6ee5SEnrico Granata } 1845f9fa6ee5SEnrico Granata 1846f9fa6ee5SEnrico Granata void 1847f9fa6ee5SEnrico Granata Debugger::NamedSummaryFormats::Add(const ConstString &type, const SummaryFormat::SharedPointer &entry) 1848f9fa6ee5SEnrico Granata { 1849f9fa6ee5SEnrico Granata GetFormatManager().NamedSummary().Add(type.AsCString(),entry); 1850f9fa6ee5SEnrico Granata } 1851f9fa6ee5SEnrico Granata 1852f9fa6ee5SEnrico Granata bool 1853f9fa6ee5SEnrico Granata Debugger::NamedSummaryFormats::Delete(const ConstString &type) 1854f9fa6ee5SEnrico Granata { 1855f9fa6ee5SEnrico Granata return GetFormatManager().NamedSummary().Delete(type.AsCString()); 1856f9fa6ee5SEnrico Granata } 1857f9fa6ee5SEnrico Granata 1858f9fa6ee5SEnrico Granata void 1859f9fa6ee5SEnrico Granata Debugger::NamedSummaryFormats::Clear() 1860f9fa6ee5SEnrico Granata { 1861f9fa6ee5SEnrico Granata GetFormatManager().NamedSummary().Clear(); 1862f9fa6ee5SEnrico Granata } 1863f9fa6ee5SEnrico Granata 1864f9fa6ee5SEnrico Granata void 1865f9fa6ee5SEnrico Granata Debugger::NamedSummaryFormats::LoopThrough(SummaryFormat::SummaryCallback callback, void* callback_baton) 1866f9fa6ee5SEnrico Granata { 1867f9fa6ee5SEnrico Granata GetFormatManager().NamedSummary().LoopThrough(callback, callback_baton); 1868f9fa6ee5SEnrico Granata } 1869f9fa6ee5SEnrico Granata 1870f9fa6ee5SEnrico Granata uint32_t 1871f9fa6ee5SEnrico Granata Debugger::NamedSummaryFormats::GetCurrentRevision() 1872f9fa6ee5SEnrico Granata { 1873f9fa6ee5SEnrico Granata return GetFormatManager().GetCurrentRevision(); 1874f9fa6ee5SEnrico Granata } 1875f9fa6ee5SEnrico Granata 1876f9fa6ee5SEnrico Granata uint32_t 1877f9fa6ee5SEnrico Granata Debugger::NamedSummaryFormats::GetCount() 1878f9fa6ee5SEnrico Granata { 1879f9fa6ee5SEnrico Granata return GetFormatManager().NamedSummary().GetCount(); 1880f9fa6ee5SEnrico Granata } 1881f9fa6ee5SEnrico Granata 18821b654882SGreg Clayton #pragma mark Debugger::SettingsController 18831b654882SGreg Clayton 18843df9a8dfSCaroline Tice //-------------------------------------------------- 18851b654882SGreg Clayton // class Debugger::SettingsController 18863df9a8dfSCaroline Tice //-------------------------------------------------- 18873df9a8dfSCaroline Tice 18881b654882SGreg Clayton Debugger::SettingsController::SettingsController () : 1889101c7c20SCaroline Tice UserSettingsController ("", lldb::UserSettingsControllerSP()) 18903df9a8dfSCaroline Tice { 189191123da2SCaroline Tice m_default_settings.reset (new DebuggerInstanceSettings (*this, false, 189291123da2SCaroline Tice InstanceSettings::GetDefaultName().AsCString())); 18933df9a8dfSCaroline Tice } 18943df9a8dfSCaroline Tice 18951b654882SGreg Clayton Debugger::SettingsController::~SettingsController () 18963df9a8dfSCaroline Tice { 18973df9a8dfSCaroline Tice } 18983df9a8dfSCaroline Tice 18993df9a8dfSCaroline Tice 19003df9a8dfSCaroline Tice lldb::InstanceSettingsSP 19011b654882SGreg Clayton Debugger::SettingsController::CreateInstanceSettings (const char *instance_name) 19023df9a8dfSCaroline Tice { 1903dbe54508SGreg Clayton DebuggerInstanceSettings *new_settings = new DebuggerInstanceSettings (*GetSettingsController(), 190491123da2SCaroline Tice false, instance_name); 19053df9a8dfSCaroline Tice lldb::InstanceSettingsSP new_settings_sp (new_settings); 19063df9a8dfSCaroline Tice return new_settings_sp; 19073df9a8dfSCaroline Tice } 19083df9a8dfSCaroline Tice 19091b654882SGreg Clayton #pragma mark DebuggerInstanceSettings 19103df9a8dfSCaroline Tice //-------------------------------------------------- 19113df9a8dfSCaroline Tice // class DebuggerInstanceSettings 19123df9a8dfSCaroline Tice //-------------------------------------------------- 19133df9a8dfSCaroline Tice 1914a7015092SGreg Clayton DebuggerInstanceSettings::DebuggerInstanceSettings 1915a7015092SGreg Clayton ( 1916a7015092SGreg Clayton UserSettingsController &owner, 1917a7015092SGreg Clayton bool live_instance, 1918a7015092SGreg Clayton const char *name 1919a7015092SGreg Clayton ) : 192085851ddeSGreg Clayton InstanceSettings (owner, name ? name : InstanceSettings::InvalidName().AsCString(), live_instance), 1921a7015092SGreg Clayton m_term_width (80), 19223df9a8dfSCaroline Tice m_prompt (), 19230603aa9dSGreg Clayton m_frame_format (), 19240603aa9dSGreg Clayton m_thread_format (), 1925daccaa9eSCaroline Tice m_script_lang (), 19263bcdb29cSJim Ingham m_use_external_editor (false), 19273bcdb29cSJim Ingham m_auto_confirm_on (false) 19283df9a8dfSCaroline Tice { 1929f20e8239SCaroline Tice // CopyInstanceSettings is a pure virtual function in InstanceSettings; it therefore cannot be called 1930f20e8239SCaroline Tice // until the vtables for DebuggerInstanceSettings are properly set up, i.e. AFTER all the initializers. 1931f20e8239SCaroline Tice // For this reason it has to be called here, rather than in the initializer or in the parent constructor. 19329e41c15dSCaroline Tice // The same is true of CreateInstanceName(). 19339e41c15dSCaroline Tice 19349e41c15dSCaroline Tice if (GetInstanceName() == InstanceSettings::InvalidName()) 19359e41c15dSCaroline Tice { 19369e41c15dSCaroline Tice ChangeInstanceName (std::string (CreateInstanceName().AsCString())); 19379e41c15dSCaroline Tice m_owner.RegisterInstanceSettings (this); 19389e41c15dSCaroline Tice } 1939f20e8239SCaroline Tice 1940f20e8239SCaroline Tice if (live_instance) 19413df9a8dfSCaroline Tice { 19423df9a8dfSCaroline Tice const lldb::InstanceSettingsSP &pending_settings = m_owner.FindPendingSettings (m_instance_name); 19433df9a8dfSCaroline Tice CopyInstanceSettings (pending_settings, false); 19443df9a8dfSCaroline Tice } 19453df9a8dfSCaroline Tice } 19463df9a8dfSCaroline Tice 19473df9a8dfSCaroline Tice DebuggerInstanceSettings::DebuggerInstanceSettings (const DebuggerInstanceSettings &rhs) : 194899d0faf2SGreg Clayton InstanceSettings (*Debugger::GetSettingsController(), CreateInstanceName ().AsCString()), 19493df9a8dfSCaroline Tice m_prompt (rhs.m_prompt), 19500603aa9dSGreg Clayton m_frame_format (rhs.m_frame_format), 19510603aa9dSGreg Clayton m_thread_format (rhs.m_thread_format), 1952daccaa9eSCaroline Tice m_script_lang (rhs.m_script_lang), 19533bcdb29cSJim Ingham m_use_external_editor (rhs.m_use_external_editor), 19543bcdb29cSJim Ingham m_auto_confirm_on(rhs.m_auto_confirm_on) 19553df9a8dfSCaroline Tice { 19563df9a8dfSCaroline Tice const lldb::InstanceSettingsSP &pending_settings = m_owner.FindPendingSettings (m_instance_name); 19573df9a8dfSCaroline Tice CopyInstanceSettings (pending_settings, false); 19583df9a8dfSCaroline Tice m_owner.RemovePendingSettings (m_instance_name); 19593df9a8dfSCaroline Tice } 19603df9a8dfSCaroline Tice 19613df9a8dfSCaroline Tice DebuggerInstanceSettings::~DebuggerInstanceSettings () 19623df9a8dfSCaroline Tice { 19633df9a8dfSCaroline Tice } 19643df9a8dfSCaroline Tice 19653df9a8dfSCaroline Tice DebuggerInstanceSettings& 19663df9a8dfSCaroline Tice DebuggerInstanceSettings::operator= (const DebuggerInstanceSettings &rhs) 19673df9a8dfSCaroline Tice { 19683df9a8dfSCaroline Tice if (this != &rhs) 19693df9a8dfSCaroline Tice { 19701b654882SGreg Clayton m_term_width = rhs.m_term_width; 19713df9a8dfSCaroline Tice m_prompt = rhs.m_prompt; 19720603aa9dSGreg Clayton m_frame_format = rhs.m_frame_format; 19730603aa9dSGreg Clayton m_thread_format = rhs.m_thread_format; 19743df9a8dfSCaroline Tice m_script_lang = rhs.m_script_lang; 1975daccaa9eSCaroline Tice m_use_external_editor = rhs.m_use_external_editor; 19763bcdb29cSJim Ingham m_auto_confirm_on = rhs.m_auto_confirm_on; 19773df9a8dfSCaroline Tice } 19783df9a8dfSCaroline Tice 19793df9a8dfSCaroline Tice return *this; 19803df9a8dfSCaroline Tice } 19813df9a8dfSCaroline Tice 19821b654882SGreg Clayton bool 19831b654882SGreg Clayton DebuggerInstanceSettings::ValidTermWidthValue (const char *value, Error err) 19841b654882SGreg Clayton { 19851b654882SGreg Clayton bool valid = false; 19861b654882SGreg Clayton 19871b654882SGreg Clayton // Verify we have a value string. 19881b654882SGreg Clayton if (value == NULL || value[0] == '\0') 19891b654882SGreg Clayton { 19901b654882SGreg Clayton err.SetErrorString ("Missing value. Can't set terminal width without a value.\n"); 19911b654882SGreg Clayton } 19921b654882SGreg Clayton else 19931b654882SGreg Clayton { 19941b654882SGreg Clayton char *end = NULL; 19951b654882SGreg Clayton const uint32_t width = ::strtoul (value, &end, 0); 19961b654882SGreg Clayton 1997ea9fc181SJohnny Chen if (end && end[0] == '\0') 19981b654882SGreg Clayton { 1999433d7741SJohnny Chen if (width >= 10 && width <= 1024) 20001b654882SGreg Clayton valid = true; 20011b654882SGreg Clayton else 20021b654882SGreg Clayton err.SetErrorString ("Invalid term-width value; value must be between 10 and 1024.\n"); 20031b654882SGreg Clayton } 20041b654882SGreg Clayton else 20051b654882SGreg Clayton err.SetErrorStringWithFormat ("'%s' is not a valid unsigned integer string.\n", value); 20061b654882SGreg Clayton } 20071b654882SGreg Clayton 20081b654882SGreg Clayton return valid; 20091b654882SGreg Clayton } 20101b654882SGreg Clayton 20111b654882SGreg Clayton 20123df9a8dfSCaroline Tice void 20133df9a8dfSCaroline Tice DebuggerInstanceSettings::UpdateInstanceSettingsVariable (const ConstString &var_name, 20143df9a8dfSCaroline Tice const char *index_value, 20153df9a8dfSCaroline Tice const char *value, 20163df9a8dfSCaroline Tice const ConstString &instance_name, 20173df9a8dfSCaroline Tice const SettingEntry &entry, 2018e0d378b3SGreg Clayton VarSetOperationType op, 20193df9a8dfSCaroline Tice Error &err, 20203df9a8dfSCaroline Tice bool pending) 20213df9a8dfSCaroline Tice { 20220603aa9dSGreg Clayton 20230603aa9dSGreg Clayton if (var_name == TermWidthVarName()) 20240603aa9dSGreg Clayton { 20250603aa9dSGreg Clayton if (ValidTermWidthValue (value, err)) 20260603aa9dSGreg Clayton { 20270603aa9dSGreg Clayton m_term_width = ::strtoul (value, NULL, 0); 20280603aa9dSGreg Clayton } 20290603aa9dSGreg Clayton } 20300603aa9dSGreg Clayton else if (var_name == PromptVarName()) 20313df9a8dfSCaroline Tice { 20323df9a8dfSCaroline Tice UserSettingsController::UpdateStringVariable (op, m_prompt, value, err); 20333df9a8dfSCaroline Tice if (!pending) 20343df9a8dfSCaroline Tice { 203549e2737eSCaroline Tice // 'instance_name' is actually (probably) in the form '[<instance_name>]'; if so, we need to 203649e2737eSCaroline Tice // strip off the brackets before passing it to BroadcastPromptChange. 203749e2737eSCaroline Tice 203849e2737eSCaroline Tice std::string tmp_instance_name (instance_name.AsCString()); 203949e2737eSCaroline Tice if ((tmp_instance_name[0] == '[') 204049e2737eSCaroline Tice && (tmp_instance_name[instance_name.GetLength() - 1] == ']')) 204149e2737eSCaroline Tice tmp_instance_name = tmp_instance_name.substr (1, instance_name.GetLength() - 2); 204249e2737eSCaroline Tice ConstString new_name (tmp_instance_name.c_str()); 204349e2737eSCaroline Tice 204449e2737eSCaroline Tice BroadcastPromptChange (new_name, m_prompt.c_str()); 20453df9a8dfSCaroline Tice } 20463df9a8dfSCaroline Tice } 20470603aa9dSGreg Clayton else if (var_name == GetFrameFormatName()) 20480603aa9dSGreg Clayton { 20490603aa9dSGreg Clayton UserSettingsController::UpdateStringVariable (op, m_frame_format, value, err); 20500603aa9dSGreg Clayton } 20510603aa9dSGreg Clayton else if (var_name == GetThreadFormatName()) 20520603aa9dSGreg Clayton { 20530603aa9dSGreg Clayton UserSettingsController::UpdateStringVariable (op, m_thread_format, value, err); 20540603aa9dSGreg Clayton } 20553df9a8dfSCaroline Tice else if (var_name == ScriptLangVarName()) 20563df9a8dfSCaroline Tice { 20573df9a8dfSCaroline Tice bool success; 20583df9a8dfSCaroline Tice m_script_lang = Args::StringToScriptLanguage (value, eScriptLanguageDefault, 20593df9a8dfSCaroline Tice &success); 20603df9a8dfSCaroline Tice } 2061daccaa9eSCaroline Tice else if (var_name == UseExternalEditorVarName ()) 2062daccaa9eSCaroline Tice { 2063385aa28cSGreg Clayton UserSettingsController::UpdateBooleanVariable (op, m_use_external_editor, value, false, err); 2064daccaa9eSCaroline Tice } 20653bcdb29cSJim Ingham else if (var_name == AutoConfirmName ()) 20663bcdb29cSJim Ingham { 2067385aa28cSGreg Clayton UserSettingsController::UpdateBooleanVariable (op, m_auto_confirm_on, value, false, err); 20683bcdb29cSJim Ingham } 20693df9a8dfSCaroline Tice } 20703df9a8dfSCaroline Tice 207112cecd74SCaroline Tice bool 20723df9a8dfSCaroline Tice DebuggerInstanceSettings::GetInstanceSettingsValue (const SettingEntry &entry, 20733df9a8dfSCaroline Tice const ConstString &var_name, 2074daccaa9eSCaroline Tice StringList &value, 207512cecd74SCaroline Tice Error *err) 20763df9a8dfSCaroline Tice { 20773df9a8dfSCaroline Tice if (var_name == PromptVarName()) 20783df9a8dfSCaroline Tice { 20790603aa9dSGreg Clayton value.AppendString (m_prompt.c_str(), m_prompt.size()); 20803df9a8dfSCaroline Tice 20813df9a8dfSCaroline Tice } 20823df9a8dfSCaroline Tice else if (var_name == ScriptLangVarName()) 20833df9a8dfSCaroline Tice { 20843df9a8dfSCaroline Tice value.AppendString (ScriptInterpreter::LanguageToString (m_script_lang).c_str()); 20853df9a8dfSCaroline Tice } 2086101c7c20SCaroline Tice else if (var_name == TermWidthVarName()) 2087101c7c20SCaroline Tice { 2088101c7c20SCaroline Tice StreamString width_str; 2089101c7c20SCaroline Tice width_str.Printf ("%d", m_term_width); 2090101c7c20SCaroline Tice value.AppendString (width_str.GetData()); 2091101c7c20SCaroline Tice } 20920603aa9dSGreg Clayton else if (var_name == GetFrameFormatName ()) 20930603aa9dSGreg Clayton { 20940603aa9dSGreg Clayton value.AppendString(m_frame_format.c_str(), m_frame_format.size()); 20950603aa9dSGreg Clayton } 20960603aa9dSGreg Clayton else if (var_name == GetThreadFormatName ()) 20970603aa9dSGreg Clayton { 20980603aa9dSGreg Clayton value.AppendString(m_thread_format.c_str(), m_thread_format.size()); 20990603aa9dSGreg Clayton } 2100daccaa9eSCaroline Tice else if (var_name == UseExternalEditorVarName()) 2101daccaa9eSCaroline Tice { 2102daccaa9eSCaroline Tice if (m_use_external_editor) 2103daccaa9eSCaroline Tice value.AppendString ("true"); 2104daccaa9eSCaroline Tice else 2105daccaa9eSCaroline Tice value.AppendString ("false"); 2106daccaa9eSCaroline Tice } 21073bcdb29cSJim Ingham else if (var_name == AutoConfirmName()) 21083bcdb29cSJim Ingham { 21093bcdb29cSJim Ingham if (m_auto_confirm_on) 21103bcdb29cSJim Ingham value.AppendString ("true"); 21113bcdb29cSJim Ingham else 21123bcdb29cSJim Ingham value.AppendString ("false"); 21133bcdb29cSJim Ingham } 2114daccaa9eSCaroline Tice else 211512cecd74SCaroline Tice { 211612cecd74SCaroline Tice if (err) 211712cecd74SCaroline Tice err->SetErrorStringWithFormat ("unrecognized variable name '%s'", var_name.AsCString()); 211812cecd74SCaroline Tice return false; 211912cecd74SCaroline Tice } 212012cecd74SCaroline Tice return true; 21213df9a8dfSCaroline Tice } 21223df9a8dfSCaroline Tice 21233df9a8dfSCaroline Tice void 21243df9a8dfSCaroline Tice DebuggerInstanceSettings::CopyInstanceSettings (const lldb::InstanceSettingsSP &new_settings, 21253df9a8dfSCaroline Tice bool pending) 21263df9a8dfSCaroline Tice { 21273df9a8dfSCaroline Tice if (new_settings.get() == NULL) 21283df9a8dfSCaroline Tice return; 21293df9a8dfSCaroline Tice 21303df9a8dfSCaroline Tice DebuggerInstanceSettings *new_debugger_settings = (DebuggerInstanceSettings *) new_settings.get(); 21313df9a8dfSCaroline Tice 21323df9a8dfSCaroline Tice m_prompt = new_debugger_settings->m_prompt; 21333df9a8dfSCaroline Tice if (!pending) 213449e2737eSCaroline Tice { 213549e2737eSCaroline Tice // 'instance_name' is actually (probably) in the form '[<instance_name>]'; if so, we need to 213649e2737eSCaroline Tice // strip off the brackets before passing it to BroadcastPromptChange. 213749e2737eSCaroline Tice 213849e2737eSCaroline Tice std::string tmp_instance_name (m_instance_name.AsCString()); 213949e2737eSCaroline Tice if ((tmp_instance_name[0] == '[') 214049e2737eSCaroline Tice && (tmp_instance_name[m_instance_name.GetLength() - 1] == ']')) 214149e2737eSCaroline Tice tmp_instance_name = tmp_instance_name.substr (1, m_instance_name.GetLength() - 2); 214249e2737eSCaroline Tice ConstString new_name (tmp_instance_name.c_str()); 214349e2737eSCaroline Tice 214449e2737eSCaroline Tice BroadcastPromptChange (new_name, m_prompt.c_str()); 214549e2737eSCaroline Tice } 21460603aa9dSGreg Clayton m_frame_format = new_debugger_settings->m_frame_format; 21470603aa9dSGreg Clayton m_thread_format = new_debugger_settings->m_thread_format; 2148daccaa9eSCaroline Tice m_term_width = new_debugger_settings->m_term_width; 21493df9a8dfSCaroline Tice m_script_lang = new_debugger_settings->m_script_lang; 2150daccaa9eSCaroline Tice m_use_external_editor = new_debugger_settings->m_use_external_editor; 21513bcdb29cSJim Ingham m_auto_confirm_on = new_debugger_settings->m_auto_confirm_on; 21523df9a8dfSCaroline Tice } 21533df9a8dfSCaroline Tice 21543df9a8dfSCaroline Tice 21553df9a8dfSCaroline Tice bool 21563df9a8dfSCaroline Tice DebuggerInstanceSettings::BroadcastPromptChange (const ConstString &instance_name, const char *new_prompt) 21573df9a8dfSCaroline Tice { 21583df9a8dfSCaroline Tice std::string tmp_prompt; 21593df9a8dfSCaroline Tice 21603df9a8dfSCaroline Tice if (new_prompt != NULL) 21613df9a8dfSCaroline Tice { 21623df9a8dfSCaroline Tice tmp_prompt = new_prompt ; 21633df9a8dfSCaroline Tice int len = tmp_prompt.size(); 21643df9a8dfSCaroline Tice if (len > 1 21653df9a8dfSCaroline Tice && (tmp_prompt[0] == '\'' || tmp_prompt[0] == '"') 21663df9a8dfSCaroline Tice && (tmp_prompt[len-1] == tmp_prompt[0])) 21673df9a8dfSCaroline Tice { 21683df9a8dfSCaroline Tice tmp_prompt = tmp_prompt.substr(1,len-2); 21693df9a8dfSCaroline Tice } 21703df9a8dfSCaroline Tice len = tmp_prompt.size(); 21713df9a8dfSCaroline Tice if (tmp_prompt[len-1] != ' ') 21723df9a8dfSCaroline Tice tmp_prompt.append(" "); 21733df9a8dfSCaroline Tice } 21743df9a8dfSCaroline Tice EventSP new_event_sp; 21753df9a8dfSCaroline Tice new_event_sp.reset (new Event(CommandInterpreter::eBroadcastBitResetPrompt, 21763df9a8dfSCaroline Tice new EventDataBytes (tmp_prompt.c_str()))); 21773df9a8dfSCaroline Tice 21783df9a8dfSCaroline Tice if (instance_name.GetLength() != 0) 21793df9a8dfSCaroline Tice { 21803df9a8dfSCaroline Tice // Set prompt for a particular instance. 21813df9a8dfSCaroline Tice Debugger *dbg = Debugger::FindDebuggerWithInstanceName (instance_name).get(); 21823df9a8dfSCaroline Tice if (dbg != NULL) 21833df9a8dfSCaroline Tice { 21843df9a8dfSCaroline Tice dbg->GetCommandInterpreter().BroadcastEvent (new_event_sp); 21853df9a8dfSCaroline Tice } 21863df9a8dfSCaroline Tice } 21873df9a8dfSCaroline Tice 21883df9a8dfSCaroline Tice return true; 21893df9a8dfSCaroline Tice } 21903df9a8dfSCaroline Tice 21913df9a8dfSCaroline Tice const ConstString 21923df9a8dfSCaroline Tice DebuggerInstanceSettings::CreateInstanceName () 21933df9a8dfSCaroline Tice { 21943df9a8dfSCaroline Tice static int instance_count = 1; 21953df9a8dfSCaroline Tice StreamString sstr; 21963df9a8dfSCaroline Tice 21973df9a8dfSCaroline Tice sstr.Printf ("debugger_%d", instance_count); 21983df9a8dfSCaroline Tice ++instance_count; 21993df9a8dfSCaroline Tice 22003df9a8dfSCaroline Tice const ConstString ret_val (sstr.GetData()); 22013df9a8dfSCaroline Tice 22023df9a8dfSCaroline Tice return ret_val; 22033df9a8dfSCaroline Tice } 22043df9a8dfSCaroline Tice 22053df9a8dfSCaroline Tice const ConstString & 22063df9a8dfSCaroline Tice DebuggerInstanceSettings::PromptVarName () 22073df9a8dfSCaroline Tice { 22083df9a8dfSCaroline Tice static ConstString prompt_var_name ("prompt"); 22093df9a8dfSCaroline Tice 22103df9a8dfSCaroline Tice return prompt_var_name; 22113df9a8dfSCaroline Tice } 22123df9a8dfSCaroline Tice 22133df9a8dfSCaroline Tice const ConstString & 22140603aa9dSGreg Clayton DebuggerInstanceSettings::GetFrameFormatName () 22150603aa9dSGreg Clayton { 22160603aa9dSGreg Clayton static ConstString prompt_var_name ("frame-format"); 22170603aa9dSGreg Clayton 22180603aa9dSGreg Clayton return prompt_var_name; 22190603aa9dSGreg Clayton } 22200603aa9dSGreg Clayton 22210603aa9dSGreg Clayton const ConstString & 22220603aa9dSGreg Clayton DebuggerInstanceSettings::GetThreadFormatName () 22230603aa9dSGreg Clayton { 22240603aa9dSGreg Clayton static ConstString prompt_var_name ("thread-format"); 22250603aa9dSGreg Clayton 22260603aa9dSGreg Clayton return prompt_var_name; 22270603aa9dSGreg Clayton } 22280603aa9dSGreg Clayton 22290603aa9dSGreg Clayton const ConstString & 22303df9a8dfSCaroline Tice DebuggerInstanceSettings::ScriptLangVarName () 22313df9a8dfSCaroline Tice { 22323df9a8dfSCaroline Tice static ConstString script_lang_var_name ("script-lang"); 22333df9a8dfSCaroline Tice 22343df9a8dfSCaroline Tice return script_lang_var_name; 22353df9a8dfSCaroline Tice } 22363df9a8dfSCaroline Tice 2237101c7c20SCaroline Tice const ConstString & 2238101c7c20SCaroline Tice DebuggerInstanceSettings::TermWidthVarName () 2239101c7c20SCaroline Tice { 2240101c7c20SCaroline Tice static ConstString term_width_var_name ("term-width"); 2241101c7c20SCaroline Tice 2242101c7c20SCaroline Tice return term_width_var_name; 2243101c7c20SCaroline Tice } 2244101c7c20SCaroline Tice 2245daccaa9eSCaroline Tice const ConstString & 2246daccaa9eSCaroline Tice DebuggerInstanceSettings::UseExternalEditorVarName () 2247daccaa9eSCaroline Tice { 2248daccaa9eSCaroline Tice static ConstString use_external_editor_var_name ("use-external-editor"); 2249daccaa9eSCaroline Tice 2250daccaa9eSCaroline Tice return use_external_editor_var_name; 2251daccaa9eSCaroline Tice } 2252daccaa9eSCaroline Tice 22533bcdb29cSJim Ingham const ConstString & 22543bcdb29cSJim Ingham DebuggerInstanceSettings::AutoConfirmName () 22553bcdb29cSJim Ingham { 22563bcdb29cSJim Ingham static ConstString use_external_editor_var_name ("auto-confirm"); 22573bcdb29cSJim Ingham 22583bcdb29cSJim Ingham return use_external_editor_var_name; 22593bcdb29cSJim Ingham } 22603bcdb29cSJim Ingham 22613df9a8dfSCaroline Tice //-------------------------------------------------- 22621b654882SGreg Clayton // SettingsController Variable Tables 22633df9a8dfSCaroline Tice //-------------------------------------------------- 22643df9a8dfSCaroline Tice 22653df9a8dfSCaroline Tice 22663df9a8dfSCaroline Tice SettingEntry 22671b654882SGreg Clayton Debugger::SettingsController::global_settings_table[] = 22683df9a8dfSCaroline Tice { 22693df9a8dfSCaroline Tice //{ "var-name", var-type, "default", enum-table, init'd, hidden, "help-text"}, 2270101c7c20SCaroline Tice // The Debugger level global table should always be empty; all Debugger settable variables should be instance 2271101c7c20SCaroline Tice // variables. 22723df9a8dfSCaroline Tice { NULL, eSetVarTypeNone, NULL, NULL, 0, 0, NULL } 22733df9a8dfSCaroline Tice }; 22743df9a8dfSCaroline Tice 2275bb562b13SGreg Clayton #define MODULE_WITH_FUNC "{ ${module.file.basename}{`${function.name}${function.pc-offset}}}" 22760603aa9dSGreg Clayton #define FILE_AND_LINE "{ at ${line.file.basename}:${line.number}}" 22773df9a8dfSCaroline Tice 22780603aa9dSGreg Clayton #define DEFAULT_THREAD_FORMAT "thread #${thread.index}: tid = ${thread.id}"\ 22790603aa9dSGreg Clayton "{, ${frame.pc}}"\ 22800603aa9dSGreg Clayton MODULE_WITH_FUNC\ 2281cf4b9078SGreg Clayton FILE_AND_LINE\ 22820603aa9dSGreg Clayton "{, stop reason = ${thread.stop-reason}}"\ 22830603aa9dSGreg Clayton "\\n" 22840603aa9dSGreg Clayton 2285315d2cabSGreg Clayton //#define DEFAULT_THREAD_FORMAT "thread #${thread.index}: tid = ${thread.id}"\ 2286315d2cabSGreg Clayton // "{, ${frame.pc}}"\ 2287315d2cabSGreg Clayton // MODULE_WITH_FUNC\ 2288315d2cabSGreg Clayton // FILE_AND_LINE\ 2289315d2cabSGreg Clayton // "{, stop reason = ${thread.stop-reason}}"\ 2290315d2cabSGreg Clayton // "{, name = ${thread.name}}"\ 2291315d2cabSGreg Clayton // "{, queue = ${thread.queue}}"\ 2292315d2cabSGreg Clayton // "\\n" 2293315d2cabSGreg Clayton 22940603aa9dSGreg Clayton #define DEFAULT_FRAME_FORMAT "frame #${frame.index}: ${frame.pc}"\ 22950603aa9dSGreg Clayton MODULE_WITH_FUNC\ 22960603aa9dSGreg Clayton FILE_AND_LINE\ 22970603aa9dSGreg Clayton "\\n" 22983df9a8dfSCaroline Tice 22993df9a8dfSCaroline Tice SettingEntry 23001b654882SGreg Clayton Debugger::SettingsController::instance_settings_table[] = 23013df9a8dfSCaroline Tice { 23020603aa9dSGreg Clayton // NAME Setting variable type Default Enum Init'd Hidden Help 23030603aa9dSGreg Clayton // ======================= ======================= ====================== ==== ====== ====== ====================== 23040603aa9dSGreg Clayton { "frame-format", eSetVarTypeString, DEFAULT_FRAME_FORMAT, NULL, false, false, "The default frame format string to use when displaying thread information." }, 23053bcdb29cSJim Ingham { "prompt", eSetVarTypeString, "(lldb) ", NULL, false, false, "The debugger command line prompt displayed for the user." }, 23063bcdb29cSJim Ingham { "script-lang", eSetVarTypeString, "python", NULL, false, false, "The script language to be used for evaluating user-written scripts." }, 23073bcdb29cSJim Ingham { "term-width", eSetVarTypeInt, "80" , NULL, false, false, "The maximum number of columns to use for displaying text." }, 23080603aa9dSGreg Clayton { "thread-format", eSetVarTypeString, DEFAULT_THREAD_FORMAT, NULL, false, false, "The default thread format string to use when displaying thread information." }, 230906e827ccSJim Ingham { "use-external-editor", eSetVarTypeBoolean, "false", NULL, false, false, "Whether to use an external editor or not." }, 231006e827ccSJim Ingham { "auto-confirm", eSetVarTypeBoolean, "false", NULL, false, false, "If true all confirmation prompts will receive their default reply." }, 23110603aa9dSGreg Clayton { NULL, eSetVarTypeNone, NULL, NULL, false, false, NULL } 23123df9a8dfSCaroline Tice }; 2313