Start line:  
End line:  

Snippet Preview

Snippet HTML Code

Stack Overflow Questions
  /*
   ***** BEGIN LICENSE BLOCK *****
   * Version: CPL 1.0/GPL 2.0/LGPL 2.1
   *
   * The contents of this file are subject to the Common Public
   * License Version 1.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.eclipse.org/legal/cpl-v10.html
   *
  * Software distributed under the License is distributed on an "AS
  * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
  * implied. See the License for the specific language governing
  * rights and limitations under the License.
  * 
  * Alternatively, the contents of this file may be used under the terms of
  * either of the GNU General Public License Version 2 or later (the "GPL"),
  * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  * in which case the provisions of the GPL or the LGPL are applicable instead
  * of those above. If you wish to allow use of your version of this file only
  * under the terms of either the GPL or the LGPL, and not to allow others to
  * use your version of this file under the terms of the CPL, indicate your
  * decision by deleting the provisions above and replace them with the notice
  * and other provisions required by the GPL or the LGPL. If you do not delete
  * the provisions above, a recipient may use your version of this file under
  * the terms of any one of the CPL, the GPL or the LGPL.
  ***** END LICENSE BLOCK *****/
 
 package org.jruby.runtime.invokedynamic;
 
 import org.jruby.Ruby;
 import static java.lang.invoke.MethodHandles.*;
 import static java.lang.invoke.MethodType.*;
 
 public class MathLinker {
     private static final Logger LOG = LoggerFactory.getLogger("MathLinker");
     
     public static CallSite fixnumOperatorBootstrap(Lookup lookupString nameMethodType typelong valueString fileint linethrows NoSuchMethodExceptionIllegalAccessException {
         String[] names = name.split(":");
         String operator = JavaNameMangler.demangleMethodName(names[1]);
         JRubyCallSite site = new JRubyCallSite(lookuptype.filelineoperatorfalsefalsetrue);
         
         MethodHandle target = lookup.findStatic(MathLinker.class"fixnumOperator"
                 methodType(IRubyObject.classThreadContext.classIRubyObject.classIRubyObject.classJRubyCallSite.classlong.class));
         target = insertArguments(target, 3, sitevalue);
         
         site.setTarget(target);
         return site;
     }
     
     public static CallSite fixnumBooleanBootstrap(Lookup lookupString nameMethodType typelong valueString fileint linethrows NoSuchMethodExceptionIllegalAccessException {
         String[] names = name.split(":");
         String operator = JavaNameMangler.demangleMethodName(names[1]);
         JRubyCallSite site = new JRubyCallSite(lookuptype.filelineoperatorfalsefalsetrue);
         
         MethodHandle target = lookup.findStatic(MathLinker.class"fixnumBoolean"
                 methodType(boolean.classThreadContext.classIRubyObject.classIRubyObject.classJRubyCallSite.classlong.class));
         target = insertArguments(target, 3, sitevalue);
         
         site.setTarget(target);
         return site;
     }
     
     public static CallSite floatOperatorBootstrap(Lookup lookupString nameMethodType typedouble valueString fileint linethrows NoSuchMethodExceptionIllegalAccessException {
         String[] names = name.split(":");
         String operator = JavaNameMangler.demangleMethodName(names[1]);
         JRubyCallSite site = new JRubyCallSite(lookuptype.filelineoperatorfalsefalsetrue);
         
         MethodHandle target = lookup.findStatic(MathLinker.class"floatOperator"
                 methodType(IRubyObject.classThreadContext.classIRubyObject.classIRubyObject.classJRubyCallSite.classdouble.class));
         target = insertArguments(target, 3, sitevalue);
         
         site.setTarget(target);
         return site;
     }
     
     public static IRubyObject fixnumOperator(ThreadContext contextIRubyObject callerIRubyObject selfJRubyCallSite sitelong valuethrows Throwable {
         String operator = site.name();
         String opMethod = MethodIndex.getFastFixnumOpsMethod(operator);
         String name = "fixnum_" + opMethod;
         MethodType type = methodType(IRubyObject.classThreadContext.classIRubyObject.classIRubyObject.class);
         MethodHandle target = null;
         
         if (operator.equals("+") || operator.equals("-")) {
             if (value == 1) {
                name += "_one";
                target = lookup().findStatic(MathLinker.classnametype);
            } else if (value == 2) {
                name += "_two";
                target = lookup().findStatic(MathLinker.classnametype);
            }
        }
        
        if (target == null) {
            type = type.insertParameterTypes(3, long.class);
            target = lookup().findStatic(MathLinker.classnametype);
            target = insertArguments(target, 3, value);
        }
        MethodHandle fallback = lookup().findStatic(MathLinker.class"fixnumOperatorFail"
                methodType(IRubyObject.classThreadContext.classIRubyObject.classIRubyObject.classJRubyCallSite.classRubyFixnum.class));
        fallback = insertArguments(fallback, 3, sitecontext.runtime.newFixnum(value));
        
        MethodHandle test = lookup().findStatic(MathLinker.class"fixnumTest"methodType(boolean.classRuby.classIRubyObject.class));
        test = test.bindTo(context.runtime);
        test = permuteArguments(testmethodType(boolean.classThreadContext.classIRubyObject.classIRubyObject.class), new int[] {2});
        
        if (..info(name + "\tFixnum operation at site #" + site.siteID() + " (" + site.file() + ":" + site.line() + ") bound directly");
        site.setTarget(guardWithTest(testtargetfallback));
        return (IRubyObject)site.getTarget().invokeWithArguments(contextcallerself);
    }
    
    public static boolean fixnumBoolean(ThreadContext contextIRubyObject callerIRubyObject selfJRubyCallSite sitelong valuethrows Throwable {
        String operator = site.name();
        String opMethod = MethodIndex.getFastFixnumOpsMethod(operator);
        String name = "fixnum_boolean_" + opMethod;
        MethodType type = methodType(boolean.classThreadContext.classIRubyObject.classIRubyObject.class);
        MethodHandle target = null;
        
        if (target == null) {
            type = type.insertParameterTypes(3, long.class);
            target = lookup().findStatic(MathLinker.classnametype);
            target = insertArguments(target, 3, value);
        }
        MethodHandle fallback = lookup().findStatic(MathLinker.class"fixnumBooleanFail"
                methodType(boolean.classThreadContext.classIRubyObject.classIRubyObject.classJRubyCallSite.classRubyFixnum.class));
        fallback = insertArguments(fallback, 3, sitecontext.runtime.newFixnum(value));
        
        MethodHandle test = lookup().findStatic(MathLinker.class"fixnumTest"methodType(boolean.classRuby.classIRubyObject.class));
        test = test.bindTo(context.runtime);
        test = permuteArguments(testmethodType(boolean.classThreadContext.classIRubyObject.classIRubyObject.class), new int[] {2});
        
        if (..info(name + "\tFixnum boolean operation at site #" + site.siteID() + " (" + site.file() + ":" + site.line() + ") bound directly");
        site.setTarget(guardWithTest(testtargetfallback));
        return (Boolean)site.getTarget().invokeWithArguments(contextcallerself);
    }
    
    public static boolean fixnumTest(Ruby runtimeIRubyObject self) {
        return self instanceof RubyFixnum && !runtime.isFixnumReopened();
    }
    
    public static IRubyObject fixnumOperatorFail(ThreadContext contextIRubyObject callerIRubyObject selfJRubyCallSite siteRubyFixnum valuethrows Throwable {
        String operator = site.name();
        RubyClass selfClass = InvokeDynamicSupport.pollAndGetClass(contextself);
        CacheEntry entry = site.entry;
        
        if (entry.typeOk(selfClass)) {
            return entry.method.call(contextselfselfClassoperatorvalue);
        } else {
            entry = selfClass.searchWithCache(operator);
            if (InvokeDynamicSupport.methodMissing(entrysite.callType(), operatorcaller)) {
                return InvokeDynamicSupport.callMethodMissing(entrysite.callType(), contextselfoperatorvalue);
            }
            site.entry = entry;
            return entry.method.call(contextselfselfClassoperatorvalue);
        }
    }
    
    public static boolean fixnumBooleanFail(ThreadContext contextIRubyObject callerIRubyObject selfJRubyCallSite siteRubyFixnum valuethrows Throwable {
        return fixnumOperatorFail(contextcallerselfsitevalue).isTrue();
    }
    public static IRubyObject fixnum_op_plus(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_plus(contextvalue);
    }
    public static IRubyObject fixnum_op_minus(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_minus(contextvalue);
    }
    public static IRubyObject fixnum_op_mul(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_mul(contextvalue);
    }
    public static IRubyObject fixnum_op_equal(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_equal(contextvalue);
    }
    public static IRubyObject fixnum_op_lt(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_lt(contextvalue);
    }
    public static IRubyObject fixnum_op_le(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_le(contextvalue);
    }
    public static IRubyObject fixnum_op_gt(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_gt(contextvalue);
    }
    public static IRubyObject fixnum_op_ge(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_ge(contextvalue);
    }
    public static boolean fixnum_boolean_op_equal(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_equal_boolean(contextvalue);
    }
    public static boolean fixnum_boolean_op_lt(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_lt_boolean(contextvalue);
    }
    public static boolean fixnum_boolean_op_le(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_le_boolean(contextvalue);
    }
    public static boolean fixnum_boolean_op_gt(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_gt_boolean(contextvalue);
    }
    public static boolean fixnum_boolean_op_ge(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_ge_boolean(contextvalue);
    }
    public static IRubyObject fixnum_op_cmp(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_cmp(contextvalue);
    }
    public static IRubyObject fixnum_op_and(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_and(contextvalue);
    }
    public static IRubyObject fixnum_op_or(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_or(contextvalue);
    }
    public static IRubyObject fixnum_op_xor(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_xor(contextvalue);
    }
    public static IRubyObject fixnum_op_rshift(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_rshift(value);
    }
    public static IRubyObject fixnum_op_lshift(ThreadContext contextIRubyObject callerIRubyObject selflong valuethrows Throwable {
        return ((RubyFixnum)self).op_lshift(value);
    }
    public static IRubyObject fixnum_op_plus_one(ThreadContext contextIRubyObject callerIRubyObject selfthrows Throwable {
        return ((RubyFixnum)self).op_plus_one(context);
    }
    public static IRubyObject fixnum_op_minus_one(ThreadContext contextIRubyObject callerIRubyObject selfthrows Throwable {
        return ((RubyFixnum)self).op_minus_one(context);
    }
    public static IRubyObject fixnum_op_plus_two(ThreadContext contextIRubyObject callerIRubyObject selfthrows Throwable {
        return ((RubyFixnum)self).op_plus_two(context);
    }
    public static IRubyObject fixnum_op_minus_two(ThreadContext contextIRubyObject callerIRubyObject selfthrows Throwable {
        return ((RubyFixnum)self).op_minus_two(context);
    }
    
    public static IRubyObject floatOperator(ThreadContext contextIRubyObject callerIRubyObject selfJRubyCallSite sitedouble valuethrows Throwable {
        String operator = site.name();
        String opMethod = MethodIndex.getFastFixnumOpsMethod(operator);
        String name = "float_" + opMethod;
        MethodType type = methodType(IRubyObject.classThreadContext.classIRubyObject.classIRubyObject.class);
        MethodHandle target = null;
        
        if (target == null) {
            type = type.insertParameterTypes(3, double.class);
            target = lookup().findStatic(MathLinker.classnametype);
            target = insertArguments(target, 3, value);
        }
        MethodHandle fallback = lookup().findStatic(MathLinker.class"floatOperatorFail"
                methodType(IRubyObject.classThreadContext.classIRubyObject.classIRubyObject.classJRubyCallSite.classRubyFloat.class));
        fallback = insertArguments(fallback, 3, sitecontext.runtime.newFloat(value));
        
        MethodHandle test = lookup().findStatic(MathLinker.class"floatTest"methodType(boolean.classRuby.classIRubyObject.class));
        test = test.bindTo(context.runtime);
        test = permuteArguments(testmethodType(boolean.classThreadContext.classIRubyObject.classIRubyObject.class), new int[] {2});
        
        if (..info(name + "\tFloat operation at site #" + site.siteID() + " (" + site.file() + ":" + site.line() + ") bound directly");
        site.setTarget(guardWithTest(testtargetfallback));
        return (IRubyObject)site.getTarget().invokeWithArguments(contextcallerself);
    }
    
    public static boolean floatTest(Ruby runtimeIRubyObject self) {
        return self instanceof RubyFloat && !runtime.isFloatReopened();
    }
    
    public static IRubyObject floatOperatorFail(ThreadContext contextIRubyObject callerIRubyObject selfJRubyCallSite siteRubyFloat valuethrows Throwable {
        String operator = site.name();
        RubyClass selfClass = InvokeDynamicSupport.pollAndGetClass(contextself);
        CacheEntry entry = site.entry;
        
        if (entry.typeOk(selfClass)) {
            return entry.method.call(contextselfselfClassoperatorvalue);
        } else {
            entry = selfClass.searchWithCache(operator);
            if (InvokeDynamicSupport.methodMissing(entrysite.callType(), operatorcaller)) {
                return InvokeDynamicSupport.callMethodMissing(entrysite.callType(), contextselfoperatorvalue);
            }
            site.entry = entry;
            return entry.method.call(contextselfselfClassoperatorvalue);
        }
    }
    public static IRubyObject float_op_plus(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_plus(contextvalue);
    }
    public static IRubyObject float_op_minus(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_minus(contextvalue);
    }
    public static IRubyObject float_op_mul(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_mul(contextvalue);
    }
    public static IRubyObject float_op_equal(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_equal(contextvalue);
    }
    public static IRubyObject float_op_lt(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_lt(contextvalue);
    }
    public static IRubyObject float_op_le(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_le(contextvalue);
    }
    public static IRubyObject float_op_gt(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_gt(contextvalue);
    }
    public static IRubyObject float_op_ge(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_ge(contextvalue);
    }
    public static IRubyObject float_op_cmp(ThreadContext contextIRubyObject callerIRubyObject selfdouble valuethrows Throwable {
        return ((RubyFloat)self).op_cmp(contextvalue);
    }
New to GrepCode? Check out our FAQ X