ReceiveClosureInstr.java
package org.jruby.ir.instructions;
import org.jruby.ir.IRFlags;
import org.jruby.ir.IRScope;
import org.jruby.ir.IRVisitor;
import org.jruby.ir.Operation;
import org.jruby.ir.operands.Operand;
import org.jruby.ir.operands.Variable;
import org.jruby.ir.operands.WrappedIRClosure;
import org.jruby.ir.transformations.inlining.CloneInfo;
import org.jruby.ir.transformations.inlining.InlineCloneInfo;
import org.jruby.ir.transformations.inlining.SimpleCloneInfo;
/* Receive the closure argument (either implicit or explicit in Ruby source code) */
public class ReceiveClosureInstr extends Instr implements ResultInstr, FixedArityInstr {
    private Variable result;
    public ReceiveClosureInstr(Variable result) {
        super(Operation.RECV_CLOSURE);
        assert result != null: "ReceiveClosureInstr result is null";
        this.result = result;
    }
    @Override
    public Operand[] getOperands() {
        return EMPTY_OPERANDS;
    }
    @Override
    public Variable getResult() {
        return result;
    }
    @Override
    public void updateResult(Variable v) {
        this.result = v;
    }
    @Override
    public boolean computeScopeFlags(IRScope scope) {
        scope.getFlags().add(IRFlags.RECEIVES_CLOSURE_ARG);
        return true;
    }
    @Override
    public Instr clone(CloneInfo info) {
        if (info instanceof SimpleCloneInfo) return new ReceiveClosureInstr(info.getRenamedVariable(result));
        // SSS FIXME: This code below is for inlining and is untested.
        InlineCloneInfo ii = (InlineCloneInfo) info;
        // SSS FIXME: This is not strictly correct -- we have to wrap the block into an
        // operand type that converts the static code block to a proc which is a closure.
        if (ii.getCallClosure() instanceof WrappedIRClosure) return NopInstr.NOP;
        return new CopyInstr(ii.getRenamedVariable(result), ii.getCallClosure());
    }
    @Override
    public void visit(IRVisitor visitor) {
        visitor.ReceiveClosureInstr(this);
    }
}