/*
* Copyright (C) 2007 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.dx.rop.code;
import com.android.dx.rop.cst.CstUtf8;
import com.android.dx.util.Hex;
/**
* Information about a source position for code, which includes both a
* line number and original bytecode address.
*/
public final class SourcePosition {
/** non-null; convenient "no information known" instance */
public static final SourcePosition NO_INFO =
new SourcePosition(null, -1, -1);
/** null-ok; name of the file of origin or <code>null</code> if unknown */
private final CstUtf8 sourceFile;
/**
* >= -1; the bytecode address, or <code>-1</code> if that
* information is unknown
*/
private final int address;
/**
* >= -1; the line number, or <code>-1</code> if that
* information is unknown
*/
private final int line;
/**
* Constructs an instance.
*
* @param sourceFile null-ok; name of the file of origin or
* <code>null</code> if unknown
* @param address >= -1; original bytecode address or <code>-1</code>
* if unknown
* @param line >= -1; original line number or <code>-1</code> if
* unknown
*/
public SourcePosition(CstUtf8 sourceFile, int address, int line) {
if (address < -1) {
throw new IllegalArgumentException("address < -1");
}
if (line < -1) {
throw new IllegalArgumentException("line < -1");
}
this.sourceFile = sourceFile;
this.address = address;
this.line = line;
}
/** {@inheritDoc} */
@Override
public String toString() {
StringBuffer sb = new StringBuffer(50);
if (sourceFile != null) {
sb.append(sourceFile.toHuman());
sb.append(":");
}
if (line >= 0) {
sb.append(line);
}
sb.append('@');
if (address < 0) {
sb.append("????");
} else {
sb.append(Hex.u2(address));
}
return sb.toString();
}
/** {@inheritDoc} */
@Override
public boolean equals(Object other) {
if (!(other instanceof SourcePosition)) {
return false;
}
if (this == other) {
return true;
}
SourcePosition pos = (SourcePosition) other;
return (address == pos.address) && sameLineAndFile(pos);
}
/** {@inheritDoc} */
@Override
public int hashCode() {
return sourceFile.hashCode() + address + line;
}
/**
* Returns whether the lines match between this instance and
* the one given.
*
* @param other non-null; the instance to compare to
* @return <code>true</code> iff the lines match
*/
public boolean sameLine(SourcePosition other) {
return (line == other.line);
}
/**
* Returns whether the lines and files match between this instance and
* the one given.
*
* @param other non-null; the instance to compare to
* @return <code>true</code> iff the lines and files match
*/
public boolean sameLineAndFile(SourcePosition other) {
return (line == other.line) &&
((sourceFile == other.sourceFile) ||
((sourceFile != null) && sourceFile.equals(other.sourceFile)));
}
/**
* Gets the source file, if known.
*
* @return null-ok; the source file or <code>null</code> if unknown
*/
public CstUtf8 getSourceFile() {
return sourceFile;
}
/**
* Gets the original bytecode address.
*
* @return >= -1; the address or <code>-1</code> if unknown
*/
public int getAddress() {
return address;
}
/**
* Gets the original line number.
*
* @return >= -1; the original line number or <code>-1</code> if
* unknown
*/
public int getLine() {
return line;
}
}
|