TypeResolver.java

package org.jruby.ext.ffi;

import org.jruby.Ruby;
import org.jruby.RubySymbol;
import org.jruby.RubyHash;
import org.jruby.runtime.builtin.IRubyObject;

import java.util.Collections;
import java.util.IdentityHashMap;
import java.util.Map;

public final class TypeResolver {
    private final FFI ffi;
    private volatile Map<RubySymbol, Type> symbolTypeCache = Collections.emptyMap();

    TypeResolver(FFI ffi) {
        this.ffi = ffi;
    }

    public final Type findType(Ruby runtime, IRubyObject name) {
        return findType(runtime, name, null);
    }

    public final Type findType(Ruby runtime, IRubyObject name, IRubyObject typeMap) {

        if (name instanceof Type) {
            return (Type) name;

        } else if (name instanceof RubySymbol) {
            Object obj = ((RubySymbol)name).getFFIHandle();
            if (obj instanceof Type) {
                return ((Type) obj);
            }

            if (typeMap != null && typeMap instanceof RubyHash) {
                Type type = (Type)((RubyHash)typeMap).get(name);
                if (type != null && !type.isNil()) {
                    return type;
                }
            }

            Type type = symbolTypeCache.get(name);
            if (type != null) {
                return type;
            }

            return lookupAndCacheType(runtime, (RubySymbol)name, (RubyHash)typeMap);

        } else {
            return lookupType(runtime, name, typeMap);
        }
    }

    private synchronized Type lookupAndCacheType(Ruby runtime, RubySymbol name, RubyHash typeMap) {
        Type type = lookupType(runtime, name, typeMap);

        Map<RubySymbol, Type> map = new IdentityHashMap<RubySymbol, Type>(symbolTypeCache);
        map.put(name, type);
        symbolTypeCache = map;
        name.setFFIHandle(type);

        return type;
    }

    private Type lookupType(Ruby runtime, IRubyObject name) {
        return lookupType(runtime, name, null);
    }

    private Type lookupType(Ruby runtime, IRubyObject name, IRubyObject typeMap) {
        IRubyObject type = ffi.typedefs.fastARef(name);
        if (type instanceof Type) {
            return (Type) type;
        }

        IRubyObject args[] = new IRubyObject[]{name, typeMap};
        if ((type = ffi.ffiModule.callMethod(runtime.getCurrentContext(), "find_type", args)) instanceof Type) {
            return (Type) type;
        }

        throw runtime.newTypeError("cannot resolve type " + name);
    }
}