org.jruby.ir.instructions.EQQInstr Maven / Gradle / Ivy
package org.jruby.ir.instructions;
import org.jruby.RubyArray;
import org.jruby.ir.IRVisitor;
import org.jruby.ir.Operation;
import org.jruby.ir.operands.Operand;
import org.jruby.ir.operands.UndefinedValue;
import org.jruby.ir.operands.Variable;
import org.jruby.ir.transformations.inlining.InlinerInfo;
import org.jruby.runtime.Block;
import org.jruby.runtime.DynamicScope;
import org.jruby.runtime.ThreadContext;
import org.jruby.runtime.builtin.IRubyObject;
import java.util.Map;
// If v2 is an array, compare v1 with every element of v2 and stop on first match!
public class EQQInstr extends Instr implements ResultInstr {
private Operand arg1;
private Operand arg2;
private Variable result;
public EQQInstr(Variable result, Operand v1, Operand v2) {
super(Operation.EQQ);
assert result != null: "EQQInstr result is null";
this.arg1 = v1;
this.arg2 = v2;
this.result = result;
}
public Operand[] getOperands() {
return new Operand[]{arg1, arg2};
}
public Variable getResult() {
return result;
}
public void updateResult(Variable v) {
this.result = v;
}
@Override
public void simplifyOperands(Map valueMap, boolean force) {
arg1 = arg1.getSimplifiedOperand(valueMap, force);
arg2 = arg2.getSimplifiedOperand(valueMap, force);
}
@Override
public String toString() {
return super.toString() + "(" + arg1 + ", " + arg2 + ")";
}
@Override
public Instr cloneForInlining(InlinerInfo ii) {
return new EQQInstr(ii.getRenamedVariable(result),
arg1.cloneForInlining(ii), arg2.cloneForInlining(ii));
}
@Override
public Object interpret(ThreadContext context, DynamicScope currDynScope, IRubyObject self, Object[] temp, Block block) {
IRubyObject receiver = (IRubyObject) arg1.retrieve(context, self, currDynScope, temp);
IRubyObject value = (IRubyObject) arg2.retrieve(context, self, currDynScope, temp);
boolean isUndefValue = value == UndefinedValue.UNDEFINED;
if (receiver instanceof RubyArray) {
RubyArray testVals = (RubyArray)receiver;
for (int i = 0, n = testVals.getLength(); i < n; i++) {
IRubyObject v = testVals.eltInternal(i);
IRubyObject eqqVal = isUndefValue ? v : v.callMethod(context, "===", value);
if (eqqVal.isTrue()) return eqqVal;
}
return context.runtime.newBoolean(false);
} else {
return isUndefValue ? receiver : receiver.callMethod(context, "===", value);
}
}
@Override
public void visit(IRVisitor visitor) {
visitor.EQQInstr(this);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy