| /* |
| * 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(); |
| } |
| } |