All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.robovm.objc.ObjCRuntime Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2012 RoboVM AB
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.robovm.objc;

import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;

import org.robovm.rt.VM;
import org.robovm.rt.bro.Bro;
import org.robovm.rt.bro.Runtime;
import org.robovm.rt.bro.Struct;
import org.robovm.rt.bro.annotation.Bridge;
import org.robovm.rt.bro.annotation.ByVal;
import org.robovm.rt.bro.annotation.Library;
import org.robovm.rt.bro.annotation.Pointer;
import org.robovm.rt.bro.annotation.StructRet;
import org.robovm.rt.bro.ptr.BytePtr;

/**
 *
 */
@Library("objc")
public class ObjCRuntime {

    private static final Map, Integer> structSizes = new HashMap<>();
    private static final Map, Integer> structAttributes = new HashMap<>();

    static {
        Bro.bind(ObjCRuntime.class);
    }
    
    public static void bind() {
        Class caller = VM.getStackClasses(0, 1)[0];
        bind(caller);
    }
    
    public static void bind(Class c) {
        for (Method method : c.getDeclaredMethods()) {
            Bridge bridge = method.getAnnotation(Bridge.class);
            if (bridge != null && (bridge.symbol() == null || "".equals(bridge.symbol()))) {
                Class[] paramTypes = method.getParameterTypes();
                if (paramTypes.length >= 2) {
                    // An ObjC method takes at least 2 parameters (self, selector)
                    // or (super, selector).
                    if (paramTypes[1] == Selector.class) {
                        String symbol = null;
                        if (paramTypes[0] == ObjCSuper.class) {
                            symbol = "objc_msgSendSuper";
                        } else if (ObjCObject.class.isAssignableFrom(paramTypes[0])) {
                            // self should be an instance of ObjCObject
                            symbol = "objc_msgSend";
                        } else if (paramTypes[0] == long.class) {
                            // Also allow @Pointer long as type of self
                            Annotation[] paramAnnos = method.getParameterAnnotations()[0];
                            for (Annotation a : paramAnnos) {
                                if (a.annotationType() == Pointer.class) {
                                    symbol = "objc_msgSend";
                                    break;
                                }
                            }
                        }
                        if (symbol != null) {
                            // So this is a bridge to an ObjC method. If the method
                            // returns a struct by value we may have to use the
                            // special stret versions of objc_msgSend/objc_msgSendSuper.
                            if (isStret(method)) {
                                symbol += "_stret";
                            }
                            long f = Runtime.resolveBridge("objc", symbol, method);
                            VM.bindBridgeMethod(method, f);
                        }
                    }
                    
                }
            }
        }
        Bro.bind(c);
    }
    
    static boolean isStret(Method method) {
        Class returnType = method.getReturnType();
        if (returnType.getSuperclass() == Struct.class 
                && (method.getAnnotation(ByVal.class) != null 
                || returnType.getAnnotation(ByVal.class) != null)) {
            int structSize = getStructSize(returnType);
            if (Bro.IS_X86 && Bro.IS_32BIT) {
                if (structSize > 2 && structSize != 4 && structSize != 8) {
                    // On x86 stret has to be used for all structs except
                    // of size 1, 2, 4 and 8 bytes.
                    return true;
                }
            } else if (Bro.IS_X86 && Bro.IS_64BIT) {
                if (structSize > 16) {
                    return true;
                }

                // System V Application Binary Interface AMD64 Architecture Processor Supplement.
                // 3.2.3: If it contains un- aligned fields, it has class MEMORY
                int attr = getStructAttributes(returnType);
                if ((attr & ATTR_UNALIGNED) != 0) {
                    return true;
                }
            } else if (Bro.IS_ARM && Bro.IS_32BIT) {
                if (structSize > 4) {
                    // On ARM stret has to be used for structs
                    // larger than 4 bytes
                    return true;
                }

                // only Integer like structures are returned in r0.
                int attr = getStructAttributes(returnType);
                if ((attr & ATTR_NOT_SINGLE_INT_STRUCT) != 0) {
                    return true;
                }
            } else if (Bro.IS_ARM && Bro.IS_64BIT) {
                // iOS ARM64 doesn't have objc_msgSend_stret
                return false;
            } else {
                throw new Error("Unsupported architecture");
            }
        }
        return false;
    }
    
    static synchronized int getStructSize(Class cls) {
        Integer size = structSizes.get(cls);
        if (size == null) {
            try {
                Method sizeOf = cls.getMethod("sizeOf");
                size = (Integer) sizeOf.invoke(null);
            } catch (Exception e) {
                throw new Error(e);
            }
            structSizes.put(cls, size);
        }
        return size;
    }

    final static int ATTR_UNALIGNED                    = 1 << 0; // contains at least one field that is not aligned to its nature alignment
    public final static int ATTR_NOT_SINGLE_INT_STRUCT = 1 << 1; // indicates that structure is not one integer field wrap (up to 32 bit one)
    static synchronized int getStructAttributes(Class cls) {
        Integer attr = structAttributes.get(cls);
        if (attr == null) {
            try {
                Method stretMetadata = cls.getMethod("$attr$stretMetadata");
                attr = (Integer) stretMetadata.invoke(null);
            } catch (Exception e) {
                throw new Error(e);
            }
            structAttributes.put(cls, attr);
        }
        return attr;
    }

    /* selector */
    
    @Bridge
    public static native BytePtr sel_getName(Selector sel);

    @Bridge
    public static native Selector sel_registerName(BytePtr str);
    
    /* objc */
    
    @Bridge
    public static native @Pointer long objc_getClass(@Pointer long name);
    
    @Bridge
    public static native @Pointer long objc_getMetaClass(@Pointer long name);

    @Bridge
    public static native @Pointer long objc_allocateClassPair(@Pointer long superclass, @Pointer long name, @Pointer long extraBytes);
    
    @Bridge
    public static native void objc_registerClassPair(@Pointer long cls);
    
    @Bridge
    public static native @Pointer long objc_getProtocol(@Pointer long name);
    
    @Bridge
    public static native @Pointer long objc_allocateProtocol(@Pointer long name);
    
    @Bridge
    public static native void objc_registerProtocol(@Pointer long protocol);
    
    @Bridge
    public static native void objc_setAssociatedObject(@Pointer long object, @Pointer long key, @Pointer long value, int policy);

    @Bridge
    public static native @Pointer long objc_getAssociatedObject(@Pointer long object, @Pointer long key);

    @Bridge
    public static native @Pointer long objc_copyProtocolList (@Pointer long outCount);

    /* object */
    
    @Bridge
    public static native @Pointer long object_getClass(@Pointer long object);
    
    @Bridge
    public static native @Pointer long object_setClass(@Pointer long object, @Pointer long cls);

    @Bridge
    public static native @Pointer long object_getClassName(@Pointer long object);
    
    @Bridge
    public static native @Pointer long object_dispose(@Pointer long object);
    
    @Bridge
    public static native @Pointer long object_getIvar(@Pointer long object, @Pointer long ivar);

    @Bridge
    public static native void object_setIvar(@Pointer long object, @Pointer long ivar, @Pointer long value);
    
    /* class */

    @Bridge
    public static native @Pointer long class_getSuperclass(@Pointer long cls);

    @Bridge
    public static native @Pointer long class_isMetaClass(@Pointer long cls);
    
    @Bridge
    public static native @Pointer long class_getName(@Pointer long cls);
    
    @Bridge
    public static native @Pointer long class_getInstanceMethod(@Pointer long aClass, @Pointer long aSelector);

    @Bridge
    public static native @Pointer long class_getClassMethod(@Pointer long aClass, @Pointer long aSelector);
    
    @Bridge
    public static native  boolean class_addProtocol(@Pointer long cls, @Pointer long protocol);
    
    @Bridge
    public static native boolean class_conformsToProtocol(@Pointer long cls, @Pointer long protocol);

    @Bridge
    public static native @Pointer long class_copyProtocolList(@Pointer long cls, @Pointer long outCount);
    
    @Bridge
    public static native @Pointer long class_copyMethodList(@Pointer long cls, @Pointer long outCount);

    @Bridge
    public static native @Pointer long class_copyPropertyList(@Pointer long cls, @Pointer long outCount);

    @Bridge
    public static native boolean class_addMethod(@Pointer long cls, @Pointer long name, @Pointer long imp, @Pointer long types);
    
    @Bridge
    public static native @Pointer long class_replaceMethod(@Pointer long cls, @Pointer long name, @Pointer long imp, @Pointer long types);
    
    @Bridge
    public static native @Pointer long class_getMethodImplementation(@Pointer long cls, @Pointer long name);
    
    @Bridge
    public static native boolean class_addIvar(@Pointer long cls, @Pointer long name, int size, byte alignment, @Pointer long types);
    
    @Bridge
    public static native @Pointer long class_getInstanceVariable(@Pointer long cls, @Pointer long name);
    
    @Bridge
    public static native @Pointer long class_getClassVariable(@Pointer long cls, @Pointer long name);
    
    @Bridge
    public static native @Pointer long class_getIvarLayout(@Pointer long cls);
    
    @Bridge
    public static native void class_setIvarLayout(@Pointer long cls, @Pointer long layout);
    
    @Bridge
    public static native @Pointer long class_getProperty(@Pointer long cls, @Pointer long name);
    
    @Bridge
    public static native boolean class_respondsToSelector(@Pointer long cls, @Pointer long sel);
    
    /* protocol */
    
    @Bridge
    public static native @Pointer long protocol_getName(@Pointer long protocol);
    
    @Bridge
    public static native void protocol_addProtocol(@Pointer long protocol, @Pointer long addition);
    
    @Bridge
    public static native boolean protocol_conformsToProtocol(@Pointer long protocol, @Pointer long other);

    @Bridge
    public static native @Pointer long protocol_getProperty(@Pointer long protocol, @Pointer long name, boolean isRequiredProperty, boolean isInstanceProperty);
    
    @Bridge
    public static native @Pointer long protocol_copyProtocolList(@Pointer long protocol, @Pointer long outCount);
    
    /* ivar */
    
    @Bridge
    public static native @Pointer long ivar_getName(@Pointer long ivar);

    @Bridge
    public static native @Pointer long ivar_getTypeEncoding(@Pointer long ivar);
    
    @Bridge
    public static native int ivar_getOffset(@Pointer long ivar);
    
    /* method */
    
    @Bridge
    public static native @Pointer long method_getName(@Pointer long m);
    
    @Bridge
    public static native @Pointer long method_copyReturnType(@Pointer long m);

    @Bridge
    public static native @Pointer long method_copyArgumentType(@Pointer long m, int index);
    
    @Bridge
    public static native int method_getNumberOfArguments(@Pointer long m);
    
    @Bridge
    public static native @Pointer long method_getTypeEncoding(@Pointer long method);
    
    @Bridge
    public static native @Pointer long method_getImplementation(@Pointer long method);
    
    @Bridge
    public static native @Pointer long method_setImplementation(@Pointer long m, @Pointer long imp);
    
    @Bridge
    public static native void method_exchangeImplementations(@Pointer long m1, @Pointer long m2);
    
    /* property */
    
    @Bridge
    public static native @Pointer long property_getName(@Pointer long property);

    @Bridge
    public static native BytePtr property_getAttributes(@Pointer long property);

    @Bridge
    public static native BytePtr property_copyAttributeValue(@Pointer long property, BytePtr attributeName);
    
    /* message */
    
    @Bridge(symbol = "objc_msgSend")
    public static native @Pointer long ptr_objc_msgSend(@Pointer long receiver, @Pointer long selector);

    @Bridge(symbol = "objc_msgSendSuper")
    public static native @Pointer long ptr_objc_msgSendSuper(@Pointer long zuper, @Pointer long selector);

    @Bridge(symbol = "objc_msgSend")
    public static native void void_objc_msgSend(@Pointer long receiver, @Pointer long selector);

    @Bridge(symbol = "objc_msgSendSuper")
    public static native void void_objc_msgSendSuper(@Pointer long zuper, @Pointer long selector);
    
    @Bridge(symbol = "objc_msgSend")
    public static native boolean boolean_objc_msgSend(@Pointer long receiver, @Pointer long selector);
    
    @Bridge(symbol = "objc_msgSend")
    public static native void void_objc_msgSend_ptr(@Pointer long receiver, @Pointer long selector, @Pointer long ptr);
    
    @Bridge(symbol = "objc_msgSend")
    public static native void void_objc_msgSend_boolean(@Pointer long receiver, @Pointer long selector, boolean b);
    
    @Bridge(symbol = "objc_msgSend")
    public static native int int_objc_msgSend(@Pointer long receiver, @Pointer long selector);

    @Bridge(symbol = "objc_msgSend")
    public static native @Pointer long ptr_objc_msgSend_ptr(@Pointer long receiver, @Pointer long selector, @Pointer long ptr);
    
    @Bridge(symbol = "objc_msgSend")
    public static native @Pointer long ptr_objc_msgSend_int(@Pointer long receiver, @Pointer long selector, int i);
    
    @Bridge(symbol = "objc_msgSend")
    public static native @Pointer long ptr_objc_msgSend_long(@Pointer long receiver, @Pointer long selector, long l);

    @Bridge(symbol = "objc_msgSend")
    public static native char char_objc_msgSend_int(@Pointer long receiver, @Pointer long selector, int i);
    
    @Bridge(symbol = "objc_msgSend_stret", optional = true) // Optional since it's not available on iOS ARM64
    public static native void objc_msgSend_stret(@StructRet @Pointer long ret, @Pointer long receiver, @Pointer long selector);
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy