org.jruby.ir.instructions.BSwitchInstr Maven / Gradle / Ivy
package org.jruby.ir.instructions;
import org.jruby.RubyFixnum;
import org.jruby.ir.IRVisitor;
import org.jruby.ir.Operation;
import org.jruby.ir.operands.Label;
import org.jruby.ir.operands.Operand;
import org.jruby.ir.persistence.IRReaderDecoder;
import org.jruby.ir.persistence.IRWriterEncoder;
import org.jruby.ir.transformations.inlining.CloneInfo;
import org.jruby.parser.StaticScope;
import org.jruby.runtime.DynamicScope;
import org.jruby.runtime.ThreadContext;
import org.jruby.runtime.builtin.IRubyObject;
import java.util.Arrays;
/**
* Represents a multiple-target jump instruction based on a switch-like table
*/
public class BSwitchInstr extends MultiBranchInstr {
private final int[] jumps;
private Operand operand;
private final Label rubyCase;
private final Label[] targets;
private final Label elseTarget;
public BSwitchInstr(int[] jumps, Operand operand, Label rubyCase, Label[] targets, Label elseTarget) {
super(Operation.B_SWITCH);
// We depend on the jump table being sorted, so ensure that's the case here
assert jumpsAreSorted(jumps);
this.jumps = jumps;
this.operand = operand;
this.rubyCase = rubyCase;
this.targets = targets;
this.elseTarget = elseTarget;
}
@Override
public Operand[] getOperands() {
return new Operand[] {operand};
}
public Operand getCaseOperand() {
return operand;
}
public int[] getJumps() {
return jumps;
}
public Label getRubyCaseLabel() {
return rubyCase;
}
public Label[] getTargets() {
return targets;
}
public Label getElseTarget() {
return elseTarget;
}
@Override
public void setOperand(int i, Operand operand) {
assert i == 0;
this.operand = operand;
}
@Override
public Instr clone(CloneInfo info) {
Operand operand = this.operand.cloneForInlining(info);
Label rubyCase = info.getRenamedLabel(this.rubyCase);
Label[] targets = new Label[this.targets.length];
for (int i = 0; i < targets.length; i++) targets[i] = info.getRenamedLabel(this.targets[i]);
Label elseTarget = info.getRenamedLabel(this.elseTarget);
return new BSwitchInstr(jumps, operand, rubyCase, targets, elseTarget);
}
@Override
public void encode(IRWriterEncoder e) {
super.encode(e);
e.encode(jumps);
e.encode(operand);
e.encode(rubyCase);
e.encode(targets); // FXXXX
e.encode(elseTarget);
}
public static BSwitchInstr decode(IRReaderDecoder d) {
return new BSwitchInstr(d.decodeIntArray(), d.decodeOperand(), d.decodeLabel(), d.decodeLabelArray(), d.decodeLabel());
}
@Override
public int interpretAndGetNewIPC(ThreadContext context, DynamicScope currDynScope, StaticScope currScope, IRubyObject self, Object[] temp, int ipc) {
Object result = operand.retrieve(context, self, currScope, currDynScope, temp);
if (!(result instanceof RubyFixnum)) {
// not a fixnum, fall back on old case logic
return rubyCase.getTargetPC();
}
int value = ((RubyFixnum) result).getIntValue();
int index = Arrays.binarySearch(jumps, value);
if (index < 0) {
return elseTarget.getTargetPC();
}
return targets[index].getTargetPC();
}
@Override
public void visit(IRVisitor visitor) {
visitor.BSwitchInstr(this);
}
@Override
public Label[] getJumpTargets() {
Label[] jumpTargets = new Label[targets.length + 2];
jumpTargets[0] = rubyCase;
System.arraycopy(targets, 0, jumpTargets, 1, targets.length);
jumpTargets[jumpTargets.length - 1] = elseTarget;
return jumpTargets;
}
private static boolean jumpsAreSorted(final int[] jumps) {
if ( jumps.length == 0 ) return true;
int prev = jumps[0];
for ( int i = 1; i < jumps.length; i++ ) {
int curr = jumps[i];
if ( prev > curr ) return false; // not sorted
prev = curr;
}
return true;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy