|  | //===-- SBFunction.cpp ----------------------------------------------------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "lldb/API/SBFunction.h" | 
|  | #include "SBReproducerPrivate.h" | 
|  | #include "lldb/API/SBProcess.h" | 
|  | #include "lldb/API/SBStream.h" | 
|  | #include "lldb/Core/Disassembler.h" | 
|  | #include "lldb/Core/Module.h" | 
|  | #include "lldb/Symbol/CompileUnit.h" | 
|  | #include "lldb/Symbol/Function.h" | 
|  | #include "lldb/Symbol/Type.h" | 
|  | #include "lldb/Symbol/VariableList.h" | 
|  | #include "lldb/Target/ExecutionContext.h" | 
|  | #include "lldb/Target/Target.h" | 
|  |  | 
|  | using namespace lldb; | 
|  | using namespace lldb_private; | 
|  |  | 
|  | SBFunction::SBFunction() : m_opaque_ptr(nullptr) { | 
|  | LLDB_RECORD_CONSTRUCTOR_NO_ARGS(SBFunction); | 
|  | } | 
|  |  | 
|  | SBFunction::SBFunction(lldb_private::Function *lldb_object_ptr) | 
|  | : m_opaque_ptr(lldb_object_ptr) {} | 
|  |  | 
|  | SBFunction::SBFunction(const lldb::SBFunction &rhs) | 
|  | : m_opaque_ptr(rhs.m_opaque_ptr) { | 
|  | LLDB_RECORD_CONSTRUCTOR(SBFunction, (const lldb::SBFunction &), rhs); | 
|  | } | 
|  |  | 
|  | const SBFunction &SBFunction::operator=(const SBFunction &rhs) { | 
|  | LLDB_RECORD_METHOD(const lldb::SBFunction &, | 
|  | SBFunction, operator=,(const lldb::SBFunction &), rhs); | 
|  |  | 
|  | m_opaque_ptr = rhs.m_opaque_ptr; | 
|  | return LLDB_RECORD_RESULT(*this); | 
|  | } | 
|  |  | 
|  | SBFunction::~SBFunction() { m_opaque_ptr = nullptr; } | 
|  |  | 
|  | bool SBFunction::IsValid() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(bool, SBFunction, IsValid); | 
|  | return this->operator bool(); | 
|  | } | 
|  | SBFunction::operator bool() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(bool, SBFunction, operator bool); | 
|  |  | 
|  | return m_opaque_ptr != nullptr; | 
|  | } | 
|  |  | 
|  | const char *SBFunction::GetName() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(const char *, SBFunction, GetName); | 
|  |  | 
|  | const char *cstr = nullptr; | 
|  | if (m_opaque_ptr) | 
|  | cstr = m_opaque_ptr->GetName().AsCString(); | 
|  |  | 
|  | return cstr; | 
|  | } | 
|  |  | 
|  | const char *SBFunction::GetDisplayName() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(const char *, SBFunction, GetDisplayName); | 
|  |  | 
|  | const char *cstr = nullptr; | 
|  | if (m_opaque_ptr) | 
|  | cstr = m_opaque_ptr->GetMangled().GetDisplayDemangledName().AsCString(); | 
|  |  | 
|  | return cstr; | 
|  | } | 
|  |  | 
|  | const char *SBFunction::GetMangledName() const { | 
|  | LLDB_RECORD_METHOD_CONST_NO_ARGS(const char *, SBFunction, GetMangledName); | 
|  |  | 
|  | const char *cstr = nullptr; | 
|  | if (m_opaque_ptr) | 
|  | cstr = m_opaque_ptr->GetMangled().GetMangledName().AsCString(); | 
|  | return cstr; | 
|  | } | 
|  |  | 
|  | bool SBFunction::operator==(const SBFunction &rhs) const { | 
|  | LLDB_RECORD_METHOD_CONST( | 
|  | bool, SBFunction, operator==,(const lldb::SBFunction &), rhs); | 
|  |  | 
|  | return m_opaque_ptr == rhs.m_opaque_ptr; | 
|  | } | 
|  |  | 
|  | bool SBFunction::operator!=(const SBFunction &rhs) const { | 
|  | LLDB_RECORD_METHOD_CONST( | 
|  | bool, SBFunction, operator!=,(const lldb::SBFunction &), rhs); | 
|  |  | 
|  | return m_opaque_ptr != rhs.m_opaque_ptr; | 
|  | } | 
|  |  | 
|  | bool SBFunction::GetDescription(SBStream &s) { | 
|  | LLDB_RECORD_METHOD(bool, SBFunction, GetDescription, (lldb::SBStream &), s); | 
|  |  | 
|  | if (m_opaque_ptr) { | 
|  | s.Printf("SBFunction: id = 0x%8.8" PRIx64 ", name = %s", | 
|  | m_opaque_ptr->GetID(), m_opaque_ptr->GetName().AsCString()); | 
|  | Type *func_type = m_opaque_ptr->GetType(); | 
|  | if (func_type) | 
|  | s.Printf(", type = %s", func_type->GetName().AsCString()); | 
|  | return true; | 
|  | } | 
|  | s.Printf("No value"); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | SBInstructionList SBFunction::GetInstructions(SBTarget target) { | 
|  | LLDB_RECORD_METHOD(lldb::SBInstructionList, SBFunction, GetInstructions, | 
|  | (lldb::SBTarget), target); | 
|  |  | 
|  | return LLDB_RECORD_RESULT(GetInstructions(target, nullptr)); | 
|  | } | 
|  |  | 
|  | SBInstructionList SBFunction::GetInstructions(SBTarget target, | 
|  | const char *flavor) { | 
|  | LLDB_RECORD_METHOD(lldb::SBInstructionList, SBFunction, GetInstructions, | 
|  | (lldb::SBTarget, const char *), target, flavor); | 
|  |  | 
|  | SBInstructionList sb_instructions; | 
|  | if (m_opaque_ptr) { | 
|  | TargetSP target_sp(target.GetSP()); | 
|  | std::unique_lock<std::recursive_mutex> lock; | 
|  | ModuleSP module_sp( | 
|  | m_opaque_ptr->GetAddressRange().GetBaseAddress().GetModule()); | 
|  | if (target_sp && module_sp) { | 
|  | lock = std::unique_lock<std::recursive_mutex>(target_sp->GetAPIMutex()); | 
|  | const bool prefer_file_cache = false; | 
|  | sb_instructions.SetDisassembler(Disassembler::DisassembleRange( | 
|  | module_sp->GetArchitecture(), nullptr, flavor, *target_sp, | 
|  | m_opaque_ptr->GetAddressRange(), prefer_file_cache)); | 
|  | } | 
|  | } | 
|  | return LLDB_RECORD_RESULT(sb_instructions); | 
|  | } | 
|  |  | 
|  | lldb_private::Function *SBFunction::get() { return m_opaque_ptr; } | 
|  |  | 
|  | void SBFunction::reset(lldb_private::Function *lldb_object_ptr) { | 
|  | m_opaque_ptr = lldb_object_ptr; | 
|  | } | 
|  |  | 
|  | SBAddress SBFunction::GetStartAddress() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBAddress, SBFunction, GetStartAddress); | 
|  |  | 
|  | SBAddress addr; | 
|  | if (m_opaque_ptr) | 
|  | addr.SetAddress(m_opaque_ptr->GetAddressRange().GetBaseAddress()); | 
|  | return LLDB_RECORD_RESULT(addr); | 
|  | } | 
|  |  | 
|  | SBAddress SBFunction::GetEndAddress() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBAddress, SBFunction, GetEndAddress); | 
|  |  | 
|  | SBAddress addr; | 
|  | if (m_opaque_ptr) { | 
|  | addr_t byte_size = m_opaque_ptr->GetAddressRange().GetByteSize(); | 
|  | if (byte_size > 0) { | 
|  | addr.SetAddress(m_opaque_ptr->GetAddressRange().GetBaseAddress()); | 
|  | addr->Slide(byte_size); | 
|  | } | 
|  | } | 
|  | return LLDB_RECORD_RESULT(addr); | 
|  | } | 
|  |  | 
|  | const char *SBFunction::GetArgumentName(uint32_t arg_idx) { | 
|  | LLDB_RECORD_METHOD(const char *, SBFunction, GetArgumentName, (uint32_t), | 
|  | arg_idx); | 
|  |  | 
|  | if (m_opaque_ptr) { | 
|  | Block &block = m_opaque_ptr->GetBlock(true); | 
|  | VariableListSP variable_list_sp = block.GetBlockVariableList(true); | 
|  | if (variable_list_sp) { | 
|  | VariableList arguments; | 
|  | variable_list_sp->AppendVariablesWithScope(eValueTypeVariableArgument, | 
|  | arguments, true); | 
|  | lldb::VariableSP variable_sp = arguments.GetVariableAtIndex(arg_idx); | 
|  | if (variable_sp) | 
|  | return variable_sp->GetName().GetCString(); | 
|  | } | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | uint32_t SBFunction::GetPrologueByteSize() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(uint32_t, SBFunction, GetPrologueByteSize); | 
|  |  | 
|  | if (m_opaque_ptr) | 
|  | return m_opaque_ptr->GetPrologueByteSize(); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | SBType SBFunction::GetType() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBType, SBFunction, GetType); | 
|  |  | 
|  | SBType sb_type; | 
|  | if (m_opaque_ptr) { | 
|  | Type *function_type = m_opaque_ptr->GetType(); | 
|  | if (function_type) | 
|  | sb_type.ref().SetType(function_type->shared_from_this()); | 
|  | } | 
|  | return LLDB_RECORD_RESULT(sb_type); | 
|  | } | 
|  |  | 
|  | SBBlock SBFunction::GetBlock() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::SBBlock, SBFunction, GetBlock); | 
|  |  | 
|  | SBBlock sb_block; | 
|  | if (m_opaque_ptr) | 
|  | sb_block.SetPtr(&m_opaque_ptr->GetBlock(true)); | 
|  | return LLDB_RECORD_RESULT(sb_block); | 
|  | } | 
|  |  | 
|  | lldb::LanguageType SBFunction::GetLanguage() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(lldb::LanguageType, SBFunction, GetLanguage); | 
|  |  | 
|  | if (m_opaque_ptr) { | 
|  | if (m_opaque_ptr->GetCompileUnit()) | 
|  | return m_opaque_ptr->GetCompileUnit()->GetLanguage(); | 
|  | } | 
|  | return lldb::eLanguageTypeUnknown; | 
|  | } | 
|  |  | 
|  | bool SBFunction::GetIsOptimized() { | 
|  | LLDB_RECORD_METHOD_NO_ARGS(bool, SBFunction, GetIsOptimized); | 
|  |  | 
|  | if (m_opaque_ptr) { | 
|  | if (m_opaque_ptr->GetCompileUnit()) | 
|  | return m_opaque_ptr->GetCompileUnit()->GetIsOptimized(); | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | namespace lldb_private { | 
|  | namespace repro { | 
|  |  | 
|  | template <> | 
|  | void RegisterMethods<SBFunction>(Registry &R) { | 
|  | LLDB_REGISTER_CONSTRUCTOR(SBFunction, ()); | 
|  | LLDB_REGISTER_CONSTRUCTOR(SBFunction, (const lldb::SBFunction &)); | 
|  | LLDB_REGISTER_METHOD(const lldb::SBFunction &, | 
|  | SBFunction, operator=,(const lldb::SBFunction &)); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, SBFunction, IsValid, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(bool, SBFunction, operator bool, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(const char *, SBFunction, GetName, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(const char *, SBFunction, GetDisplayName, ()); | 
|  | LLDB_REGISTER_METHOD_CONST(const char *, SBFunction, GetMangledName, ()); | 
|  | LLDB_REGISTER_METHOD_CONST( | 
|  | bool, SBFunction, operator==,(const lldb::SBFunction &)); | 
|  | LLDB_REGISTER_METHOD_CONST( | 
|  | bool, SBFunction, operator!=,(const lldb::SBFunction &)); | 
|  | LLDB_REGISTER_METHOD(bool, SBFunction, GetDescription, (lldb::SBStream &)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBInstructionList, SBFunction, GetInstructions, | 
|  | (lldb::SBTarget)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBInstructionList, SBFunction, GetInstructions, | 
|  | (lldb::SBTarget, const char *)); | 
|  | LLDB_REGISTER_METHOD(lldb::SBAddress, SBFunction, GetStartAddress, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::SBAddress, SBFunction, GetEndAddress, ()); | 
|  | LLDB_REGISTER_METHOD(const char *, SBFunction, GetArgumentName, (uint32_t)); | 
|  | LLDB_REGISTER_METHOD(uint32_t, SBFunction, GetPrologueByteSize, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::SBType, SBFunction, GetType, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::SBBlock, SBFunction, GetBlock, ()); | 
|  | LLDB_REGISTER_METHOD(lldb::LanguageType, SBFunction, GetLanguage, ()); | 
|  | LLDB_REGISTER_METHOD(bool, SBFunction, GetIsOptimized, ()); | 
|  | } | 
|  |  | 
|  | } | 
|  | } |