edu.umd.cs.findbugs.ba.Location Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of spotbugs Show documentation
Show all versions of spotbugs Show documentation
SpotBugs: Because it's easy!
/*
* Bytecode Analysis Framework
* Copyright (C) 2003,2004 University of Maryland
*
* 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 edu.umd.cs.findbugs.ba;
import java.util.Objects;
import javax.annotation.Nonnull;
import org.apache.bcel.generic.InstructionHandle;
/**
* A class representing a location in the CFG for a method. Essentially, it
* represents a static instruction, with the important caveat that CFGs
* have inlined JSR subroutines, meaning that a single InstructionHandle in a
* CFG may represent several static locations. To this end, a Location is
* comprised of both an InstructionHandle and the BasicBlock that contains it.
*
*
* Location objects may be compared with each other using the equals() method,
* and may be used as keys in tree and hash maps and sets. Note that
* it is only valid to compare Locations produced from the same CFG.
*
* @author David Hovemeyer
* @see CFG
*/
public class Location implements Comparable {
private final InstructionHandle handle;
private final BasicBlock basicBlock;
private int hash;
/**
* Constructor.
*
* @param handle
* the instruction
* @param basicBlock
* the basic block containing the instruction
*/
public Location(@Nonnull InstructionHandle handle, @Nonnull BasicBlock basicBlock) {
Objects.requireNonNull(handle, "handle cannot be null");
Objects.requireNonNull(basicBlock, "basicBlock cannot be null");
this.handle = handle;
this.basicBlock = basicBlock;
}
public static Location getFirstLocation(@Nonnull BasicBlock basicBlock) {
InstructionHandle location = basicBlock.getFirstInstruction();
if (location == null) {
return null;
}
return new Location(location, basicBlock);
}
public static Location getLastLocation(@Nonnull BasicBlock basicBlock) {
InstructionHandle lastInstruction = basicBlock.getLastInstruction();
/*
* if (lastInstruction == null) lastInstruction =
* basicBlock.getExceptionThrower(); if (lastInstruction == null)
* lastInstruction = basicBlock.getFirstInstruction();
*/
if (lastInstruction == null) {
return null;
}
return new Location(lastInstruction, basicBlock);
}
/**
* Get the instruction handle.
*/
@Nonnull
public InstructionHandle getHandle() {
return handle;
}
/**
* Get the basic block.
*/
@Nonnull
public BasicBlock getBasicBlock() {
return basicBlock;
}
/**
* Return whether or not the Location is positioned at the first instruction
* in the basic block.
*/
public boolean isFirstInstructionInBasicBlock() {
return !basicBlock.isEmpty() && handle == basicBlock.getFirstInstruction();
}
/**
* Return whether or not the Location is positioned at the last instruction
* in the basic block.
*/
public boolean isLastInstructionInBasicBlock() {
return !basicBlock.isEmpty() && handle == basicBlock.getLastInstruction();
}
@Override
public int compareTo(Location other) {
int pos = handle.getPosition() - other.handle.getPosition();
return pos;
}
@Override
public int hashCode() {
if (hash == 0) {
return hash = System.identityHashCode(basicBlock) + handle.getPosition();
}
return hash;
}
@Override
public boolean equals(Object o) {
if (!(o instanceof Location)) {
return false;
}
Location other = (Location) o;
return basicBlock == other.basicBlock && handle == other.handle;
}
@Override
public String toString() {
return handle.toString() + " in basic block " + basicBlock.getLabel();
}
/**
* @return a compact string of the form "bb:xx", where "bb" is the basic
* block number and "xx" is the bytecode offset
*/
public String toCompactString() {
return basicBlock.getLabel() + ":" + handle.getPosition();
}
}