Annotate AnnotationItems

This commit is contained in:
Ben Gruver 2013-02-27 20:58:09 -08:00
parent 6fb7190811
commit d5f47bae49
6 changed files with 308 additions and 1 deletions

View File

@ -0,0 +1,75 @@
/*
* Copyright 2013, Google Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.jf.dexlib2.dexbacked.raw;
import org.jf.dexlib2.dexbacked.DexBackedDexFile;
import org.jf.dexlib2.dexbacked.DexReader;
import org.jf.dexlib2.util.AnnotatedBytes;
import javax.annotation.Nonnull;
public class AnnotationItem {
public static final int VISIBILITY_OFFSET = 0;
public static final int ANNOTATION_OFFSET = 1;
@Nonnull
public static SectionAnnotator getAnnotator() {
return new SectionAnnotator() {
@Nonnull @Override public String getItemName() {
return "annotation_item";
}
@Override
protected void annotateItem(@Nonnull AnnotatedBytes out, @Nonnull DexBackedDexFile dexFile, int itemIndex) {
int visibility = dexFile.readUbyte(out.getCursor());
out.annotate(1, "visibility = %d: %s", visibility, getAnnotationVisibility(visibility));
DexReader reader = dexFile.readerAt(out.getCursor());
EncodedValue.annotateEncodedAnnotation(out, reader);
}
};
}
private static String getAnnotationVisibility(int visibility) {
switch (visibility) {
case 0:
return "build";
case 1:
return "runtime";
case 2:
return "system";
default:
return "invalid visibility";
}
}
}

View File

@ -0,0 +1,171 @@
/*
* Copyright 2013, Google Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.jf.dexlib2.dexbacked.raw;
import org.jf.dexlib2.dexbacked.DexReader;
import org.jf.dexlib2.util.AnnotatedBytes;
import org.jf.util.ExceptionWithContext;
import javax.annotation.Nonnull;
public class EncodedValue {
public static void annotateEncodedValue(@Nonnull AnnotatedBytes out, @Nonnull DexReader reader) {
int valueArgType = reader.readUbyte();
int valueArg = valueArgType >>> 5;
int valueType = valueArgType & 0x1f;
switch (valueType) {
case 0x00:
out.annotate(1, "valueArg = %d, valueType = 0x%x: byte", valueArg, valueType);
int intValue = reader.readByte();
out.annotate(1, "value = 0x%x", intValue);
break;
case 0x02:
out.annotate(1, "valueArg = %d, valueType = 0x%x: short", valueArg, valueType);
intValue = reader.readSizedInt(valueArg+1);
out.annotate(valueArg + 1, "value = 0x%x", intValue);
break;
case 0x03:
out.annotate(1, "valueArg = %d, valueType = 0x%x: char", valueArg, valueType);
intValue = reader.readSizedSmallUint(valueArg+1);
out.annotate(valueArg+1, "value = 0x%x", intValue);
break;
case 0x04:
out.annotate(1, "valueArg = %d, valueType = 0x%x: int", valueArg, valueType);
intValue = reader.readSizedInt(valueArg+1);
out.annotate(valueArg+1, "value = 0x%x", intValue);
break;
case 0x06:
out.annotate(1, "valueArg = %d, valueType = 0x%x: long", valueArg, valueType);
long longValue = reader.readSizedLong(valueArg+1);
out.annotate(valueArg+1, "value = 0x%x", longValue);
break;
case 0x10:
out.annotate(1, "valueArg = %d, valueType = 0x%x: float", valueArg, valueType);
float floatValue = Float.intBitsToFloat(reader.readSizedRightExtendedInt(valueArg + 1));
out.annotate(valueArg+1, "value = %f", floatValue);
break;
case 0x11:
out.annotate(1, "valueArg = %d, valueType = 0x%x: double", valueArg, valueType);
double doubleValue = Double.longBitsToDouble(reader.readSizedRightExtendedLong(valueArg + 1));
out.annotate(valueArg+1, "value = %f", doubleValue);
break;
case 0x17:
out.annotate(1, "valueArg = %d, valueType = 0x%x: string", valueArg, valueType);
int stringIndex = reader.readSizedSmallUint(valueArg + 1);
out.annotate(valueArg+1, "value = %s",
StringIdItem.getReferenceAnnotation(reader.dexBuf, stringIndex, true));
break;
case 0x18:
out.annotate(1, "valueArg = %d, valueType = 0x%x: type", valueArg, valueType);
int typeIndex = reader.readSizedSmallUint(valueArg+1);
out.annotate(valueArg+1, "value = %s", TypeIdItem.getReferenceAnnotation(reader.dexBuf, typeIndex));
break;
case 0x19:
out.annotate(1, "valueArg = %d, valueType = 0x%x: field", valueArg, valueType);
int fieldIndex = reader.readSizedSmallUint(valueArg+1);
out.annotate(valueArg+1, "value = %s", FieldIdItem.getReferenceAnnotation(reader.dexBuf, fieldIndex));
break;
case 0x1a:
out.annotate(1, "valueArg = %d, valueType = 0x%x: method", valueArg, valueType);
int methodIndex = reader.readSizedSmallUint(valueArg+1);
out.annotate(valueArg+1, "value = %s", MethodIdItem.getReferenceAnnotation(reader.dexBuf, methodIndex));
break;
case 0x1b:
out.annotate(1, "valueArg = %d, valueType = 0x%x: enum", valueArg, valueType);
fieldIndex = reader.readSizedSmallUint(valueArg+1);
out.annotate(valueArg+1, "value = %s", FieldIdItem.getReferenceAnnotation(reader.dexBuf, fieldIndex));
break;
case 0x1c:
out.annotate(1, "valueArg = %d, valueType = 0x%x: array", valueArg, valueType);
annotateEncodedArray(out, reader);
break;
case 0x1d:
out.annotate(1, "valueArg = %d, valueType = 0x%x: annotation", valueArg, valueType);
annotateEncodedAnnotation(out, reader);
break;
case 0x1e:
out.annotate(1, "valueArg = %d, valueType = 0x%x: null", valueArg, valueType);
break;
case 0x1f:
out.annotate(1, "valueArg = %d, valueType = 0x%x: boolean, value=%s", valueArg, valueType, valueArg==1);
break;
default:
throw new ExceptionWithContext("Invalid encoded value type 0x%x at offset 0x%x", valueType,
out.getCursor());
}
}
public static void annotateEncodedAnnotation(@Nonnull AnnotatedBytes out, @Nonnull DexReader reader) {
assert out.getCursor() == reader.getOffset();
int mark = reader.getOffset();
int typeIndex = reader.readSmallUleb128();
out.annotate(reader.getOffset() - mark, TypeIdItem.getReferenceAnnotation(reader.dexBuf, typeIndex));
mark = reader.getOffset();
int size = reader.readSmallUleb128();
out.annotate(reader.getOffset() - mark, "size: %d", size);
for (int i=0; i<size; i++) {
out.annotate(0, "element[%d]", i);
out.indent();
mark = reader.getOffset();
int nameIndex = reader.readSmallUleb128();
out.annotate(reader.getOffset() - mark, "name = %s",
StringIdItem.getReferenceAnnotation(reader.dexBuf, nameIndex));
annotateEncodedValue(out, reader);
out.deindent();
}
}
public static void annotateEncodedArray(@Nonnull AnnotatedBytes out, @Nonnull DexReader reader) {
assert out.getCursor() == reader.getOffset();
int mark = reader.getOffset();
int size = reader.readSmallUleb128();
out.annotate(reader.getOffset() - mark, "size: %d", size);
for (int i=0; i<size; i++) {
out.annotate(0, "element[%d]", i);
out.indent();
annotateEncodedValue(out, reader);
out.deindent();
}
}
}

View File

@ -63,4 +63,30 @@ public class FieldIdItem {
} }
}; };
} }
@Nonnull
public static String asString(@Nonnull DexBackedDexFile dexFile, int fieldIndex) {
int fieldOffset = dexFile.getFieldIdItemOffset(fieldIndex);
int classIndex = dexFile.readUshort(fieldOffset + CLASS_OFFSET);
String classType = dexFile.getType(classIndex);
int typeIndex = dexFile.readUshort(fieldOffset + TYPE_OFFSET);
String fieldType = dexFile.getType(typeIndex);
int nameIndex = dexFile.readSmallUint(fieldOffset + NAME_OFFSET);
String fieldName = dexFile.getString(nameIndex);
return String.format("%s->%s:%s", classType, fieldName, fieldType);
}
@Nonnull
public static String getReferenceAnnotation(@Nonnull DexBackedDexFile dexFile, int fieldIndex) {
try {
String fieldString = asString(dexFile, fieldIndex);
return String.format("field_id_item[%d]: %s", fieldIndex, fieldString);
} catch (Exception ex) {
ex.printStackTrace(System.err);
}
return String.format("field_id_item[%d]", fieldIndex);
}
} }

View File

@ -63,4 +63,30 @@ public class MethodIdItem {
} }
}; };
} }
@Nonnull
public static String asString(@Nonnull DexBackedDexFile dexFile, int methodIndex) {
int methodOffset = dexFile.getMethodIdItemOffset(methodIndex);
int classIndex = dexFile.readUshort(methodOffset + CLASS_OFFSET);
String classType = dexFile.getType(classIndex);
int protoIndex = dexFile.readUshort(methodOffset + PROTO_OFFSET);
String protoString = ProtoIdItem.asString(dexFile, protoIndex);
int nameIndex = dexFile.readSmallUint(methodOffset + NAME_OFFSET);
String methodName = dexFile.getString(nameIndex);
return String.format("%s->%s%s", classType, methodName, protoString);
}
@Nonnull
public static String getReferenceAnnotation(@Nonnull DexBackedDexFile dexFile, int methodIndex) {
try {
String methodString = asString(dexFile, methodIndex);
return String.format("method_id_item[%d]: %s", methodIndex, methodString);
} catch (Exception ex) {
ex.printStackTrace(System.err);
}
return String.format("method_id_item[%d]", methodIndex);
}
} }

View File

@ -79,7 +79,8 @@ public class RawDexFile {
annotators = ImmutableMap.of( annotators = ImmutableMap.of(
ItemType.TYPE_LIST, TypeListItem.getAnnotator(), ItemType.TYPE_LIST, TypeListItem.getAnnotator(),
ItemType.ANNOTATION_SET_REF_LIST, AnnotationSetRefList.getAnnotator(), ItemType.ANNOTATION_SET_REF_LIST, AnnotationSetRefList.getAnnotator(),
ItemType.MAP_LIST, MapItem.getAnnotator()); ItemType.MAP_LIST, MapItem.getAnnotator(),
ItemType.ANNOTATION_ITEM, AnnotationItem.getAnnotator());
} }
public void dumpTo(Writer out, int width) throws IOException { public void dumpTo(Writer out, int width) throws IOException {

View File

@ -68,8 +68,16 @@ public class StringIdItem {
@Nonnull @Nonnull
public static String getReferenceAnnotation(@Nonnull DexBackedDexFile dexFile, int stringIndex) { public static String getReferenceAnnotation(@Nonnull DexBackedDexFile dexFile, int stringIndex) {
return getReferenceAnnotation(dexFile, stringIndex, false);
}
public static String getReferenceAnnotation(@Nonnull DexBackedDexFile dexFile, int stringIndex, boolean quote) {
try { try {
String string = dexFile.getString(stringIndex); String string = dexFile.getString(stringIndex);
if (quote) {
string = String.format("\"%s\"", StringUtils.escapeString(string));
}
return String.format("string_id_item[%d]: %s", stringIndex, string); return String.format("string_id_item[%d]: %s", stringIndex, string);
} catch (Exception ex) { } catch (Exception ex) {
ex.printStackTrace(System.err); ex.printStackTrace(System.err);