org.jruby.truffle.language.supercall.SuperCallNode Maven / Gradle / Ivy
The newest version!
/*
* Copyright (c) 2013, 2017 Oracle and/or its affiliates. All rights reserved. This
* code is released under a tri EPL/GPL/LGPL license. You can use it,
* redistribute it and/or modify it under the terms of the:
*
* Eclipse Public License version 1.0
* GNU General Public License version 2
* GNU Lesser General Public License version 2.1
*/
package org.jruby.truffle.language.supercall;
import com.oracle.truffle.api.CompilerDirectives;
import com.oracle.truffle.api.frame.VirtualFrame;
import com.oracle.truffle.api.object.DynamicObject;
import com.oracle.truffle.api.profiles.ConditionProfile;
import org.jcodings.specific.UTF8Encoding;
import org.jruby.truffle.core.array.ArrayUtils;
import org.jruby.truffle.core.cast.ProcOrNullNode;
import org.jruby.truffle.core.cast.ProcOrNullNodeGen;
import org.jruby.truffle.language.RubyNode;
import org.jruby.truffle.language.arguments.RubyArguments;
import org.jruby.truffle.language.dispatch.CallDispatchHeadNode;
import org.jruby.truffle.language.dispatch.DispatchHeadNodeFactory;
import org.jruby.truffle.language.methods.CallInternalMethodNode;
import org.jruby.truffle.language.methods.CallInternalMethodNodeGen;
import org.jruby.truffle.language.methods.DeclarationContext;
import org.jruby.truffle.language.methods.InternalMethod;
public class SuperCallNode extends RubyNode {
private final ConditionProfile missingProfile = ConditionProfile.createBinaryProfile();
@Child private RubyNode arguments;
@Child private RubyNode block;
@Child private ProcOrNullNode procOrNullNode = ProcOrNullNodeGen.create(null);
@Child private LookupSuperMethodNode lookupSuperMethodNode = LookupSuperMethodNodeGen.create(null);
@Child private CallInternalMethodNode callMethodNode = CallInternalMethodNodeGen.create(null, new RubyNode[] {});
@Child private CallDispatchHeadNode callMethodMissingNode;
public SuperCallNode(RubyNode arguments, RubyNode block) {
this.arguments = arguments;
this.block = block;
}
@Override
public final Object execute(VirtualFrame frame) {
final Object self = RubyArguments.getSelf(frame);
// Execute the arguments
final Object[] superArguments = (Object[]) arguments.execute(frame);
// Execute the block
final DynamicObject blockObject = procOrNullNode.executeProcOrNull(block.execute(frame));
final InternalMethod superMethod = lookupSuperMethodNode.executeLookupSuperMethod(frame, self);
if (missingProfile.profile(superMethod == null)) {
final String name = RubyArguments.getMethod(frame).getSharedMethodInfo().getName(); // use the original name
final Object[] methodMissingArguments = ArrayUtils.unshift(superArguments, getContext().getSymbolTable().getSymbol(name));
return callMethodMissing(frame, self, blockObject, methodMissingArguments);
}
final Object[] frameArguments = RubyArguments.pack(null, null, superMethod, DeclarationContext.METHOD, null, self, blockObject, superArguments);
return callMethodNode.executeCallMethod(frame, superMethod, frameArguments);
}
@Override
public Object isDefined(VirtualFrame frame) {
final Object self = RubyArguments.getSelf(frame);
final InternalMethod superMethod = lookupSuperMethodNode.executeLookupSuperMethod(frame, self);
if (superMethod == null) {
return nil();
} else {
return create7BitString("super", UTF8Encoding.INSTANCE);
}
}
private Object callMethodMissing(VirtualFrame frame, Object receiver, DynamicObject block, Object[] arguments) {
if (callMethodMissingNode == null) {
CompilerDirectives.transferToInterpreterAndInvalidate();
callMethodMissingNode = insert(DispatchHeadNodeFactory.createMethodCallOnSelf());
}
return callMethodMissingNode.callWithBlock(frame, receiver, "method_missing", block, arguments);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy