org.jruby.runtime.MethodIndex Maven / Gradle / Ivy
/*
***** BEGIN LICENSE BLOCK *****
* Version: EPL 1.0/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Eclipse 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/epl-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.
*
* Copyright (C) 2006-2007 Thomas E Enebo
* Copyright (C) 2006-2007 Charles Nutter
*
* 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 EPL, 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 EPL, the GPL or the LGPL.
***** END LICENSE BLOCK *****/
package org.jruby.runtime;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import org.jruby.runtime.callsite.LtCallSite;
import org.jruby.runtime.callsite.LeCallSite;
import org.jruby.runtime.callsite.MinusCallSite;
import org.jruby.runtime.callsite.MulCallSite;
import org.jruby.runtime.callsite.NormalCachingCallSite;
import org.jruby.runtime.callsite.GtCallSite;
import org.jruby.runtime.callsite.PlusCallSite;
import org.jruby.runtime.callsite.GeCallSite;
import org.jruby.RubyInstanceConfig;
import org.jruby.runtime.callsite.CmpCallSite;
import org.jruby.runtime.callsite.EqCallSite;
import org.jruby.runtime.callsite.BitAndCallSite;
import org.jruby.runtime.callsite.BitOrCallSite;
import org.jruby.runtime.callsite.FunctionalCachingCallSite;
import org.jruby.runtime.callsite.RespondToCallSite;
import org.jruby.runtime.callsite.ShiftLeftCallSite;
import org.jruby.runtime.callsite.ShiftRightCallSite;
import org.jruby.runtime.callsite.SuperCallSite;
import org.jruby.runtime.callsite.VariableCachingCallSite;
import org.jruby.runtime.callsite.XorCallSite;
import org.jruby.runtime.invokedynamic.MethodNames;
import org.jruby.util.log.Logger;
import org.jruby.util.log.LoggerFactory;
/**
*
* @author headius
*/
public class MethodIndex {
private static final boolean DEBUG = false;
private static final Logger LOG = LoggerFactory.getLogger(MethodIndex.class);
@Deprecated
public static final int NO_METHOD = MethodNames.DUMMY.ordinal();
@Deprecated
public static final int OP_EQUAL = MethodNames.OP_EQUAL.ordinal();
@Deprecated
public static final int EQL = MethodNames.EQL.ordinal();
@Deprecated
public static final int HASH = MethodNames.HASH.ordinal();
@Deprecated
public static final int OP_CMP = MethodNames.OP_CMP.ordinal();
@Deprecated
public static final int MAX_METHODS = MethodNames.values().length;
@Deprecated
public static final String[] METHOD_NAMES = {
"",
"==",
"eql?",
"hash",
"<=>"
};
public static final Set FRAME_AWARE_METHODS = Collections.synchronizedSet(new HashSet());
public static final Set SCOPE_AWARE_METHODS = Collections.synchronizedSet(new HashSet());
public static CallSite getCallSite(String name) {
// fast and safe respond_to? call site logic
if (name.equals("respond_to?")) return new RespondToCallSite();
// only use fast ops if we're not tracing
if (RubyInstanceConfig.FASTOPS_COMPILE_ENABLED &&
!(RubyInstanceConfig.FULL_TRACE_ENABLED)) return getFastFixnumOpsCallSite(name);
return new NormalCachingCallSite(name);
}
private static final Map FIXNUM_OPS;
static {
String[][] fastFixnumOps = {
{"+", "op_plus"},
{"-", "op_minus"},
{"*", "op_mul"},
{"==", "op_equal"},
{"<", "op_lt"},
{"<=", "op_le"},
{">", "op_gt"},
{">=", "op_ge"},
{"<=>", "op_cmp"},
{"&", "op_and"},
{"|", "op_or"},
{"^", "op_xor"},
{">>", "op_rshift"},
{"<<", "op_lshift"}
};
FIXNUM_OPS = new HashMap<>(fastFixnumOps.length + 1, 1);
for (String[] fastOp : fastFixnumOps) FIXNUM_OPS.put(fastOp[0], fastOp[1]);
}
private static final Map FLOAT_OPS;
static {
final String[][] fastFloatOps = {
{"+", "op_plus"},
{"-", "op_minus"},
{"*", "op_mul"},
{"==", "op_equal"},
{"<", "op_lt"},
{"<=", "op_le"},
{">", "op_gt"},
{">=", "op_ge"},
{"<=>", "op_cmp"}
};
FLOAT_OPS = new HashMap<>(fastFloatOps.length + 1, 1);
for (String[] fastOp : fastFloatOps) FLOAT_OPS.put(fastOp[0], fastOp[1]);
}
public static boolean hasFastFixnumOps(String name) {
return FIXNUM_OPS.containsKey(name);
}
public static String getFastFixnumOpsMethod(String name) {
return FIXNUM_OPS.get(name);
}
public static CallSite getFastFixnumOpsCallSite(String name) {
switch (name) {
case "+" : return new PlusCallSite();
case "-" : return new MinusCallSite();
case "*" : return new MulCallSite();
case "<" : return new LtCallSite();
case "<=" : return new LeCallSite();
case ">" : return new GtCallSite();
case ">=" : return new GeCallSite();
case "==" : return new EqCallSite();
case "<=>" : return new CmpCallSite();
case "&" : return new BitAndCallSite();
case "|" : return new BitOrCallSite();
case "^" : return new XorCallSite();
case ">>" : return new ShiftRightCallSite();
case "<<" : return new ShiftLeftCallSite();
}
return new NormalCachingCallSite(name);
}
public static boolean hasFastFloatOps(String name) {
return FLOAT_OPS.containsKey(name);
}
public static String getFastFloatOpsMethod(String name) {
return FLOAT_OPS.get(name);
}
public static CallSite getFastFloatOpsCallSite(String name) {
if (name.equals("+")) {
return new PlusCallSite();
} else if (name.equals("-")) {
return new MinusCallSite();
} else if (name.equals("*")) {
return new MulCallSite();
} else if (name.equals("<")) {
return new LtCallSite();
} else if (name.equals("<=")) {
return new LeCallSite();
} else if (name.equals(">")) {
return new GtCallSite();
} else if (name.equals(">=")) {
return new GeCallSite();
} else if (name.equals("==")) {
return new EqCallSite();
} else if (name.equals("<=>")) {
return new CmpCallSite();
}
return new NormalCachingCallSite(name);
}
public static CallSite getFunctionalCallSite(String name) {
return new FunctionalCachingCallSite(name);
}
public static CallSite getVariableCallSite(String name) {
return new VariableCachingCallSite(name);
}
public static CallSite getSuperCallSite() {
return new SuperCallSite();
}
public static void addFrameAwareMethods(String... methods) {
if (DEBUG) LOG.debug("Adding frame-aware method names: {}", Arrays.toString(methods));
FRAME_AWARE_METHODS.addAll(Arrays.asList(methods));
}
public static void addScopeAwareMethods(String... methods) {
if (DEBUG) LOG.debug("Adding scope-aware method names: {}", Arrays.toString(methods));
SCOPE_AWARE_METHODS.addAll(Arrays.asList(methods));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy