Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Calling Objective-C method from imp

I'm trying to get a Method at runtime and then use its data structure to call it's implementation. Just to clarify, this is for learning purposes, not for any practical reason. So here is my code.

#import <Foundation/Foundation.h>
#import <stdio.h>

#import <objc/runtime.h>

@interface Test : NSObject 
-(void)method;
@end

@implementation Test

-(void)method {
    puts("this is a method");
}

@end 

int main(int argc, char *argv[]) {

    struct objc_method *t = (struct objc_method*) class_getInstanceMethod([Test class], @selector(method));
    Test *ztest = [Test new];

    (t->method_imp)(ztest, t->method_name);

    [ztest release];
    return 0;
}

The definition of struct objc_method is as follows (defined in objc/runtime.h)

typedef struct objc_method *Method;

....

struct objc_method {
    SEL method_name                                          OBJC2_UNAVAILABLE;
    char *method_types                                       OBJC2_UNAVAILABLE;
    IMP method_imp                                           OBJC2_UNAVAILABLE;
}                                                            OBJC2_UNAVAILABLE;

however when I try to compile my code, I get this error.

error: dereferencing pointer to incomplete type

But when I add this to my code (to explicitly declare an objc_method), it works just as expected.

struct objc_method {
    SEL method_name;
    char *method_types;
    IMP method_imp;
};

typedef struct objc_method* Method;

Could someone explain to me why my code works when I explicitly declare this structure, and not when I import it from objc/runtime.h? Does it have anything to do with OBJC2_UNAVAILABLE? I can't find a definition for that, but it is defined in my environment.

EDIT:

I ran gcc -E code.m -o out.m to see what OBJC2_UNAVAILABLE was getting replaced with, it turns out that OBJC2_UNAVAILABLE was defined as __attribute__((unavailable)) in my environment. Can someone explain what that means and why Method still works if this structure is "unavailable"?

like image 419
John Corbett Avatar asked Aug 02 '12 05:08

John Corbett


People also ask

What is IMP objective-C?

IMP is a C type referring to the implementation of a method, also known as an implementation pointer. It is a pointer to the start of a method implementation. Syntax: id (*IMP)(id, SEL, …)

What is IMP method?

Immediate Payment Service (IMPS) is a real-time fund transfer service that is available on a 24/7 basis on all 365 days a year. IMPS services can be availed even on bank holidays. The funds sent through this method of money transfer is credited immediately to the payee/beneficiary account.


2 Answers

I just took a good look at the objc runtime header and found what the problem is, and how to fix it :)

So if you look at the area of the file that contains the transparent definition of the structure, you'll see that it's in the body of:

#if !__OBJC2__

...

#endif

Since this IS defined, it means that the structure that we refer to is in fact forward-declared and opaque (and thus incomplete).

What we must do instead is use the functions provided for accessing these members:

IMP method_getImplementation(Method method);
SEL method_getName(Method method);
void method_getReturnType(Method method, char *dst, size_t dst_len);

Etc.

You can see the full list of accessor methods and plenty of other goodies in the Runtime Reference Guide.

Hope it helps!

like image 62
mamackenzie Avatar answered Oct 17 '22 16:10

mamackenzie


Its fields were previously defined, but it's an opaque type in ObjC-2. Use the runtime instead, and do not define the fields yourself:

int main(int argc, char *argv[]) {

    Method t = class_getInstanceMethod([Test class], @selector(method));
    Test * ztest = [Test new];

    IMP imp = method_getImplementation(t);

    typedef void (*fn)(id,SEL);
    fn f = (fn)imp;
    f(ztest,@selector(method));

    [ztest release];
    return 0;
}
like image 42
justin Avatar answered Oct 17 '22 17:10

justin