forked from OSchip/llvm-project
Added functionality to call Objective-C class methods
correctly, and added a testcase to check that it works. The main problem here is that Objective-C class method selectors are external references stored in a special data structure in the LLVM IR module for an expression. I just had to extract them and ensure that the real class object locations were properly resolved. llvm-svn: 143520
This commit is contained in:
parent
05e485879c
commit
fc89c142d3
|
@ -488,6 +488,19 @@ private:
|
||||||
//------------------------------------------------------------------
|
//------------------------------------------------------------------
|
||||||
bool
|
bool
|
||||||
HandleSymbol (llvm::Value *symbol);
|
HandleSymbol (llvm::Value *symbol);
|
||||||
|
|
||||||
|
//------------------------------------------------------------------
|
||||||
|
/// Handle a single externally-defined Objective-C class
|
||||||
|
///
|
||||||
|
/// @param[in] classlist_reference
|
||||||
|
/// The reference, usually "01L_OBJC_CLASSLIST_REFERENCES_$_n"
|
||||||
|
/// where n (if present) is an index.
|
||||||
|
///
|
||||||
|
/// @return
|
||||||
|
/// True on success; false otherwise
|
||||||
|
//------------------------------------------------------------------
|
||||||
|
bool
|
||||||
|
HandleObjCClass(llvm::Value *classlist_reference);
|
||||||
|
|
||||||
//------------------------------------------------------------------
|
//------------------------------------------------------------------
|
||||||
/// Handle all the arguments to a function call
|
/// Handle all the arguments to a function call
|
||||||
|
|
|
@ -1703,6 +1703,63 @@ IRForTarget::MaybeHandleCallArguments (CallInst *Old)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
IRForTarget::HandleObjCClass(Value *classlist_reference)
|
||||||
|
{
|
||||||
|
lldb::LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EXPRESSIONS));
|
||||||
|
|
||||||
|
GlobalVariable *global_variable = dyn_cast<GlobalVariable>(classlist_reference);
|
||||||
|
|
||||||
|
if (!global_variable)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
Constant *initializer = global_variable->getInitializer();
|
||||||
|
|
||||||
|
if (!initializer)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (!initializer->hasName())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
StringRef name(initializer->getName());
|
||||||
|
lldb_private::ConstString name_cstr(name.str().c_str());
|
||||||
|
lldb::addr_t class_ptr = m_decl_map->GetSymbolAddress(name_cstr);
|
||||||
|
|
||||||
|
if (log)
|
||||||
|
log->Printf("Found reference to Objective-C class %s (0x%llx)", name_cstr.AsCString(), (unsigned long long)class_ptr);
|
||||||
|
|
||||||
|
if (class_ptr == LLDB_INVALID_ADDRESS)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (global_variable->use_begin() == global_variable->use_end())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
LoadInst *load_instruction = NULL;
|
||||||
|
|
||||||
|
for (Value::use_iterator i = global_variable->use_begin(), e = global_variable->use_end();
|
||||||
|
i != e;
|
||||||
|
++i)
|
||||||
|
{
|
||||||
|
if ((load_instruction = dyn_cast<LoadInst>(*i)))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!load_instruction)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
IntegerType *intptr_ty = Type::getIntNTy(m_module->getContext(),
|
||||||
|
(m_module->getPointerSize() == Module::Pointer64) ? 64 : 32);
|
||||||
|
|
||||||
|
Constant *class_addr = ConstantInt::get(intptr_ty, (uint64_t)class_ptr);
|
||||||
|
Constant *class_bitcast = ConstantExpr::getIntToPtr(class_addr, load_instruction->getType());
|
||||||
|
|
||||||
|
load_instruction->replaceAllUsesWith(class_bitcast);
|
||||||
|
|
||||||
|
load_instruction->eraseFromParent();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
IRForTarget::ResolveCalls(BasicBlock &basic_block)
|
IRForTarget::ResolveCalls(BasicBlock &basic_block)
|
||||||
{
|
{
|
||||||
|
@ -1742,7 +1799,9 @@ IRForTarget::ResolveExternals (Function &llvm_function)
|
||||||
(*global).getName().str().c_str(),
|
(*global).getName().str().c_str(),
|
||||||
DeclForGlobal(global));
|
DeclForGlobal(global));
|
||||||
|
|
||||||
if ((*global).getName().str().find("OBJC_IVAR") == 0)
|
std::string global_name = (*global).getName().str();
|
||||||
|
|
||||||
|
if (global_name.find("OBJC_IVAR") == 0)
|
||||||
{
|
{
|
||||||
if (!HandleSymbol(global))
|
if (!HandleSymbol(global))
|
||||||
{
|
{
|
||||||
|
@ -1752,6 +1811,16 @@ IRForTarget::ResolveExternals (Function &llvm_function)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else if (global_name.find("OBJC_CLASSLIST_REFERENCES_$") != global_name.npos)
|
||||||
|
{
|
||||||
|
if (!HandleObjCClass(global))
|
||||||
|
{
|
||||||
|
if (m_error_stream)
|
||||||
|
m_error_stream->Printf("Error [IRForTarget]: Couldn't resolve the class for an Objective-C static method call\n");
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
else if (DeclForGlobal(global))
|
else if (DeclForGlobal(global))
|
||||||
{
|
{
|
||||||
if (!MaybeHandleVariable (global))
|
if (!MaybeHandleVariable (global))
|
||||||
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
LEVEL = ../../../make
|
||||||
|
|
||||||
|
OBJC_SOURCES := class.m
|
||||||
|
LDFLAGS = $(CFLAGS) -lobjc -framework Foundation
|
||||||
|
|
||||||
|
include $(LEVEL)/Makefile.rules
|
|
@ -0,0 +1,69 @@
|
||||||
|
"""Test calling functions in class methods."""
|
||||||
|
|
||||||
|
import os, time
|
||||||
|
import unittest2
|
||||||
|
import lldb
|
||||||
|
import lldbutil
|
||||||
|
from lldbtest import *
|
||||||
|
|
||||||
|
class TestObjCStaticMethod(TestBase):
|
||||||
|
|
||||||
|
mydir = os.path.join("lang", "objc", "objc-class-method")
|
||||||
|
|
||||||
|
@unittest2.skipUnless(sys.platform.startswith("darwin"), "requires Darwin")
|
||||||
|
@python_api_test
|
||||||
|
|
||||||
|
def test_with_dsym_and_python_api(self):
|
||||||
|
"""Test calling functions in class methods."""
|
||||||
|
self.buildDsym()
|
||||||
|
self.objc_class_method()
|
||||||
|
|
||||||
|
@python_api_test
|
||||||
|
def test_with_dwarf_and_python_api(self):
|
||||||
|
"""Test calling functions in class methods."""
|
||||||
|
self.buildDwarf()
|
||||||
|
self.objc_class_method()
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
# Call super's setUp().
|
||||||
|
TestBase.setUp(self)
|
||||||
|
# Find the line numbers to break inside main().
|
||||||
|
self.main_source = "class.m"
|
||||||
|
self.break_line = line_number(self.main_source, '// Set breakpoint here.')
|
||||||
|
|
||||||
|
#rdar://problem/9745789 "expression" can't call functions in class methods
|
||||||
|
def objc_class_method(self):
|
||||||
|
"""Test calling class methods."""
|
||||||
|
exe = os.path.join(os.getcwd(), "a.out")
|
||||||
|
|
||||||
|
target = self.dbg.CreateTarget(exe)
|
||||||
|
self.assertTrue(target, VALID_TARGET)
|
||||||
|
|
||||||
|
bpt = target.BreakpointCreateByLocation(self.main_source, self.break_line)
|
||||||
|
self.assertTrue(bpt, VALID_BREAKPOINT)
|
||||||
|
|
||||||
|
# Now launch the process, and do not stop at entry point.
|
||||||
|
process = target.LaunchSimple (None, None, os.getcwd())
|
||||||
|
|
||||||
|
self.assertTrue(process, PROCESS_IS_VALID)
|
||||||
|
|
||||||
|
# The stop reason of the thread should be breakpoint.
|
||||||
|
thread_list = lldbutil.get_threads_stopped_at_breakpoint (process, bpt)
|
||||||
|
|
||||||
|
# Make sure we stopped at the first breakpoint.
|
||||||
|
self.assertTrue (len(thread_list) != 0, "No thread stopped at our breakpoint.")
|
||||||
|
self.assertTrue (len(thread_list) == 1, "More than one thread stopped at our breakpoint.")
|
||||||
|
|
||||||
|
# Now make sure we can call a function in the class method we've stopped in.
|
||||||
|
frame = thread_list[0].GetFrameAtIndex(0)
|
||||||
|
self.assertTrue (frame, "Got a valid frame 0 frame.")
|
||||||
|
|
||||||
|
cmd_value = frame.EvaluateExpression ("(int)[Foo doSomethingWithString:@\"Hello\"]")
|
||||||
|
self.assertTrue (cmd_value.IsValid())
|
||||||
|
self.assertTrue (cmd_value.GetValueAsUnsigned() == 5)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
import atexit
|
||||||
|
lldb.SBDebugger.Initialize()
|
||||||
|
atexit.register(lambda: lldb.SBDebugger.Terminate())
|
||||||
|
unittest2.main()
|
|
@ -0,0 +1,24 @@
|
||||||
|
#import <Foundation/Foundation.h>
|
||||||
|
|
||||||
|
@interface Foo : NSObject
|
||||||
|
+(int) doSomethingWithString: (NSString *) string;
|
||||||
|
-(int) doSomethingInstance: (NSString *) string;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@implementation Foo
|
||||||
|
+(int) doSomethingWithString: (NSString *) string
|
||||||
|
{
|
||||||
|
NSLog (@"String is: %@.", string);
|
||||||
|
return [string length];
|
||||||
|
}
|
||||||
|
|
||||||
|
-(int) doSomethingInstance: (NSString *)string
|
||||||
|
{
|
||||||
|
return [Foo doSomethingWithString:string];
|
||||||
|
}
|
||||||
|
@end
|
||||||
|
|
||||||
|
int main()
|
||||||
|
{
|
||||||
|
return 0; // Set breakpoint here.
|
||||||
|
}
|
Loading…
Reference in New Issue