| //===-- ObjCLanguageRuntime.cpp ---------------------------------*- C++ -*-===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| #include "clang/AST/Type.h" |
| |
| #include "lldb/Core/Log.h" |
| #include "lldb/Core/Module.h" |
| #include "lldb/Core/PluginManager.h" |
| #include "lldb/Core/ValueObject.h" |
| #include "lldb/Symbol/ClangASTContext.h" |
| #include "lldb/Symbol/Type.h" |
| #include "lldb/Symbol/TypeList.h" |
| #include "lldb/Target/ObjCLanguageRuntime.h" |
| #include "lldb/Target/Target.h" |
| |
| using namespace lldb; |
| using namespace lldb_private; |
| |
| //---------------------------------------------------------------------- |
| // Destructor |
| //---------------------------------------------------------------------- |
| ObjCLanguageRuntime::~ObjCLanguageRuntime() |
| { |
| } |
| |
| ObjCLanguageRuntime::ObjCLanguageRuntime (Process *process) : |
| LanguageRuntime (process), |
| m_has_new_literals_and_indexing (eLazyBoolCalculate), |
| m_isa_to_descriptor_cache(), |
| m_isa_to_descriptor_cache_stop_id (UINT32_MAX) |
| { |
| |
| } |
| |
| void |
| ObjCLanguageRuntime::AddToMethodCache (lldb::addr_t class_addr, lldb::addr_t selector, lldb::addr_t impl_addr) |
| { |
| LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP)); |
| if (log) |
| { |
| log->Printf ("Caching: class 0x%llx selector 0x%llx implementation 0x%llx.", class_addr, selector, impl_addr); |
| } |
| m_impl_cache.insert (std::pair<ClassAndSel,lldb::addr_t> (ClassAndSel(class_addr, selector), impl_addr)); |
| } |
| |
| lldb::addr_t |
| ObjCLanguageRuntime::LookupInMethodCache (lldb::addr_t class_addr, lldb::addr_t selector) |
| { |
| MsgImplMap::iterator pos, end = m_impl_cache.end(); |
| pos = m_impl_cache.find (ClassAndSel(class_addr, selector)); |
| if (pos != end) |
| return (*pos).second; |
| return LLDB_INVALID_ADDRESS; |
| } |
| |
| |
| lldb::TypeSP |
| ObjCLanguageRuntime::LookupInCompleteClassCache (ConstString &name) |
| { |
| CompleteClassMap::iterator complete_class_iter = m_complete_class_cache.find(name); |
| |
| if (complete_class_iter != m_complete_class_cache.end()) |
| { |
| // Check the weak pointer to make sure the type hasn't been unloaded |
| TypeSP complete_type_sp (complete_class_iter->second.lock()); |
| |
| if (complete_type_sp) |
| return complete_type_sp; |
| else |
| m_complete_class_cache.erase(name); |
| } |
| |
| const ModuleList &modules = m_process->GetTarget().GetImages(); |
| |
| SymbolContextList sc_list; |
| const size_t matching_symbols = modules.FindSymbolsWithNameAndType (name, |
| eSymbolTypeObjCClass, |
| sc_list); |
| |
| if (matching_symbols) |
| { |
| SymbolContext sc; |
| |
| sc_list.GetContextAtIndex(0, sc); |
| |
| ModuleSP module_sp(sc.module_sp); |
| |
| if (!module_sp) |
| return TypeSP(); |
| |
| const SymbolContext null_sc; |
| const bool exact_match = true; |
| const uint32_t max_matches = UINT32_MAX; |
| TypeList types; |
| |
| const uint32_t num_types = module_sp->FindTypes (null_sc, |
| name, |
| exact_match, |
| max_matches, |
| types); |
| |
| if (num_types) |
| { |
| TypeSP incomplete_type_sp; |
| |
| uint32_t i; |
| for (i = 0; i < num_types; ++i) |
| { |
| TypeSP type_sp (types.GetTypeAtIndex(i)); |
| |
| if (ClangASTContext::IsObjCClassType(type_sp->GetClangForwardType())) |
| { |
| if (type_sp->IsCompleteObjCClass()) |
| { |
| m_complete_class_cache[name] = type_sp; |
| return type_sp; |
| } |
| else if (!incomplete_type_sp) |
| incomplete_type_sp = type_sp; |
| } |
| } |
| } |
| } |
| return TypeSP(); |
| } |
| |
| size_t |
| ObjCLanguageRuntime::GetByteOffsetForIvar (ClangASTType &parent_qual_type, const char *ivar_name) |
| { |
| return LLDB_INVALID_IVAR_OFFSET; |
| } |
| |
| |
| uint32_t |
| ObjCLanguageRuntime::ParseMethodName (const char *name, |
| ConstString *class_name, // Class name (with category if any) |
| ConstString *selector_name, // selector on its own |
| ConstString *name_sans_category, // Full function prototype with no category |
| ConstString *class_name_sans_category)// Class name with no category (or empty if no category as answer will be in "class_name" |
| { |
| if (class_name) |
| class_name->Clear(); |
| if (selector_name) |
| selector_name->Clear(); |
| if (name_sans_category) |
| name_sans_category->Clear(); |
| if (class_name_sans_category) |
| class_name_sans_category->Clear(); |
| |
| uint32_t result = 0; |
| |
| if (IsPossibleObjCMethodName (name)) |
| { |
| int name_len = strlen (name); |
| // Objective C methods must have at least: |
| // "-[" or "+[" prefix |
| // One character for a class name |
| // One character for the space between the class name |
| // One character for the method name |
| // "]" suffix |
| if (name_len >= 6 && name[name_len - 1] == ']') |
| { |
| const char *selector_name_ptr = strchr (name, ' '); |
| if (selector_name_ptr) |
| { |
| if (class_name) |
| { |
| class_name->SetCStringWithLength (name + 2, selector_name_ptr - name - 2); |
| ++result; |
| } |
| |
| // Skip the space |
| ++selector_name_ptr; |
| // Extract the objective C basename and add it to the |
| // accelerator tables |
| size_t selector_name_len = name_len - (selector_name_ptr - name) - 1; |
| if (selector_name) |
| { |
| selector_name->SetCStringWithLength (selector_name_ptr, selector_name_len); |
| ++result; |
| } |
| |
| // Also see if this is a "category" on our class. If so strip off the category name, |
| // and add the class name without it to the basename table. |
| |
| if (name_sans_category || class_name_sans_category) |
| { |
| const char *open_paren = strchr (name, '('); |
| if (open_paren) |
| { |
| if (class_name_sans_category) |
| { |
| class_name_sans_category->SetCStringWithLength (name + 2, open_paren - name - 2); |
| ++result; |
| } |
| |
| if (name_sans_category) |
| { |
| const char *close_paren = strchr (open_paren, ')'); |
| if (open_paren < close_paren) |
| { |
| std::string buffer (name, open_paren - name); |
| buffer.append (close_paren + 1); |
| name_sans_category->SetCString (buffer.c_str()); |
| ++result; |
| } |
| } |
| } |
| } |
| } |
| } |
| } |
| return result; |
| } |
| |
| bool |
| ObjCLanguageRuntime::ClassDescriptor::IsPointerValid (lldb::addr_t value, |
| uint32_t ptr_size, |
| bool allow_NULLs, |
| bool allow_tagged, |
| bool check_version_specific) const |
| { |
| if (!value) |
| return allow_NULLs; |
| if ( (value % 2) == 1 && allow_tagged) |
| return true; |
| if ((value % ptr_size) == 0) |
| return (check_version_specific ? CheckPointer(value,ptr_size) : true); |
| else |
| return false; |
| } |
| |
| ObjCLanguageRuntime::ObjCISA |
| ObjCLanguageRuntime::GetISA(const ConstString &name) |
| { |
| UpdateISAToDescriptorMap(); |
| for (const ISAToDescriptorMap::value_type &val : m_isa_to_descriptor_cache) |
| if (val.second && val.second->GetClassName() == name) |
| return val.first; |
| return 0; |
| } |
| |
| ObjCLanguageRuntime::ObjCISA |
| ObjCLanguageRuntime::GetParentClass(ObjCLanguageRuntime::ObjCISA isa) |
| { |
| ClassDescriptorSP objc_class_sp (GetClassDescriptor(isa)); |
| if (objc_class_sp) |
| { |
| ClassDescriptorSP objc_super_class_sp (objc_class_sp->GetSuperclass()); |
| if (objc_super_class_sp) |
| return objc_super_class_sp->GetISA(); |
| } |
| return 0; |
| } |
| |
| ConstString |
| ObjCLanguageRuntime::GetActualTypeName(ObjCLanguageRuntime::ObjCISA isa) |
| { |
| ClassDescriptorSP objc_class_sp (GetNonKVOClassDescriptor(isa)); |
| if (objc_class_sp) |
| return objc_class_sp->GetClassName(); |
| return ConstString(); |
| } |
| |
| ObjCLanguageRuntime::ClassDescriptorSP |
| ObjCLanguageRuntime::GetClassDescriptor (const ConstString &class_name) |
| { |
| UpdateISAToDescriptorMap(); |
| for (const ISAToDescriptorMap::value_type &val : m_isa_to_descriptor_cache) |
| if (val.second && val.second->GetClassName() == class_name) |
| return val.second; |
| return ClassDescriptorSP(); |
| |
| } |
| |
| ObjCLanguageRuntime::ClassDescriptorSP |
| ObjCLanguageRuntime::GetClassDescriptor (ValueObject& valobj) |
| { |
| ClassDescriptorSP objc_class_sp; |
| // if we get an invalid VO (which might still happen when playing around |
| // with pointers returned by the expression parser, don't consider this |
| // a valid ObjC object) |
| if (valobj.GetValue().GetContextType() != Value::eContextTypeInvalid) |
| { |
| addr_t isa_pointer = valobj.GetPointerValue(); |
| if (isa_pointer != LLDB_INVALID_ADDRESS) |
| { |
| ExecutionContext exe_ctx (valobj.GetExecutionContextRef()); |
| |
| Process *process = exe_ctx.GetProcessPtr(); |
| if (process) |
| { |
| Error error; |
| ObjCISA isa = process->ReadPointerFromMemory(isa_pointer, error); |
| if (isa != LLDB_INVALID_ADDRESS) |
| objc_class_sp = GetClassDescriptor (isa); |
| } |
| } |
| } |
| return objc_class_sp; |
| } |
| |
| ObjCLanguageRuntime::ClassDescriptorSP |
| ObjCLanguageRuntime::GetNonKVOClassDescriptor (ValueObject& valobj) |
| { |
| ObjCLanguageRuntime::ClassDescriptorSP objc_class_sp (GetClassDescriptor (valobj)); |
| if (objc_class_sp) |
| { |
| if (!objc_class_sp->IsKVO()) |
| return objc_class_sp; |
| |
| ClassDescriptorSP non_kvo_objc_class_sp(objc_class_sp->GetSuperclass()); |
| if (non_kvo_objc_class_sp && non_kvo_objc_class_sp->IsValid()) |
| return non_kvo_objc_class_sp; |
| } |
| return ClassDescriptorSP(); |
| } |
| |
| |
| ObjCLanguageRuntime::ClassDescriptorSP |
| ObjCLanguageRuntime::GetClassDescriptor (ObjCISA isa) |
| { |
| if (isa) |
| { |
| UpdateISAToDescriptorMap(); |
| ObjCLanguageRuntime::ISAToDescriptorIterator pos = m_isa_to_descriptor_cache.find(isa); |
| if (pos != m_isa_to_descriptor_cache.end()) |
| return pos->second; |
| } |
| return ClassDescriptorSP(); |
| } |
| |
| ObjCLanguageRuntime::ClassDescriptorSP |
| ObjCLanguageRuntime::GetNonKVOClassDescriptor (ObjCISA isa) |
| { |
| if (isa) |
| { |
| ClassDescriptorSP objc_class_sp = GetClassDescriptor (isa); |
| if (objc_class_sp && objc_class_sp->IsValid()) |
| { |
| if (!objc_class_sp->IsKVO()) |
| return objc_class_sp; |
| |
| ClassDescriptorSP non_kvo_objc_class_sp(objc_class_sp->GetSuperclass()); |
| if (non_kvo_objc_class_sp && non_kvo_objc_class_sp->IsValid()) |
| return non_kvo_objc_class_sp; |
| } |
| } |
| return ClassDescriptorSP(); |
| } |
| |
| |
| |