/*
* Copyright (C) 2009 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 dex.reader;
import java.lang.reflect.Modifier;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import dex.reader.DexClassImpl.MethodAnnotation;
import dex.reader.DexClassImpl.ParameterAnnotation;
import dex.reader.DexFileReader.FieldIdItem;
import dex.reader.DexFileReader.MethodsIdItem;
import dex.reader.DexFileReader.ProtIdItem;
import dex.structure.DexAnnotation;
import dex.structure.DexClass;
import dex.structure.DexMethod;
import dex.structure.DexParameter;
/* package */final class DexMethodImpl implements DexMethod {
private DexBuffer buffer;
private MethodsIdItem methodsIdItem;
private String[] stringPool;
private int[] typeIds;
private ProtIdItem protoIdItem;
private List<DexParameter> parameters;
private final int accessFlags;
private final MethodAnnotation methodAnnotation;
private Set<DexAnnotation> annotations;
private final TypeFormatter formatter = new TypeFormatter();
private final DexClass declaringClass;
private final ParameterAnnotation parameterAnnotation;
private Map<Integer, Integer> parameterIdToIndex;
private final FieldIdItem[] fieldIdItems;
public DexMethodImpl(DexBuffer buffer, DexClass declaringClass,
MethodsIdItem methodsIdItem, ProtIdItem protoIdItem,
int accessFlags, MethodAnnotation methodAnnotation,
ParameterAnnotation parameterAnnotation, String[] stringPool,
int[] typeIds, FieldIdItem[] fieldIdItems) {
this.buffer = buffer;
this.declaringClass = declaringClass;
this.methodsIdItem = methodsIdItem;
this.protoIdItem = protoIdItem;
this.accessFlags = accessFlags;
this.methodAnnotation = methodAnnotation;
this.parameterAnnotation = parameterAnnotation;
this.stringPool = stringPool;
this.typeIds = typeIds;
this.fieldIdItems = fieldIdItems;
parseAnnotations();
parseParameterAnnotations();
}
private void parseParameterAnnotations() {
parameterIdToIndex = new HashMap<Integer, Integer>();
if (parameterAnnotation != null) {
buffer.setPosition(parameterAnnotation.annotationsOff);
int numberOfParameters = buffer.readUInt();
for (int i = 0; i < numberOfParameters; i++) {
parameterIdToIndex.put(i, buffer.readUInt());
}
}
}
private void parseAnnotations() {
annotations = new HashSet<DexAnnotation>();
if (methodAnnotation != null) {
buffer.setPosition(methodAnnotation.annotationsOff);
final int size = buffer.readUInt();
for (int i = 0; i < size; i++) {
annotations.add(new DexAnnotationImpl(buffer.createCopy(),
buffer.readUInt(), typeIds, stringPool, fieldIdItems));
}
}
}
public String getName() {
return stringPool[methodsIdItem.name_idx];
}
public String getReturnType() {
return stringPool[typeIds[protoIdItem.return_type_idx]];
}
public synchronized List<DexParameter> getParameters() {
if (parameters == null) {
parameters = new LinkedList<DexParameter>();
if (protoIdItem.parameter_off != 0) {
buffer.setPosition(protoIdItem.parameter_off);
int size = buffer.readUInt();
int[] paramTypeIdx = new int[size];
for (int i = 0; i < size; i++) {
paramTypeIdx[i] = buffer.readUShort();
}
for (int i = 0; i < paramTypeIdx.length; i++) {
parameters.add(new DexParameterImpl(buffer.createCopy(),
stringPool[typeIds[paramTypeIdx[i]]],
parameterIdToIndex.get(i), typeIds, stringPool,
fieldIdItems));
}
}
}
return parameters;
}
public int getModifiers() {
return accessFlags;
}
public Set<DexAnnotation> getAnnotations() {
return annotations;
}
public DexClass getDeclaringClass() {
return declaringClass;
}
@Override
public String toString() {
StringBuilder builder = new StringBuilder();
builder.append(formatter.formatAnnotations(getAnnotations()));
builder.append(Modifier.toString(getModifiers()));
builder.append(" ");
builder.append(formatter.format(getReturnType()));
builder.append(" ");
builder.append(getName());
builder.append("(");
List<DexParameter> parameters = getParameters();
for (DexParameter dexParameter : parameters) {
builder.append(formatter.formatAnnotations(dexParameter
.getAnnotations()));
builder.append(formatter.format(dexParameter.getTypeName()));
}
builder.append(")");
return builder.toString();
}
}