130fdc8d8SChris Lattner //===-- SBFrame.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 104c5de699SEli Friedman #include "lldb/API/SBFrame.h" 1130fdc8d8SChris Lattner 1230fdc8d8SChris Lattner #include <string> 1330fdc8d8SChris Lattner #include <algorithm> 1430fdc8d8SChris Lattner 1530fdc8d8SChris Lattner #include "lldb/lldb-types.h" 1630fdc8d8SChris Lattner 1730fdc8d8SChris Lattner #include "lldb/Core/Address.h" 1830fdc8d8SChris Lattner #include "lldb/Core/ConstString.h" 1930fdc8d8SChris Lattner #include "lldb/Core/Stream.h" 2030fdc8d8SChris Lattner #include "lldb/Core/StreamFile.h" 2130fdc8d8SChris Lattner #include "lldb/Core/ValueObjectRegister.h" 2230fdc8d8SChris Lattner #include "lldb/Core/ValueObjectVariable.h" 2330fdc8d8SChris Lattner #include "lldb/Symbol/Block.h" 2430fdc8d8SChris Lattner #include "lldb/Symbol/SymbolContext.h" 2530fdc8d8SChris Lattner #include "lldb/Symbol/VariableList.h" 2630fdc8d8SChris Lattner #include "lldb/Symbol/Variable.h" 2730fdc8d8SChris Lattner #include "lldb/Target/ExecutionContext.h" 2830fdc8d8SChris Lattner #include "lldb/Target/Target.h" 2930fdc8d8SChris Lattner #include "lldb/Target/Process.h" 3030fdc8d8SChris Lattner #include "lldb/Target/RegisterContext.h" 3130fdc8d8SChris Lattner #include "lldb/Target/StackFrame.h" 3230fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 3330fdc8d8SChris Lattner 344c5de699SEli Friedman #include "lldb/API/SBDebugger.h" 354c5de699SEli Friedman #include "lldb/API/SBValue.h" 364c5de699SEli Friedman #include "lldb/API/SBAddress.h" 37*dde9cff3SCaroline Tice #include "lldb/API/SBStream.h" 384c5de699SEli Friedman #include "lldb/API/SBSymbolContext.h" 394c5de699SEli Friedman #include "lldb/API/SBThread.h" 4030fdc8d8SChris Lattner 4130fdc8d8SChris Lattner using namespace lldb; 4230fdc8d8SChris Lattner using namespace lldb_private; 4330fdc8d8SChris Lattner 4430fdc8d8SChris Lattner SBFrame::SBFrame () : 456611103cSGreg Clayton m_opaque_sp () 4630fdc8d8SChris Lattner { 4730fdc8d8SChris Lattner } 4830fdc8d8SChris Lattner 4930fdc8d8SChris Lattner SBFrame::SBFrame (const lldb::StackFrameSP &lldb_object_sp) : 506611103cSGreg Clayton m_opaque_sp (lldb_object_sp) 5130fdc8d8SChris Lattner { 5230fdc8d8SChris Lattner } 5330fdc8d8SChris Lattner 5430fdc8d8SChris Lattner SBFrame::~SBFrame() 5530fdc8d8SChris Lattner { 5630fdc8d8SChris Lattner } 5730fdc8d8SChris Lattner 5830fdc8d8SChris Lattner 5930fdc8d8SChris Lattner void 6030fdc8d8SChris Lattner SBFrame::SetFrame (const lldb::StackFrameSP &lldb_object_sp) 6130fdc8d8SChris Lattner { 626611103cSGreg Clayton m_opaque_sp = lldb_object_sp; 6330fdc8d8SChris Lattner } 6430fdc8d8SChris Lattner 6530fdc8d8SChris Lattner 6630fdc8d8SChris Lattner bool 6730fdc8d8SChris Lattner SBFrame::IsValid() const 6830fdc8d8SChris Lattner { 696611103cSGreg Clayton return (m_opaque_sp.get() != NULL); 7030fdc8d8SChris Lattner } 7130fdc8d8SChris Lattner 7230fdc8d8SChris Lattner SBSymbolContext 7330fdc8d8SChris Lattner SBFrame::GetSymbolContext (uint32_t resolve_scope) const 7430fdc8d8SChris Lattner { 7530fdc8d8SChris Lattner SBSymbolContext sb_sym_ctx; 766611103cSGreg Clayton if (m_opaque_sp) 776611103cSGreg Clayton sb_sym_ctx.SetSymbolContext(&m_opaque_sp->GetSymbolContext (resolve_scope)); 7830fdc8d8SChris Lattner return sb_sym_ctx; 7930fdc8d8SChris Lattner } 8030fdc8d8SChris Lattner 8130fdc8d8SChris Lattner SBModule 8230fdc8d8SChris Lattner SBFrame::GetModule () const 8330fdc8d8SChris Lattner { 846611103cSGreg Clayton SBModule sb_module (m_opaque_sp->GetSymbolContext (eSymbolContextModule).module_sp); 8530fdc8d8SChris Lattner return sb_module; 8630fdc8d8SChris Lattner } 8730fdc8d8SChris Lattner 8830fdc8d8SChris Lattner SBCompileUnit 8930fdc8d8SChris Lattner SBFrame::GetCompileUnit () const 9030fdc8d8SChris Lattner { 916611103cSGreg Clayton SBCompileUnit sb_comp_unit(m_opaque_sp->GetSymbolContext (eSymbolContextCompUnit).comp_unit); 9230fdc8d8SChris Lattner return sb_comp_unit; 9330fdc8d8SChris Lattner } 9430fdc8d8SChris Lattner 9530fdc8d8SChris Lattner SBFunction 9630fdc8d8SChris Lattner SBFrame::GetFunction () const 9730fdc8d8SChris Lattner { 986611103cSGreg Clayton SBFunction sb_function(m_opaque_sp->GetSymbolContext (eSymbolContextFunction).function); 9930fdc8d8SChris Lattner return sb_function; 10030fdc8d8SChris Lattner } 10130fdc8d8SChris Lattner 10230fdc8d8SChris Lattner SBBlock 10330fdc8d8SChris Lattner SBFrame::GetBlock () const 10430fdc8d8SChris Lattner { 1056611103cSGreg Clayton SBBlock sb_block(m_opaque_sp->GetSymbolContext (eSymbolContextBlock).block); 10630fdc8d8SChris Lattner return sb_block; 10730fdc8d8SChris Lattner } 10830fdc8d8SChris Lattner 10995897c6aSGreg Clayton SBBlock 11095897c6aSGreg Clayton SBFrame::GetFrameBlock () const 11195897c6aSGreg Clayton { 11295897c6aSGreg Clayton SBBlock sb_block(m_opaque_sp->GetFrameBlock ()); 11395897c6aSGreg Clayton return sb_block; 11495897c6aSGreg Clayton } 11595897c6aSGreg Clayton 11630fdc8d8SChris Lattner SBLineEntry 11730fdc8d8SChris Lattner SBFrame::GetLineEntry () const 11830fdc8d8SChris Lattner { 1196611103cSGreg Clayton SBLineEntry sb_line_entry(&m_opaque_sp->GetSymbolContext (eSymbolContextLineEntry).line_entry); 12030fdc8d8SChris Lattner return sb_line_entry; 12130fdc8d8SChris Lattner } 12230fdc8d8SChris Lattner 12330fdc8d8SChris Lattner uint32_t 12430fdc8d8SChris Lattner SBFrame::GetFrameID () const 12530fdc8d8SChris Lattner { 1266611103cSGreg Clayton if (m_opaque_sp) 1271b72fcb7SGreg Clayton return m_opaque_sp->GetFrameIndex (); 12830fdc8d8SChris Lattner else 12930fdc8d8SChris Lattner return UINT32_MAX; 13030fdc8d8SChris Lattner } 13130fdc8d8SChris Lattner 13230fdc8d8SChris Lattner lldb::addr_t 13330fdc8d8SChris Lattner SBFrame::GetPC () const 13430fdc8d8SChris Lattner { 1356611103cSGreg Clayton if (m_opaque_sp) 136f5e56de0SGreg Clayton return m_opaque_sp->GetFrameCodeAddress().GetLoadAddress (&m_opaque_sp->GetThread().GetProcess().GetTarget()); 13730fdc8d8SChris Lattner return LLDB_INVALID_ADDRESS; 13830fdc8d8SChris Lattner } 13930fdc8d8SChris Lattner 14030fdc8d8SChris Lattner bool 14130fdc8d8SChris Lattner SBFrame::SetPC (lldb::addr_t new_pc) 14230fdc8d8SChris Lattner { 1436611103cSGreg Clayton if (m_opaque_sp) 1446611103cSGreg Clayton return m_opaque_sp->GetRegisterContext()->SetPC (new_pc); 14530fdc8d8SChris Lattner return false; 14630fdc8d8SChris Lattner } 14730fdc8d8SChris Lattner 14830fdc8d8SChris Lattner lldb::addr_t 14930fdc8d8SChris Lattner SBFrame::GetSP () const 15030fdc8d8SChris Lattner { 1516611103cSGreg Clayton if (m_opaque_sp) 1526611103cSGreg Clayton return m_opaque_sp->GetRegisterContext()->GetSP(); 15330fdc8d8SChris Lattner return LLDB_INVALID_ADDRESS; 15430fdc8d8SChris Lattner } 15530fdc8d8SChris Lattner 15630fdc8d8SChris Lattner 15730fdc8d8SChris Lattner lldb::addr_t 15830fdc8d8SChris Lattner SBFrame::GetFP () const 15930fdc8d8SChris Lattner { 1606611103cSGreg Clayton if (m_opaque_sp) 1616611103cSGreg Clayton return m_opaque_sp->GetRegisterContext()->GetFP(); 16230fdc8d8SChris Lattner return LLDB_INVALID_ADDRESS; 16330fdc8d8SChris Lattner } 16430fdc8d8SChris Lattner 16530fdc8d8SChris Lattner 16630fdc8d8SChris Lattner SBAddress 16730fdc8d8SChris Lattner SBFrame::GetPCAddress () const 16830fdc8d8SChris Lattner { 16930fdc8d8SChris Lattner SBAddress sb_addr; 1706611103cSGreg Clayton if (m_opaque_sp) 1719da7bd07SGreg Clayton sb_addr.SetAddress (&m_opaque_sp->GetFrameCodeAddress()); 17230fdc8d8SChris Lattner return sb_addr; 17330fdc8d8SChris Lattner } 17430fdc8d8SChris Lattner 17530fdc8d8SChris Lattner void 17630fdc8d8SChris Lattner SBFrame::Clear() 17730fdc8d8SChris Lattner { 1786611103cSGreg Clayton m_opaque_sp.reset(); 17930fdc8d8SChris Lattner } 18030fdc8d8SChris Lattner 18130fdc8d8SChris Lattner SBValue 18230fdc8d8SChris Lattner SBFrame::LookupVar (const char *var_name) 18330fdc8d8SChris Lattner { 18430fdc8d8SChris Lattner lldb::VariableSP var_sp; 18530fdc8d8SChris Lattner if (IsValid ()) 18630fdc8d8SChris Lattner { 18730fdc8d8SChris Lattner lldb_private::VariableList variable_list; 18830fdc8d8SChris Lattner SBSymbolContext sc = GetSymbolContext (eSymbolContextEverything); 18930fdc8d8SChris Lattner 19030fdc8d8SChris Lattner SBBlock block = sc.GetBlock(); 19130fdc8d8SChris Lattner if (block.IsValid()) 19230fdc8d8SChris Lattner block.AppendVariables (true, true, &variable_list); 19330fdc8d8SChris Lattner 19430fdc8d8SChris Lattner const uint32_t num_variables = variable_list.GetSize(); 19530fdc8d8SChris Lattner 19630fdc8d8SChris Lattner bool found = false; 197c982c768SGreg Clayton for (uint32_t i = 0; i < num_variables && !found; ++i) 19830fdc8d8SChris Lattner { 19930fdc8d8SChris Lattner var_sp = variable_list.GetVariableAtIndex(i); 20030fdc8d8SChris Lattner if (var_sp 20130fdc8d8SChris Lattner && (var_sp.get()->GetName() == lldb_private::ConstString(var_name))) 20230fdc8d8SChris Lattner found = true; 20330fdc8d8SChris Lattner } 20430fdc8d8SChris Lattner if (!found) 20530fdc8d8SChris Lattner var_sp.reset(); 20630fdc8d8SChris Lattner } 207*dde9cff3SCaroline Tice if (var_sp) 208*dde9cff3SCaroline Tice { 20930fdc8d8SChris Lattner SBValue sb_value (ValueObjectSP (new ValueObjectVariable (var_sp))); 21030fdc8d8SChris Lattner return sb_value; 21130fdc8d8SChris Lattner } 21230fdc8d8SChris Lattner 213*dde9cff3SCaroline Tice SBValue sb_value; 214*dde9cff3SCaroline Tice return sb_value; 215*dde9cff3SCaroline Tice } 216*dde9cff3SCaroline Tice 21730fdc8d8SChris Lattner SBValue 21830fdc8d8SChris Lattner SBFrame::LookupVarInScope (const char *var_name, const char *scope) 21930fdc8d8SChris Lattner { 22030fdc8d8SChris Lattner lldb::VariableSP var_sp; 22130fdc8d8SChris Lattner if (IsValid()) 22230fdc8d8SChris Lattner { 22330fdc8d8SChris Lattner std::string scope_str = scope; 22430fdc8d8SChris Lattner lldb::ValueType var_scope = eValueTypeInvalid; 22530fdc8d8SChris Lattner // Convert scope_str to be all lowercase; 22630fdc8d8SChris Lattner std::transform (scope_str.begin(), scope_str.end(), scope_str.begin(), ::tolower); 22730fdc8d8SChris Lattner 22830fdc8d8SChris Lattner if (scope_str.compare ("global") == 0) 22930fdc8d8SChris Lattner var_scope = eValueTypeVariableGlobal; 23030fdc8d8SChris Lattner else if (scope_str.compare ("local") == 0) 23130fdc8d8SChris Lattner var_scope = eValueTypeVariableLocal; 23230fdc8d8SChris Lattner else if (scope_str.compare ("parameter") == 0) 23330fdc8d8SChris Lattner var_scope = eValueTypeVariableArgument; 23430fdc8d8SChris Lattner 23530fdc8d8SChris Lattner if (var_scope != eValueTypeInvalid) 23630fdc8d8SChris Lattner { 23730fdc8d8SChris Lattner lldb_private::VariableList variable_list; 23830fdc8d8SChris Lattner SBSymbolContext sc = GetSymbolContext (eSymbolContextEverything); 23930fdc8d8SChris Lattner 24030fdc8d8SChris Lattner SBBlock block = sc.GetBlock(); 24130fdc8d8SChris Lattner if (block.IsValid()) 24230fdc8d8SChris Lattner block.AppendVariables (true, true, &variable_list); 24330fdc8d8SChris Lattner 24430fdc8d8SChris Lattner const uint32_t num_variables = variable_list.GetSize(); 24530fdc8d8SChris Lattner 24630fdc8d8SChris Lattner bool found = false; 247c982c768SGreg Clayton for (uint32_t i = 0; i < num_variables && !found; ++i) 24830fdc8d8SChris Lattner { 24930fdc8d8SChris Lattner var_sp = variable_list.GetVariableAtIndex(i); 25030fdc8d8SChris Lattner if (var_sp 25130fdc8d8SChris Lattner && (var_sp.get()->GetName() == lldb_private::ConstString(var_name)) 25230fdc8d8SChris Lattner && var_sp.get()->GetScope() == var_scope) 25330fdc8d8SChris Lattner found = true; 25430fdc8d8SChris Lattner } 25530fdc8d8SChris Lattner if (!found) 25630fdc8d8SChris Lattner var_sp.reset(); 25730fdc8d8SChris Lattner } 25830fdc8d8SChris Lattner } 259*dde9cff3SCaroline Tice 260*dde9cff3SCaroline Tice if (var_sp) 261*dde9cff3SCaroline Tice { 26230fdc8d8SChris Lattner SBValue sb_value (ValueObjectSP (new ValueObjectVariable (var_sp))); 26330fdc8d8SChris Lattner return sb_value; 26430fdc8d8SChris Lattner } 26530fdc8d8SChris Lattner 266*dde9cff3SCaroline Tice SBValue sb_value; 267*dde9cff3SCaroline Tice return sb_value; 268*dde9cff3SCaroline Tice } 269*dde9cff3SCaroline Tice 27030fdc8d8SChris Lattner bool 27130fdc8d8SChris Lattner SBFrame::operator == (const SBFrame &rhs) const 27230fdc8d8SChris Lattner { 2736611103cSGreg Clayton return m_opaque_sp.get() == rhs.m_opaque_sp.get(); 27430fdc8d8SChris Lattner } 27530fdc8d8SChris Lattner 27630fdc8d8SChris Lattner bool 27730fdc8d8SChris Lattner SBFrame::operator != (const SBFrame &rhs) const 27830fdc8d8SChris Lattner { 2796611103cSGreg Clayton return m_opaque_sp.get() != rhs.m_opaque_sp.get(); 28030fdc8d8SChris Lattner } 28130fdc8d8SChris Lattner 28230fdc8d8SChris Lattner lldb_private::StackFrame * 28330fdc8d8SChris Lattner SBFrame::operator->() const 28430fdc8d8SChris Lattner { 2856611103cSGreg Clayton return m_opaque_sp.get(); 28630fdc8d8SChris Lattner } 28730fdc8d8SChris Lattner 28830fdc8d8SChris Lattner lldb_private::StackFrame * 28930fdc8d8SChris Lattner SBFrame::get() const 29030fdc8d8SChris Lattner { 2916611103cSGreg Clayton return m_opaque_sp.get(); 29230fdc8d8SChris Lattner } 29330fdc8d8SChris Lattner 29430fdc8d8SChris Lattner 29530fdc8d8SChris Lattner SBThread 29630fdc8d8SChris Lattner SBFrame::GetThread () const 29730fdc8d8SChris Lattner { 2986611103cSGreg Clayton SBThread sb_thread (m_opaque_sp->GetThread().GetSP()); 29930fdc8d8SChris Lattner return sb_thread; 30030fdc8d8SChris Lattner } 30130fdc8d8SChris Lattner 30230fdc8d8SChris Lattner const char * 30330fdc8d8SChris Lattner SBFrame::Disassemble () const 30430fdc8d8SChris Lattner { 3056611103cSGreg Clayton if (m_opaque_sp) 3066611103cSGreg Clayton return m_opaque_sp->Disassemble(); 30730fdc8d8SChris Lattner return NULL; 30830fdc8d8SChris Lattner } 30930fdc8d8SChris Lattner 31030fdc8d8SChris Lattner 31130fdc8d8SChris Lattner 31230fdc8d8SChris Lattner lldb_private::StackFrame * 31330fdc8d8SChris Lattner SBFrame::GetLLDBObjectPtr () 31430fdc8d8SChris Lattner { 3156611103cSGreg Clayton return m_opaque_sp.get(); 31630fdc8d8SChris Lattner } 31730fdc8d8SChris Lattner 31830fdc8d8SChris Lattner SBValueList 31930fdc8d8SChris Lattner SBFrame::GetVariables (bool arguments, 32030fdc8d8SChris Lattner bool locals, 32130fdc8d8SChris Lattner bool statics, 32230fdc8d8SChris Lattner bool in_scope_only) 32330fdc8d8SChris Lattner { 32430fdc8d8SChris Lattner SBValueList value_list; 3256611103cSGreg Clayton if (m_opaque_sp) 32630fdc8d8SChris Lattner { 32730fdc8d8SChris Lattner size_t i; 328288bdf9cSGreg Clayton VariableList *variable_list = m_opaque_sp->GetVariableList(true); 32930fdc8d8SChris Lattner if (variable_list) 33030fdc8d8SChris Lattner { 33130fdc8d8SChris Lattner const size_t num_variables = variable_list->GetSize(); 33230fdc8d8SChris Lattner if (num_variables) 33330fdc8d8SChris Lattner { 33430fdc8d8SChris Lattner for (i = 0; i < num_variables; ++i) 33530fdc8d8SChris Lattner { 33630fdc8d8SChris Lattner VariableSP variable_sp (variable_list->GetVariableAtIndex(i)); 33730fdc8d8SChris Lattner if (variable_sp) 33830fdc8d8SChris Lattner { 33930fdc8d8SChris Lattner bool add_variable = false; 34030fdc8d8SChris Lattner switch (variable_sp->GetScope()) 34130fdc8d8SChris Lattner { 34230fdc8d8SChris Lattner case eValueTypeVariableGlobal: 34330fdc8d8SChris Lattner case eValueTypeVariableStatic: 34430fdc8d8SChris Lattner add_variable = statics; 34530fdc8d8SChris Lattner break; 34630fdc8d8SChris Lattner 34730fdc8d8SChris Lattner case eValueTypeVariableArgument: 34830fdc8d8SChris Lattner add_variable = arguments; 34930fdc8d8SChris Lattner break; 35030fdc8d8SChris Lattner 35130fdc8d8SChris Lattner case eValueTypeVariableLocal: 35230fdc8d8SChris Lattner add_variable = locals; 35330fdc8d8SChris Lattner break; 354c982c768SGreg Clayton 355c982c768SGreg Clayton default: 356c982c768SGreg Clayton break; 35730fdc8d8SChris Lattner } 35830fdc8d8SChris Lattner if (add_variable) 35930fdc8d8SChris Lattner { 3606611103cSGreg Clayton if (in_scope_only && !variable_sp->IsInScope(m_opaque_sp.get())) 36130fdc8d8SChris Lattner continue; 36230fdc8d8SChris Lattner 363288bdf9cSGreg Clayton value_list.Append(m_opaque_sp->GetValueObjectForFrameVariable (variable_sp)); 36430fdc8d8SChris Lattner } 36530fdc8d8SChris Lattner } 36630fdc8d8SChris Lattner } 36730fdc8d8SChris Lattner } 36830fdc8d8SChris Lattner } 36930fdc8d8SChris Lattner } 37030fdc8d8SChris Lattner return value_list; 37130fdc8d8SChris Lattner } 37230fdc8d8SChris Lattner 37330fdc8d8SChris Lattner lldb::SBValueList 37430fdc8d8SChris Lattner SBFrame::GetRegisters () 37530fdc8d8SChris Lattner { 37630fdc8d8SChris Lattner SBValueList value_list; 3776611103cSGreg Clayton if (m_opaque_sp) 37830fdc8d8SChris Lattner { 3796611103cSGreg Clayton RegisterContext *reg_ctx = m_opaque_sp->GetRegisterContext(); 38030fdc8d8SChris Lattner if (reg_ctx) 38130fdc8d8SChris Lattner { 38230fdc8d8SChris Lattner const uint32_t num_sets = reg_ctx->GetRegisterSetCount(); 38330fdc8d8SChris Lattner for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) 38430fdc8d8SChris Lattner { 38530fdc8d8SChris Lattner value_list.Append(ValueObjectSP (new ValueObjectRegisterSet (reg_ctx, set_idx))); 38630fdc8d8SChris Lattner } 38730fdc8d8SChris Lattner } 38830fdc8d8SChris Lattner } 38930fdc8d8SChris Lattner return value_list; 39030fdc8d8SChris Lattner } 39130fdc8d8SChris Lattner 392*dde9cff3SCaroline Tice bool 393*dde9cff3SCaroline Tice SBFrame::GetDescription (SBStream &description) 394*dde9cff3SCaroline Tice { 395*dde9cff3SCaroline Tice if (m_opaque_sp) 396*dde9cff3SCaroline Tice { 397*dde9cff3SCaroline Tice m_opaque_sp->Dump (description.get(), true, false); 398*dde9cff3SCaroline Tice } 399*dde9cff3SCaroline Tice else 400*dde9cff3SCaroline Tice description.Printf ("No value"); 401*dde9cff3SCaroline Tice 402*dde9cff3SCaroline Tice return true; 403*dde9cff3SCaroline Tice } 404*dde9cff3SCaroline Tice 405*dde9cff3SCaroline Tice PyObject * 406*dde9cff3SCaroline Tice SBFrame::__repr__ () 407*dde9cff3SCaroline Tice { 408*dde9cff3SCaroline Tice SBStream description; 409*dde9cff3SCaroline Tice description.ref(); 410*dde9cff3SCaroline Tice GetDescription (description); 411*dde9cff3SCaroline Tice return PyString_FromString (description.GetData()); 412*dde9cff3SCaroline Tice } 413