162 lines
5.7 KiB
C
162 lines
5.7 KiB
C
/*
|
|
* Copyright (C) 2008 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.
|
|
*/
|
|
|
|
/*
|
|
* Functions to deal with class definition structures in DEX files
|
|
*/
|
|
|
|
#ifndef LIBDEX_DEXCLASS_H_
|
|
#define LIBDEX_DEXCLASS_H_
|
|
|
|
#include "DexFile.h"
|
|
#include "Leb128.h"
|
|
|
|
/* expanded form of a class_data_item header */
|
|
struct DexClassDataHeader {
|
|
u4 staticFieldsSize;
|
|
u4 instanceFieldsSize;
|
|
u4 directMethodsSize;
|
|
u4 virtualMethodsSize;
|
|
};
|
|
|
|
/* expanded form of encoded_field */
|
|
struct DexField {
|
|
u4 fieldIdx; /* index to a field_id_item */
|
|
u4 accessFlags;
|
|
};
|
|
|
|
/* expanded form of encoded_method */
|
|
struct DexMethod {
|
|
u4 methodIdx; /* index to a method_id_item */
|
|
u4 accessFlags;
|
|
u4 codeOff; /* file offset to a code_item */
|
|
};
|
|
|
|
/* expanded form of class_data_item. Note: If a particular item is
|
|
* absent (e.g., no static fields), then the corresponding pointer
|
|
* is set to NULL. */
|
|
struct DexClassData {
|
|
DexClassDataHeader header;
|
|
DexField* staticFields;
|
|
DexField* instanceFields;
|
|
DexMethod* directMethods;
|
|
DexMethod* virtualMethods;
|
|
};
|
|
|
|
/* Read and verify the header of a class_data_item. This updates the
|
|
* given data pointer to point past the end of the read data and
|
|
* returns an "okay" flag (that is, false == failure). */
|
|
bool dexReadAndVerifyClassDataHeader(const u1** pData, const u1* pLimit,
|
|
DexClassDataHeader *pHeader);
|
|
|
|
/* Read and verify an encoded_field. This updates the
|
|
* given data pointer to point past the end of the read data and
|
|
* returns an "okay" flag (that is, false == failure).
|
|
*
|
|
* The lastIndex value should be set to 0 before the first field in
|
|
* a list is read. It is updated as fields are read and used in the
|
|
* decode process.
|
|
*
|
|
* The verification done by this function is of the raw data format
|
|
* only; it does not verify that access flags or indices
|
|
* are valid. */
|
|
bool dexReadAndVerifyClassDataField(const u1** pData, const u1* pLimit,
|
|
DexField* pField, u4* lastIndex);
|
|
|
|
/* Read and verify an encoded_method. This updates the
|
|
* given data pointer to point past the end of the read data and
|
|
* returns an "okay" flag (that is, false == failure).
|
|
*
|
|
* The lastIndex value should be set to 0 before the first method in
|
|
* a list is read. It is updated as fields are read and used in the
|
|
* decode process.
|
|
*
|
|
* The verification done by this function is of the raw data format
|
|
* only; it does not verify that access flags, indices, or offsets
|
|
* are valid. */
|
|
bool dexReadAndVerifyClassDataMethod(const u1** pData, const u1* pLimit,
|
|
DexMethod* pMethod, u4* lastIndex);
|
|
|
|
/* Read, verify, and return an entire class_data_item. This updates
|
|
* the given data pointer to point past the end of the read data. This
|
|
* function allocates a single chunk of memory for the result, which
|
|
* must subsequently be free()d. This function returns NULL if there
|
|
* was trouble parsing the data. If this function is passed NULL, it
|
|
* returns an initialized empty DexClassData structure.
|
|
*
|
|
* The verification done by this function is of the raw data format
|
|
* only; it does not verify that access flags, indices, or offsets
|
|
* are valid. */
|
|
DexClassData* dexReadAndVerifyClassData(const u1** pData, const u1* pLimit);
|
|
|
|
/*
|
|
* Get the DexCode for a DexMethod. Returns NULL if the class is native
|
|
* or abstract.
|
|
*/
|
|
DEX_INLINE const DexCode* dexGetCode(const DexFile* pDexFile,
|
|
const DexMethod* pDexMethod)
|
|
{
|
|
if (pDexMethod->codeOff == 0)
|
|
return NULL;
|
|
return (const DexCode*) (pDexFile->baseAddr + pDexMethod->codeOff);
|
|
}
|
|
|
|
|
|
/* Read the header of a class_data_item without verification. This
|
|
* updates the given data pointer to point past the end of the read
|
|
* data. */
|
|
DEX_INLINE void dexReadClassDataHeader(const u1** pData,
|
|
DexClassDataHeader *pHeader) {
|
|
pHeader->staticFieldsSize = readUnsignedLeb128(pData);
|
|
pHeader->instanceFieldsSize = readUnsignedLeb128(pData);
|
|
pHeader->directMethodsSize = readUnsignedLeb128(pData);
|
|
pHeader->virtualMethodsSize = readUnsignedLeb128(pData);
|
|
}
|
|
|
|
/* Read an encoded_field without verification. This updates the
|
|
* given data pointer to point past the end of the read data.
|
|
*
|
|
* The lastIndex value should be set to 0 before the first field in
|
|
* a list is read. It is updated as fields are read and used in the
|
|
* decode process.
|
|
*/
|
|
DEX_INLINE void dexReadClassDataField(const u1** pData, DexField* pField,
|
|
u4* lastIndex) {
|
|
u4 index = *lastIndex + readUnsignedLeb128(pData);
|
|
|
|
pField->accessFlags = readUnsignedLeb128(pData);
|
|
pField->fieldIdx = index;
|
|
*lastIndex = index;
|
|
}
|
|
|
|
/* Read an encoded_method without verification. This updates the
|
|
* given data pointer to point past the end of the read data.
|
|
*
|
|
* The lastIndex value should be set to 0 before the first method in
|
|
* a list is read. It is updated as fields are read and used in the
|
|
* decode process.
|
|
*/
|
|
DEX_INLINE void dexReadClassDataMethod(const u1** pData, DexMethod* pMethod,
|
|
u4* lastIndex) {
|
|
u4 index = *lastIndex + readUnsignedLeb128(pData);
|
|
|
|
pMethod->accessFlags = readUnsignedLeb128(pData);
|
|
pMethod->codeOff = readUnsignedLeb128(pData);
|
|
pMethod->methodIdx = index;
|
|
*lastIndex = index;
|
|
}
|
|
|
|
#endif // LIBDEX_DEXCLASS_H_
|