com.ibm.wala.ssa.SSAPhiInstruction Maven / Gradle / Ivy
/*
* Copyright (c) 2002 - 2006 IBM Corporation.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* IBM Corporation - initial API and implementation
*/
package com.ibm.wala.ssa;
import com.ibm.wala.analysis.stackMachine.AbstractIntStackMachine;
import com.ibm.wala.cfg.ControlFlowGraph;
import java.util.Arrays;
import java.util.Iterator;
/**
* A phi instruction in SSA form.
*
* See any modern compiler textbook for the definition of phi and the nature of SSA.
*
*
Note: In SSA {@link IR}s, these instructions do not appear in the normal instruction
* array returned by IR.getInstructions(); instead these instructions live in {@link
* ISSABasicBlock}.
*
*
{@code getUse(i)} corresponds to the value number from the ith predecessor of the
* corresponding {@link ISSABasicBlock} {@code b} in {@link ControlFlowGraph} {@code g}, where
* predecessor order is the order of nodes returned by the {@link Iterator} {@code
* g.getPredNodes(b)}.
*
*
Note: if getUse(i) returns {@link AbstractIntStackMachine}.TOP (that is, -1), then that use
* represents an edge in the CFG which is infeasible in verifiable bytecode.
*/
public class SSAPhiInstruction extends SSAInstruction {
private final int result;
private int[] params;
public SSAPhiInstruction(int iindex, int result, int[] params) throws IllegalArgumentException {
super(iindex);
if (params == null) {
throw new IllegalArgumentException("params is null");
}
this.result = result;
this.params = params;
if (params.length == 0) {
throw new IllegalArgumentException("can't have phi with no params");
}
for (int p : params) {
if (p == 0) {
throw new IllegalArgumentException(
"zero is an invalid value number for a parameter to phi");
}
}
}
@Override
public SSAInstruction copyForSSA(SSAInstructionFactory insts, int[] defs, int[] uses)
throws IllegalArgumentException {
if (defs != null && defs.length == 0) {
throw new IllegalArgumentException();
}
return insts.PhiInstruction(
iIndex(), defs == null ? result : defs[0], uses == null ? params : uses);
}
@Override
public String toString(SymbolTable symbolTable) {
StringBuilder s = new StringBuilder();
s.append(getValueString(symbolTable, result)).append(" = phi ");
s.append(' ').append(getValueString(symbolTable, params[0]));
for (int i = 1; i < params.length; i++) {
s.append(',').append(getValueString(symbolTable, params[i]));
}
return s.toString();
}
@Override
public void visit(IVisitor v) {
if (v == null) {
throw new IllegalArgumentException("v is null");
}
v.visitPhi(this);
}
/**
* @see com.ibm.wala.ssa.SSAInstruction#getDef()
*/
@Override
public boolean hasDef() {
return true;
}
@Override
public int getDef() {
return result;
}
@Override
public int getDef(int i) {
if (i != 0) {
throw new IllegalArgumentException("invalid i: " + i);
}
return result;
}
@Override
public int getNumberOfUses() {
return params.length;
}
@Override
public int getNumberOfDefs() {
return 1;
}
@Override
public int getUse(int j) throws IllegalArgumentException {
if (j >= params.length || j < 0) {
throw new IllegalArgumentException("Bad use " + j);
}
return params[j];
}
/** Clients should not call this. only for SSA builders. I hate this. Nuke it someday. */
public void setValues(int[] i) {
if (i == null || i.length < 1) {
throw new IllegalArgumentException("illegal i: " + Arrays.toString(i));
}
this.params = i;
}
@Override
protected String getValueString(SymbolTable symbolTable, int valueNumber) {
if (valueNumber == AbstractIntStackMachine.TOP) {
return "TOP";
} else {
return super.getValueString(symbolTable, valueNumber);
}
}
@Override
public int hashCode() {
return 7823 * result;
}
@Override
public boolean isFallThrough() {
return true;
}
}