package org.jruby.runtime.callsite;
import org.jruby.RubyClass;
import org.jruby.RubyFixnum;
import org.jruby.RubyFloat;
import org.jruby.internal.runtime.AbstractIRMethod;
import org.jruby.internal.runtime.methods.DynamicMethod;
import org.jruby.internal.runtime.methods.InterpretedIRMethod;
import org.jruby.internal.runtime.methods.MixedModeIRMethod;
import org.jruby.ir.IRManager;
import org.jruby.ir.IRMethod;
import org.jruby.ir.IRScope;
import org.jruby.runtime.Block;
import org.jruby.runtime.CallType;
import org.jruby.runtime.ThreadContext;
import org.jruby.runtime.builtin.IRubyObject;
import org.jruby.util.log.Logger;
import org.jruby.util.log.LoggerFactory;
public class ProfilingCachingCallSite extends CachingCallSite {
public static final Logger LOG = LoggerFactory.getLogger(ProfilingCachingCallSite.class);
private volatile int totalMonomorphicCalls = 0;
private final IRScope hostScope;
private final long callSiteId;
public ProfilingCachingCallSite(String methodName, IRScope scope, long callSiteId) {
super(methodName, CallType.NORMAL);
this.hostScope = scope;
this.callSiteId = callSiteId;
}
private void inlineCheck(ThreadContext context, IRubyObject self, CacheEntry cache) {
if (!hostScope.inliningAllowed()) return;
boolean targetIsIR = cache.method instanceof AbstractIRMethod;
boolean siteIsIR = hostScope.compilable != null;
AbstractIRMethod targetMethod;
if (!targetIsIR) {
if (self instanceof RubyFixnum && "times".equals(methodName)) {
targetIsIR = true;
targetMethod = new MixedModeIRMethod(context.runtime.getIRManager().loadInternalMethod(context, self, "times"), cache.method.getVisibility(), cache.method.getImplementationClass());
} else {
targetMethod = null;
}
} else {
targetMethod = (AbstractIRMethod) cache.method;
}
if (targetIsIR && siteIsIR) {
IRMethod scopeToInline = (IRMethod) (targetMethod).getIRScope();
if (IRManager.IR_INLINER_VERBOSE) LOG.info("PROFILE: " + hostScope + " -> " + scopeToInline + " - " + totalMonomorphicCalls);
AbstractIRMethod hostMethod = (AbstractIRMethod) hostScope.compilable;
if (hostMethod instanceof InterpretedIRMethod) {
hostScope.inlineMethod(scopeToInline, callSiteId, cache.token, false);
} else if (hostMethod instanceof MixedModeIRMethod) {
hostScope.inlineMethodJIT(scopeToInline, callSiteId, cache.token, false);
} else {
hostScope.inlineMethodCompiled(scopeToInline, callSiteId, cache.token, false);
}
}
}
protected boolean methodMissing(DynamicMethod method, IRubyObject caller) {
return method.isUndefined() || (!methodName.equals("method_missing") && !method.isCallableFrom(caller, callType));
}
@Override
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, long fixnum) {
return call(context, caller, self, RubyFixnum.newFixnum(context.runtime, fixnum));
}
@Override
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, double flote) {
return call(context, caller, self, RubyFloat.newFloat(context.runtime, flote));
}
@Override
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, IRubyObject... args) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, args);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, args, context, self);
}
}
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, IRubyObject[] args, Block block) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, args, block);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, block, args, context, self);
}
}
@Override
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, context, self);
}
}
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, Block block) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, block);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, block, context, self);
}
}
@Override
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, IRubyObject arg1) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, arg1);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, context, self, arg1);
}
}
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, IRubyObject arg1, Block block) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, arg1, block);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, block, context, self, arg1);
}
}
@Override
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, IRubyObject arg1, IRubyObject arg2) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, arg1, arg2);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, context, self, arg1, arg2);
}
}
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, IRubyObject arg1, IRubyObject arg2, Block block) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, arg1, arg2, block);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, block, context, self, arg1, arg2);
}
}
@Override
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, IRubyObject arg1, IRubyObject arg2, IRubyObject arg3) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, arg1, arg2, arg3);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, context, self, arg1, arg2, arg3);
}
}
public IRubyObject call(ThreadContext context, IRubyObject caller, IRubyObject self, IRubyObject arg1, IRubyObject arg2, IRubyObject arg3, Block block) {
RubyClass selfType = getClass(self);
CacheEntry cache = this.cache;
if (cache.typeOk(selfType)) {
if ((totalMonomorphicCalls++ % IRManager.IR_INLINER_THRESHOLD) == 0) inlineCheck(context, self, cache);
return cache.method.call(context, self, cache.sourceModule, methodName, arg1, arg2, arg3, block);
} else {
totalMonomorphicCalls = 1;
return cacheAndCall(caller, selfType, block, context, self, arg1, arg2, arg3);
}
}
private IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType, Block block,
IRubyObject[] args, ThreadContext context, IRubyObject self) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, args, block);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, args, block);
}
}
private IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType,
IRubyObject[] args, ThreadContext context, IRubyObject self) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, args);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, args);
}
}
private IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType,
ThreadContext context, IRubyObject self) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName);
}
}
private IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType, Block block,
ThreadContext context, IRubyObject self) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, block);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, block);
}
}
protected IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType, ThreadContext context, IRubyObject self, IRubyObject arg) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, arg);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, arg);
}
}
private IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType, Block block,
ThreadContext context, IRubyObject self, IRubyObject arg) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, arg, block);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, arg, block);
}
}
protected IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType, ThreadContext context, IRubyObject self, IRubyObject arg1, IRubyObject arg2) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, arg1, arg2);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, arg1, arg2);
}
}
private IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType, Block block,
ThreadContext context, IRubyObject self, IRubyObject arg1, IRubyObject arg2) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, arg1, arg2, block);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, arg1, arg2, block);
}
}
private IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType,
ThreadContext context, IRubyObject self, IRubyObject arg1, IRubyObject arg2,
IRubyObject arg3) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, arg1, arg2, arg3);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, arg1, arg2, arg3);
}
}
private IRubyObject cacheAndCall(IRubyObject caller, RubyClass selfType, Block block,
ThreadContext context, IRubyObject self, IRubyObject arg1, IRubyObject arg2,
IRubyObject arg3) {
CacheEntry entry = selfType.searchWithCache(methodName);
DynamicMethod method = entry.method;
if (methodMissing(method, caller)) {
return callMethodMissing(context, self, selfType, method, arg1, arg2, arg3, block);
} else {
cache = entry;
return method.call(context, self, entry.sourceModule, methodName, arg1, arg2, arg3, block);
}
}
}