/* Soot - a J*va Optimization Framework
* Copyright (C) 2003 Jennifer Lhotak
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
package soot.jimple.toolkits.annotation.nullcheck;
import soot.*;
import soot.tagkit.*;
import soot.toolkits.graph.*;
import java.util.*;
import soot.toolkits.scalar.*;
import soot.jimple.*;
public class NullPointerColorer extends BodyTransformer {
public NullPointerColorer( Singletons.Global g ) {}
public static NullPointerColorer v() { return G.v().soot_jimple_toolkits_annotation_nullcheck_NullPointerColorer(); }
protected void internalTransform (Body b, String phaseName, Map options) {
BranchedRefVarsAnalysis analysis = new BranchedRefVarsAnalysis (
new ExceptionalUnitGraph(b));
Iterator it = b.getUnits().iterator();
while (it.hasNext()) {
Stmt s = (Stmt)it.next();
Iterator usesIt = s.getUseBoxes().iterator();
FlowSet beforeSet = (FlowSet)analysis.getFlowBefore(s);
while (usesIt.hasNext()) {
ValueBox vBox = (ValueBox)usesIt.next();
addColorTags(vBox, beforeSet, s, analysis);
}
Iterator defsIt = s.getDefBoxes().iterator();
FlowSet afterSet = (FlowSet)analysis.getFallFlowAfter(s);
while (defsIt.hasNext()){
ValueBox vBox = (ValueBox)defsIt.next();
addColorTags(vBox, afterSet, s, analysis);
}
}
Iterator keysIt = b.getMethod().getDeclaringClass().getTags().iterator();
boolean keysAdded = false;
while (keysIt.hasNext()){
Object next = keysIt.next();
if (next instanceof KeyTag){
if (((KeyTag)next).analysisType().equals("NullCheckTag")){
keysAdded = true;
}
}
}
if (!keysAdded){
b.getMethod().getDeclaringClass().addTag(new KeyTag(ColorTag.RED, "Nullness: Null", "NullCheckTag"));
b.getMethod().getDeclaringClass().addTag(new KeyTag(ColorTag.GREEN, "Nullness: Not Null", "NullCheckTag"));
b.getMethod().getDeclaringClass().addTag(new KeyTag(ColorTag.BLUE, "Nullness: Nullness Unknown", "NullCheckTag"));
}
}
private void addColorTags(ValueBox vBox, FlowSet set, Stmt s, BranchedRefVarsAnalysis analysis){
Value val = vBox.getValue();
if (val.getType() instanceof RefLikeType) {
//G.v().out.println(val+": "+val.getClass().toString());
int vInfo = analysis.anyRefInfo(val, set);
switch (vInfo) {
case 1 : {
// analysis.kNull
s.addTag(new StringTag(val+": Null", "NullCheckTag"));
vBox.addTag(new ColorTag(ColorTag.RED, "NullCheckTag"));
break;
}
case 2 : {
// analysis.kNonNull
s.addTag(new StringTag(val+": NonNull", "NullCheckTag"));
vBox.addTag(new ColorTag(ColorTag.GREEN, "NullCheckTag"));
break;
}
case 99 : {
// analysis.KTop:
s.addTag(new StringTag(val+": Nullness Unknown", "NullCheckTag"));
vBox.addTag(new ColorTag(ColorTag.BLUE, "NullCheckTag"));
break;
}
case 0 : {
// analysis.kBottom
s.addTag(new StringTag(val+": Nullness Unknown", "NullCheckTag"));
vBox.addTag(new ColorTag(ColorTag.BLUE, "NullCheckTag"));
break;
}
}
}
else {
}
}
}