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
|
||||
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
|
||||
|
|
|
@ -1703,6 +1703,63 @@ IRForTarget::MaybeHandleCallArguments (CallInst *Old)
|
|||
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
|
||||
IRForTarget::ResolveCalls(BasicBlock &basic_block)
|
||||
{
|
||||
|
@ -1742,7 +1799,9 @@ IRForTarget::ResolveExternals (Function &llvm_function)
|
|||
(*global).getName().str().c_str(),
|
||||
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))
|
||||
{
|
||||
|
@ -1752,6 +1811,16 @@ IRForTarget::ResolveExternals (Function &llvm_function)
|
|||
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))
|
||||
{
|
||||
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