2019-02-08 手动安装包解析resources.arsc文件

由于工作原因,需要直接扫描apk文件得到文件名,由于应用已经安装,不想通过写java方式获取,所以直接笨一点的直接解析arsc文件:

http://androidxref.com/7.1.2_r36/xref/frameworks/base/include/androidfw/ResourceTypes.h
按照上面资料一步一步弄下来,只解析了整个文件,尚未按照需求来整理,由于英文水平过差,踩了不少坑,🤦‍♀️🤦‍♀️

附上手戳代码,勿喷(c++刚学)。

#include <iostream>
#include <string.h>

#define RESOURCES_ARSC "/Users/xxx/Downloads/Apps/release/SocialMonitor_v1.0.2.20190207_release/resources.arsc"
#define GLOBAL_DBUG false
#define TYPE_DEBUG true
#define KEY_DEBUG false
#define DBUG true

using namespace std;

int keyStringPoolThunkOffset;
int typeStringPoolThunkOffset;

int baseOffset = 0;
int specOffset = 0;

string* sGlobalString;
string* sKeyString;
string* sTypeString;

int preBoolIdStart = 0;//是否已经ID存在了

struct Header {
    uint16_t type;
    uint16_t headerSize;
    uint32_t thunkSize;
    
    string toString() {
        char header[100];
        sprintf(header, "(type:0x%04x, headerSize:%d, thunkSize:%d)", type, headerSize, thunkSize);
        return header;
    }
};

struct ResTableHeader {
    Header header;
    uint32_t packageNum;
    
    void toString(){
        printf("ResTableHeader { header:%s, packageNum:%d }\n\n",
               header.toString().c_str(), packageNum);
    }
};

ResTableHeader mResTableHeader;

void parseResTableHeader(FILE* mFile) {
    printf("ResTableHeader >> offset 0x%08x\n", baseOffset);
    fseek(mFile, baseOffset, SEEK_SET);
    
    char* temp = new(std::nothrow) char[sizeof(mResTableHeader.header.type)];
    fread(temp, sizeof(mResTableHeader.header.type), 1, mFile);
    mResTableHeader.header.type = *((short*)temp);
    
    fread(temp, sizeof(mResTableHeader.header.headerSize), 1, mFile);
    mResTableHeader.header.headerSize = *((short*)temp);
    delete[] temp;
    
    temp = new(std::nothrow) char[sizeof(mResTableHeader.header.thunkSize)];
    fread(temp, sizeof(mResTableHeader.header.thunkSize), 1, mFile);
    mResTableHeader.header.thunkSize = *((int*)temp);

    fread(temp, sizeof(mResTableHeader.packageNum), 1, mFile);
    mResTableHeader.packageNum = *((int*)temp);
    delete[] temp;
    
    if (DBUG) mResTableHeader.toString();
    baseOffset += mResTableHeader.header.headerSize;
}

struct GlobalStringPool {
    Header header;
    uint32_t stringNum;
    uint32_t styleNum;
    uint32_t flag;
    uint32_t strOffset;
    uint32_t styleOffset;
    
    bool chars;
    // ...
    void toString() {
        chars = (flag == 1<<8);
        printf("GlobalStringPool { header:%s, stringNum:%d, styleNum:%d, "
               "flag:%d(%s), strOffset:%d, styleOffset:%d }\n", header.toString().c_str(), stringNum,
               styleNum, flag, chars ? "utf-8" : "utf-16", strOffset, styleOffset);
    }
};

GlobalStringPool mGlobalStringPool;

void printfGlobalStringPool(FILE* mFile) {
    int fileOffset = baseOffset + mGlobalStringPool.strOffset;
    printf("\tGlobalStringPool Content >> offset :0x%08x\n", fileOffset);
    
    fseek(mFile, fileOffset, SEEK_SET);
    
    uint8_t c_len[2];
    sGlobalString = new string[mGlobalStringPool.stringNum];
    
    for (int index = 0; index < mGlobalStringPool.stringNum; index++) {
        fread(&c_len, 2, 1, mFile);
        if (mGlobalStringPool.chars) { // add 0x00 结尾
            uint8_t* mStr = new(nothrow) uint8_t[c_len[1] + 1];
            fread(mStr, c_len[1] + 1, 1, mFile);
            *(sGlobalString + index) = (const char*)mStr;
            delete [] mStr;
            if (GLOBAL_DBUG) printf("\t(index:%d, length:%d)\t\t %s\n", (index+1), c_len[1],
                             (*(sGlobalString + index)).c_str());
        } else {
            uint16_t* mStr = new(nothrow) uint16_t[c_len[1] + 1];
            fread(mStr, c_len[1] + 1, 1, mFile);
            *(sGlobalString + index) = (const char*)mStr;
            delete [] mStr;
            if (GLOBAL_DBUG) printf("\t(index:%d, length:%d)\t\t %s\n", (index+1), c_len[1],
                             (*(sGlobalString + index)).c_str());
        }
    }
    printf("\n");
}

void parseGlobalStringPool(FILE* mFile) {
    printf("GlobalStringPool >> offset 0x%08x\n", baseOffset);
    int offset = baseOffset;
    fseek(mFile, offset, SEEK_SET);
    
    char* temp = new(std::nothrow) char[sizeof(mGlobalStringPool.header.type)];
    fread(temp, sizeof(mGlobalStringPool.header.type), 1, mFile);
    mGlobalStringPool.header.type = *((short*) temp);
    delete [] temp;
    
    fread(temp, sizeof(mGlobalStringPool.header.headerSize), 1, mFile);
    mGlobalStringPool.header.headerSize = *((short*) temp);
    
    temp = new(std::nothrow) char[sizeof(mGlobalStringPool.header.thunkSize)];
    fread(temp, sizeof(mGlobalStringPool.header.thunkSize), 1, mFile);
    mGlobalStringPool.header.thunkSize = *((int*) temp);
    
    fread(temp, sizeof(mGlobalStringPool.stringNum), 1, mFile);
    mGlobalStringPool.stringNum = *((int*) temp);
    
    fread(temp, sizeof(mGlobalStringPool.styleNum), 1, mFile);
    mGlobalStringPool.styleNum = *((int*) temp);
    
    fread(temp, sizeof(mGlobalStringPool.flag), 1, mFile);
    mGlobalStringPool.flag = *((int*) temp);
    
    fread(temp, sizeof(mGlobalStringPool.strOffset), 1, mFile);
    mGlobalStringPool.strOffset = *((int*) temp);
    
    fread(temp, sizeof(mGlobalStringPool.styleOffset), 1, mFile);
    mGlobalStringPool.styleOffset = *((int*) temp);
    
    mGlobalStringPool.toString();
    
    //打印全局字符串池
    printfGlobalStringPool(mFile);
    
    baseOffset += mGlobalStringPool.header.thunkSize;
}

struct ResTable_package{
    Header header;
    
    uint32_t id;
    uint16_t name[128];  //256 + 32 = 288
    uint32_t typeStrings;
    uint32_t lastPublicType;
    uint32_t keyStrings;
    uint32_t lastPublicKey;
    uint32_t typeIdOffset;
    
    void toString() {
        printf("ResTable_package { header:%s, id:0x%02X, typeStrings:%d,"
               " lastPublicType=%d, keyStrings:%d, lastPublicKey:%d, typeIdOffset:%d }\n",
               header.toString().c_str(), id, typeStrings, lastPublicType, keyStrings, lastPublicKey,
               typeIdOffset);
        showName();
    }
    
    void showName(){
        printf("\tPackageName { name:");
        for(int i = 0; i < 128; i++) {
            unsigned short* cs = name+i;
            printf("%s", (unsigned char*)cs);
        }
        printf(" }\n");
    }
};

ResTable_package mResTablePackage;

void parseResTablePackage(FILE* mFile) {
    
    printf("ResTable_Package >> offset 0x%08x\n", baseOffset);
    fseek(mFile, baseOffset, SEEK_SET);
    
    char* tmp = new(std::nothrow) char[sizeof(mResTablePackage.header.type)];
    fread(tmp, sizeof(mResTablePackage.header.type), 1, mFile);
    mResTablePackage.header.type = *((uint16_t*)tmp);
    
    tmp = new(std::nothrow) char[sizeof(mResTablePackage.header.headerSize)];
    fread(tmp, sizeof(mResTablePackage.header.headerSize), 1, mFile);
    mResTablePackage.header.headerSize = *((uint16_t*)tmp);
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTablePackage.header.thunkSize)];
    fread(tmp, sizeof(mResTablePackage.header.thunkSize), 1, mFile);
    mResTablePackage.header.thunkSize = *((uint32_t*)tmp);
    
    fread(tmp, sizeof(mResTablePackage.id), 1, mFile);
    mResTablePackage.id = *((uint32_t*)tmp);
    
    fread(mResTablePackage.name, sizeof(uint16_t) * 128, 1, mFile);
    
    fread(tmp, sizeof(mResTablePackage.typeStrings), 1, mFile);
    mResTablePackage.typeStrings = *((uint32_t*)tmp);

    fread(tmp, sizeof(mResTablePackage.lastPublicType), 1, mFile);
    mResTablePackage.lastPublicType = *((uint32_t*)tmp);
    
    fread(tmp, sizeof(mResTablePackage.keyStrings), 1, mFile);
    mResTablePackage.keyStrings = *((uint32_t*)tmp);
    
    fread(tmp, sizeof(mResTablePackage.lastPublicKey), 1, mFile);
    mResTablePackage.lastPublicKey = *((uint32_t*)tmp);
    
    fread(tmp, sizeof(mResTablePackage.typeIdOffset), 1, mFile);
    mResTablePackage.typeIdOffset = *((uint32_t*)tmp);
    delete [] tmp;
    
    mResTablePackage.toString();
    
    typeStringPoolThunkOffset = baseOffset + mResTablePackage.typeStrings;
    printf("\ttypeStringPoolThunkOffset>>offset 0x%08x\n", typeStringPoolThunkOffset);
    keyStringPoolThunkOffset = baseOffset + mResTablePackage.keyStrings;
    printf("\tkeyStringPoolThunkOffset>>offset 0x%08x\n", keyStringPoolThunkOffset);
    printf("\n");
    
    baseOffset += mResTablePackage.header.headerSize;
}

struct ResTypeStringPool {
    Header header;
    
    uint32_t stringCount;
    uint32_t styleCount;
    uint32_t flags;
    uint32_t stringsStart;
    uint32_t stylesStart;
    
    void toString(){
        printf("ResTypeStringPool { header:%s, stringCount:%d,"
               " styleCount:%d, flags:%d, stringsStart:%d, stylesStart:%d }\n", header.toString().c_str(), stringCount, styleCount, flags, stringsStart, stylesStart);
    }
};

ResTypeStringPool mResTypeStringPool;

void printResTypeStringPool(FILE* mF) {
    int offset = typeStringPoolThunkOffset + mResTypeStringPool.stringsStart;
    printf("\tResTypeStringPool content >> offset 0x%08x.\n", offset);

    fseek(mF, offset, SEEK_SET);
    sTypeString = new string[mResTypeStringPool.stringCount];
    
    uint8_t c_len[2];
    for (int index = 0; index < mResTypeStringPool.stringCount; index++) {
        fread(&c_len, 2, 1, mF);
        //4个字符 需要读 (4+0x0000) * 2
        uint16_t* mStr = new(std::nothrow) uint16_t[c_len[0] + 1];
        fread(mStr, c_len[0]+1, sizeof(uint16_t), mF);
        
        uint8_t* cStr = new(std::nothrow) uint8_t[c_len[0] + 2];
        for (int i = 0; i < c_len[0] + 2; i++) {
            *(cStr+i) = *(uint8_t*)(mStr+i);
        }
        delete [] mStr;
        *(sTypeString + index) = (const char*)cStr;
        if (TYPE_DEBUG) printf("\t\tindex:%d %s\n", index + 1, (*(sTypeString + index)).c_str());
    }
    printf("\n");
}

void parseResTypeStringPool(FILE* mF) {
    printf("ResTypeStringPoolHeader >> offset 0x%08x.\n", typeStringPoolThunkOffset);
    fseek(mF, typeStringPoolThunkOffset, SEEK_SET);
    
    char* tmp = new(std::nothrow) char[sizeof(mResTypeStringPool.header.type)];
    fread(tmp, sizeof(mResTypeStringPool.header.type), 1, mF);
    mResTypeStringPool.header.type = *(uint16_t*) tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTypeStringPool.header.headerSize)];
    fread(tmp, sizeof(mResTypeStringPool.header.headerSize), 1, mF);
    mResTypeStringPool.header.headerSize = *(uint16_t*) tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTypeStringPool.header.thunkSize)];
    fread(tmp, sizeof(mResTypeStringPool.header.thunkSize), 1, mF);
    mResTypeStringPool.header.thunkSize = *(uint32_t*) tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTypeStringPool.stringCount)];
    fread(tmp, sizeof(mResTypeStringPool.stringCount), 1, mF);
    mResTypeStringPool.stringCount = *(uint32_t*) tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTypeStringPool.styleCount)];
    fread(tmp, sizeof(mResTypeStringPool.styleCount), 1, mF);
    mResTypeStringPool.styleCount = *(uint32_t*) tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTypeStringPool.flags)];
    fread(tmp, sizeof(mResTypeStringPool.flags), 1, mF);
    mResTypeStringPool.flags = *(uint32_t*) tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTypeStringPool.stringsStart)];
    fread(tmp, sizeof(mResTypeStringPool.stringsStart), 1, mF);
    mResTypeStringPool.stringsStart = *(uint32_t*) tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTypeStringPool.stylesStart)];
    fread(tmp, sizeof(mResTypeStringPool.stylesStart), 1, mF);
    mResTypeStringPool.stylesStart = *(uint32_t*) tmp;
    delete [] tmp;
    
    mResTypeStringPool.toString();
    printResTypeStringPool(mF);
    
    baseOffset += mResTypeStringPool.header.thunkSize;
}

struct ResKeyStringPool {
    Header header;
    
    uint32_t stringCount;
    uint32_t styleCount;
    uint32_t flags;
    uint32_t stringsStart;
    uint32_t stylesStart;
    
    void toString(){
        printf("ResKeyStringPool { header:%s, stringCount:%d,"
               " styleCount:%d, flags:%d, stringsStart:%d, stylesStart:%d }\n", header.toString().c_str(), stringCount, styleCount, flags, stringsStart, stylesStart);
    }
};

ResKeyStringPool mResKeyStringPool;

void printResKeyStringPool(FILE* mF) {
    int offset = keyStringPoolThunkOffset + mResKeyStringPool.stringsStart;
    printf("\tResKeyStringPool content >> offset 0x%08x.\n", offset);
    
    bool chars = (mResKeyStringPool.flags == (1<<8));
    fseek(mF, offset, SEEK_SET);
    
    uint8_t c_len[2];
    sKeyString = new(nothrow) string[mResKeyStringPool.stringCount];
    
    for (int index = 0; index < mResKeyStringPool.stringCount; index++) {
        fread(&c_len, 2, 1, mF);
        if (chars) {
            uint8_t* mStr = new(std::nothrow) uint8_t[c_len[0] + 1];
            fread(mStr, c_len[0] + 1, 1, mF);
            *(sKeyString + index) = (const char*)mStr;
            delete[] mStr;
            
            if (KEY_DEBUG) {
                printf("\t(index:%d, length:%d)\t\t %s\n", index+1, c_len[0], (*(sKeyString + index)).c_str());
            }
        } else {
            // nothing
        }
    }
    printf("\n");
}

void parseResKeyStringPool(FILE* mF) {
    
    printf("Start parse mResKeyStringPool >> offset : 0x%08x.\n", keyStringPoolThunkOffset);
    fseek(mF, keyStringPoolThunkOffset, SEEK_SET);
    
    char* tmp = new(std::nothrow) char[sizeof(mResKeyStringPool.header.type)];
    fread(tmp, sizeof(mResKeyStringPool.header.type), 1, mF);
    mResKeyStringPool.header.type = *(uint16_t*) tmp;
    
    fread(tmp, sizeof(mResKeyStringPool.header.headerSize), 1, mF);
    mResKeyStringPool.header.headerSize = *(uint16_t*) tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResKeyStringPool.header.thunkSize)];
    fread(tmp, sizeof(mResKeyStringPool.header.thunkSize), 1, mF);
    mResKeyStringPool.header.thunkSize = *(uint32_t*) tmp;
    
    fread(tmp, sizeof(mResKeyStringPool.stringCount), 1, mF);
    mResKeyStringPool.stringCount = *(uint32_t*) tmp;
    
    fread(tmp, sizeof(mResKeyStringPool.styleCount), 1, mF);
    mResKeyStringPool.styleCount = *(uint32_t*) tmp;
    
    fread(tmp, sizeof(mResKeyStringPool.flags), 1, mF);
    mResKeyStringPool.flags = *(uint32_t*) tmp;
    
    fread(tmp, sizeof(mResKeyStringPool.stringsStart), 1, mF);
    mResKeyStringPool.stringsStart = *(uint32_t*) tmp;
    
    fread(tmp, sizeof(mResKeyStringPool.stylesStart), 1, mF);
    mResKeyStringPool.stylesStart = *(uint32_t*) tmp;
    delete [] tmp;
    
    mResKeyStringPool.toString();
    printResKeyStringPool(mF);
    
    baseOffset += mResKeyStringPool.header.thunkSize;
}

// ------

struct ResTable_typeSpec
{
    Header header;
    // The type identifier this chunk is holding.  Type IDs start
    // at 1 (corresponding to the value of the type bits in a
    // resource identifier).  0 is invalid.
    uint8_t id;

    // Must be 0.
    uint8_t res0;
    // Must be 0.
    uint16_t res1;

    // Number of uint32_t entry configuration masks that follow.
    uint32_t entryCount;
    enum {
        // Additional flag indicating an entry is public.
        SPEC_PUBLIC = 0x40000000
    };
    
    void toString(){
        printf("ResTable_typeSpec { header:%s, id:0x%02d, res0:%d, res1:%d, "
               "entryCount:%d }\n", header.toString().c_str(), id, res0, res1, entryCount);
    }
};

ResTable_typeSpec mResTable_typeSpec;

void parseResTable_typeSpec(FILE* mF, int offset) {
    
    printf("ResTable_typeSpec >> offset:0x%02x\n", offset);
    
    fseek(mF, offset, SEEK_SET);
    uint8_t* tmp = new(std::nothrow) uint8_t[sizeof(mResTable_typeSpec.header.type)];
    fread(tmp, sizeof(mResTable_typeSpec.header.type), 1, mF);
    mResTable_typeSpec.header.type = *(uint16_t*)tmp;
    
    fread(tmp, sizeof(mResTable_typeSpec.header.headerSize), 1, mF);
    mResTable_typeSpec.header.headerSize = *(uint16_t*)tmp;
    
    tmp = new(std::nothrow) uint8_t[sizeof(mResTable_typeSpec.header.thunkSize)];
    fread(tmp, sizeof(mResTable_typeSpec.header.thunkSize), 1, mF);
    mResTable_typeSpec.header.thunkSize = *(uint32_t*)tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) uint8_t[sizeof(mResTable_typeSpec.id)];
    fread(tmp, sizeof(mResTable_typeSpec.id), 1, mF);
    mResTable_typeSpec.id = *(uint8_t*)tmp;
    
    printf("mResTable_typeSpec>>typeId:%s.\n", (*(sTypeString + (mResTable_typeSpec.id - 1))).c_str());
    
    fread(tmp, sizeof(mResTable_typeSpec.res0), 1, mF);
    mResTable_typeSpec.res0 = *(uint8_t*)tmp;
    
    tmp = new(std::nothrow) uint8_t[sizeof(mResTable_typeSpec.res1)];
    fread(tmp, sizeof(mResTable_typeSpec.res1), 1, mF);
    mResTable_typeSpec.res1 = *(uint8_t*)tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) uint8_t[sizeof(mResTable_typeSpec.entryCount)];
    fread(tmp, sizeof(mResTable_typeSpec.entryCount), 1, mF);
    mResTable_typeSpec.entryCount = *(uint32_t*)tmp;
    delete [] tmp;
    
    mResTable_typeSpec.toString();
    
    printf("\tmResTable_typeSpec.CONIFIG_* >> offset 0x%08x\n", offset+ mResTable_typeSpec.header.headerSize);
    
    tmp = new(std::nothrow) uint8_t[4];
    printf("\t");
    for (int i = 1; i <= mResTable_typeSpec.entryCount; i++) {
        fread(tmp, 4, 1, mF);
        int specArr = *((uint32_t *)tmp);
        
        printf("%08X ", specArr);
        if (i % 8 == 0 && i != 0) {
            printf("\n\t");
        }
    }
    delete [] tmp;
    printf("\n\n");
    
    baseOffset += mResTable_typeSpec.header.thunkSize;
}

struct ResTable_type{
    Header header;
    
    uint8_t id;
    uint8_t res0;
    uint16_t res1;
    uint32_t entryCount;
    uint32_t entriesStart;
    
    //... 56
    
    void toString() {
        printf("ResTable_type { header:%s, id:0x%02x, res0:%d, res1:%d, entryCount:%d"
               ", entriesStart:%d }\n", header.toString().c_str(), id, res0, res1, entryCount, entriesStart);
    }
};

ResTable_type mResTableType;

char* getResId(int entryId){
    char* str = new char[11];
    sprintf(str, "0x%02x%02x%04x", mResTablePackage.id, mResTable_typeSpec.id, entryId);
    //return (mResTablePackage.id << 24) | ((mResTable_typeSpec.id)<<16) | (entryId & 0xFFFF);
    return str;
}

void parseResTable_type(FILE* mF, int offset) {
    
    printf("ResTable_type offset >> 0x%02x\n", offset);
    fseek(mF, offset, SEEK_SET);
    
    char* tmp = new(std::nothrow) char[sizeof(mResTableType.header.type)];
    fread(tmp, sizeof(mResTableType.header.type), 1, mF);
    mResTableType.header.type = *(uint16_t*)tmp;
    
    fread(tmp, sizeof(mResTableType.header.headerSize), 1, mF);
    mResTableType.header.headerSize = *(uint16_t*)tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTableType.header.thunkSize)];
    fread(tmp, sizeof(mResTableType.header.thunkSize), 1, mF);
    mResTableType.header.thunkSize = *(uint32_t*)tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTableType.id)];
    fread(tmp, sizeof(mResTableType.id), 1, mF);
    mResTableType.id = *(uint8_t*)tmp;
    
    fread(tmp, sizeof(mResTableType.res0), 1, mF);
    mResTableType.res0 = *(uint8_t*)tmp;
    delete[] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTableType.res1)];
    fread(tmp, sizeof(mResTableType.res1), 1, mF);
    mResTableType.res1 = *(uint16_t*)tmp;
    delete [] tmp;
    
    tmp = new(std::nothrow) char[sizeof(mResTableType.entryCount)];
    fread(tmp, sizeof(mResTableType.entryCount), 1, mF);
    mResTableType.entryCount = *(uint32_t*)tmp;
    
    fread(tmp, sizeof(mResTableType.entriesStart), 1, mF);
    mResTableType.entriesStart = *(uint32_t*)tmp;
    delete [] tmp;
    
    // -----------ResTable_Config  56
    tmp = new(std::nothrow) char[mResTableType.header.headerSize - 20];
    fread(tmp, mResTableType.header.headerSize - 20, 1, mF);
    /*
    uint32_t config_size = *((uint32_t*)tmp);
    uint32_t config_imsi = *((uint32_t*)(tmp + 4));
    uint32_t config_locale = *((uint32_t*)(tmp + 8));
    uint32_t config_screenType = *((uint32_t*)(tmp + 12));
    uint32_t config_input = *((uint32_t*)(tmp + 16));
    uint32_t config_screenSize = *((uint32_t*)(tmp + 20));
    uint32_t config_version = *((uint32_t*)(tmp + 24));
    uint32_t config_screenConfig = *((uint32_t*)(tmp + 28));
    uint32_t config_screenSizeDp = *((uint32_t*)(tmp + 32));
    char config_localeScript[4];
    memcmp(config_localeScript, tmp + 36, 4);
    char config_localeVariant[8];
    memcmp(config_localeVariant, tmp + 40, 8);
    uint32_t screenConfig2 = *((uint32_t*)(tmp + 48));
    */
    delete [] tmp;
    printf("Skip config ...");
    
    mResTableType.toString();
    
    printf("\t\tElement>> offset 0x%02x\n", offset + mResTableType.header.headerSize);
    fseek(mF , offset + mResTableType.header.headerSize, SEEK_SET);
    
    uint32_t* items = new(nothrow) uint32_t[mResTableType.entryCount];
    fread(items, mResTableType.entryCount, sizeof(uint32_t), mF);
    
    int hasNum = 0;
    printf("\t\tElement:\n\t");
    for (int i = 1; i <= mResTableType.entryCount; i++) {
        printf("%08X ", *(items + i - 1));
        if (i % 8 == 0 && i !=0) {
            printf("\n\t");
        }
        if (*(items + i - 1) != 0xFFFFFFFF) {
            hasNum ++;
        }
    }
        
    printf("\n\t\tSkip element... >> %d\n", hasNum);
    
    /*printf("Config: { size:%d, imsi:%d, locale:%d, screenType:%d, input:%d, screenSize:%d, version:%d"
           " screenConfig:%d, screenSizeDp:%d, localeScript:%s, localeVariant:%s, screenConfig2:%d }\n",
           config_size, config_imsi, config_locale, config_screenType, config_input, config_screenSize,
           config_version, config_screenConfig, config_screenSizeDp, config_localeScript,
           config_localeVariant, screenConfig2);*/
    
    printf("\t\tStart parse name/value ...\n");
    printf("\tEntry >> offset 0x%02x\n", offset + mResTableType.entriesStart);
    
    int offsetEntry = offset + mResTableType.entriesStart, bodySize = 0;
    fseek(mF, offsetEntry, SEEK_SET);
    
    for (int i = 0; i < mResTableType.entryCount; i++) {
        //offsetEntry += bodySize;
        if (*(items + i) == -1) {
            //printf("\t>>Skip %d->%d\n", i, *(items + i));
            continue;
        }
        
        fseek(mF, offsetEntry + *(items + i), SEEK_SET);
        printf("\tOffsetEntry >> 0x%08x >> ", offsetEntry);
        
        if (mResTableType.id == 0x05) {
            if (preBoolIdStart != 0){
                preBoolIdStart++;
                printf("  resId:0x%08x  ", (mResTablePackage.id << 24 | mResTableType.id << 16 | (preBoolIdStart & 0xFFFF)));
            } else {
                printf("  resId:0x%08x  ", (mResTablePackage.id << 24 | mResTableType.id << 16 | (i & 0xFFFF)));
            }
        } else {
            printf("  resId:0x%08x  ", (mResTablePackage.id << 24 | mResTableType.id << 16 | (i & 0xFFFF)));
        }
        
        uint8_t* entry = new(nothrow) uint8_t[8];
        fread(entry, 8, sizeof(uint8_t), mF);
        
        uint16_t entrySize = *((uint16_t*) entry);
        uint16_t entryFlags = *((uint16_t*) (entry + sizeof(uint16_t)));
        uint32_t entryKey = *((uint32_t*) (entry + sizeof(uint16_t) * 2));
        delete[] entry;
        
        if ((entryFlags & 0x0001) == 1) { // name/vaule
            uint8_t* mapEntry = new(nothrow) uint8_t[sizeof(uint32_t)*2];
            fread(mapEntry, sizeof(uint32_t) * 2, 1, mF);
            
            uint32_t mapEntry_parent = *((uint32_t*)(mapEntry));
            uint32_t mapEntry_count = *((uint32_t*)(mapEntry + sizeof(uint32_t)));
            delete [] mapEntry;
            
            printf("\tResTable_map_entry >> (entrySize:%d, entryKey:%d::%s, mapEntry_parent:%d,"
                   " mapEntry_count:%d)\n",
                   entrySize, entryKey, (*(sKeyString+entryKey)).c_str(),
                   mapEntry_parent, mapEntry_count);
            
            for (int i = 0; i < mapEntry_count; i++ ) {
                uint8_t* resValue = new(nothrow) uint8_t[sizeof(uint32_t)]; // ResTable_map
                fread(resValue, sizeof(uint32_t), 1, mF);
                uint32_t mapName = *((uint16_t*)(resValue));
                delete [] resValue;
                
                resValue = new(nothrow) uint8_t[sizeof(uint16_t)]; //Res_Value
                fread(resValue, sizeof(uint16_t), 1, mF);
                uint16_t valueSize = *((uint16_t*)(resValue));
                delete [] resValue;
                
                resValue = new(nothrow) uint8_t[sizeof(uint8_t)];
                fread(resValue, sizeof(uint8_t), 1, mF);
                uint16_t valueRes0 = *((uint8_t*)(resValue));
                
                resValue = new(nothrow) uint8_t[sizeof(uint8_t)];
                fread(resValue, sizeof(uint8_t), 1, mF);
                uint16_t valueDataType = *((uint8_t*)(resValue));
                delete [] resValue;
                
                resValue = new(nothrow) uint8_t[sizeof(uint32_t)];
                fread(resValue, sizeof(uint32_t), 1, mF);
                uint32_t valueData = *((uint32_t*)(resValue));
                delete [] resValue;
                
                printf("\tValueDataType>>%d mapName>>%d \n", valueDataType, mapName);
                switch (valueDataType) {
                    case 0x00:
                        printf("\t\t__(key::value) >> valueSize:%d, valueRes0:%d,"
                               " valueDataType:null, valueData:%d.\n", valueSize, valueRes0,
                               valueData);
                        break;
                    case 0x01:// ref类型
                        printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                               " valueDataType:ref, %s>>0x%08x>>%s.\n", valueSize, valueRes0,
                               (*(sKeyString+entryKey)).c_str(),
                               valueData, (valueData >> 24 == 0x7f) ? "@":"@android:"); // 因为从0开始算起
                        break;
                    case 0x02:// attr类型
                        printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                               " valueDataType:ref, %s>>0x%08x>>%s.\n", valueSize, valueRes0,
                               (*(sKeyString+entryKey)).c_str(),
                               valueData, (valueData >> 24 == 0x7f) ? "@":"@android:");
                        break;
                    case 0x03:
                        printf("\t\t__(key::value) >> valueSize:%d, valueRes0:%d,"
                               " valueDataType:string, data:%s.\n", valueSize, valueRes0,
                               (*(sGlobalString+valueData+1)).c_str());
                        break;
                    case 0x05:
                        {//dimension
                            string dim[] = {"px", "dip", "sp", "pt", "in", "mm"};
                            printf("\t\t__(key::value) >> valueSize:%d, valueRes0:%d,"
                                   "%s>>%d.\n", valueSize, valueRes0, (*(sKeyString + entryKey)).c_str(), valueData);
                        }
                        break;
                    case 0x10:
                        printf("\t\t__(key::value) >> valueSize:%d, valueRes0:%d,"
                               " %s:%d.\n", valueSize, valueRes0, (*(sKeyString + entryKey)).c_str(), valueData);
                        break;
                    case 0x11:
                        {
                            printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                                   " valueDataType:bool, %s>>0x%08x.\n", valueSize, valueRes0,
                                   (*(sKeyString+entryKey)).c_str(), valueData);
                        }
                        break;
                    case 0x12:
                        {
                            bool res = (valueData != 0);
                            printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                                   " valueDataType:bool, %s>>%s.\n", valueSize, valueRes0,
                                   (*(sKeyString+entryKey)).c_str(), res ? "true":"false");
                        }
                        break;
                    default:
                        break;
                }
            }
            bodySize = entrySize + mapEntry_count * 12;
        } else {
            uint8_t* resValue = new(nothrow) uint8_t[sizeof(uint16_t)];
            fread(resValue, sizeof(uint16_t), 1, mF);
            uint16_t valueSize = *((uint16_t*)(resValue));
            delete [] resValue;
            
            resValue = new(nothrow) uint8_t[sizeof(uint8_t)];
            fread(resValue, sizeof(uint8_t), 1, mF);
            uint16_t valueRes0 = *((uint8_t*)(resValue));
            
            resValue = new(nothrow) uint8_t[sizeof(uint8_t)];
            fread(resValue, sizeof(uint8_t), 1, mF);
            short short_valueDataType = *((uint8_t*)(resValue));
            int valueDataType = short_valueDataType;
            delete [] resValue;
            
            resValue = new(nothrow) uint8_t[sizeof(uint32_t)];
            fread(resValue, sizeof(uint32_t), 1, mF);
            uint32_t valueData = *((uint32_t*)(resValue));
            delete [] resValue;
            
            printf("ValueDataType>>%d\n", valueDataType);
            switch (valueDataType) {
                case 0x01:// 引用类型
                    printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                           " valueDataType:ref, %s>>0x%08x>>%s.\n", valueSize, valueRes0,
                           (*(sKeyString+entryKey)).c_str(),
                           valueData, (valueData >> 24 == 0x7f) ? "@":"@android:"); // 因为从0开始算起
                    break;
                case 0x03:
                    printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                           " valueDataType:string, %s>>%s.\n", valueSize, valueRes0,
                           (*(sKeyString+entryKey)).c_str(),
                           (*(sGlobalString + valueData + 1)).c_str()); // 因为从0开始算起
                    break;
                case 0x04: //float
                    {
                        string dim[] = {"px", "dip", "sp", "pt", "in", "mm"};
                        printf("\t\t__(key::value) >> valueSize:%d, valueRes0:%d,"
                               "%s >> float >> %d.\n", valueSize, valueRes0, (*(sKeyString + entryKey)).c_str(), valueData);
                    }
                    break;
                case 0x05:
                    {//dimension
                        string dim[] = {"px", "dip", "sp", "pt", "in", "mm"};
                        printf("\t\t__(key::value) >> valueSize:%d, valueRes0:%d,"
                               "%s>>%d.\n", valueSize, valueRes0, (*(sKeyString + entryKey)).c_str(), valueData);
                    }
                    break;
                case 0x06:{ // 复数
                        printf("\t\t__(key::value) >> valueSize:%d, valueRes0:%d,"
                               "%s>>复数>>%d.\n", valueSize, valueRes0, (*(sKeyString + entryKey)).c_str(), valueData);
                    }
                    break;
                case 0x10:
                    printf("\t\t__(key::value) >> valueSize:%d, valueRes0:%d,"
                           "%s>>%d.\n", valueSize, valueRes0, (*(sKeyString + entryKey)).c_str(), valueData);
                    break;
                case 0x11:
                    {
                        printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                           " valueDataType:bool, %s>>0x%08x.\n", valueSize, valueRes0,
                           (*(sKeyString+entryKey)).c_str(), valueData);
                    }
                    break;
                case 0x12:
                    {
                        bool res = (valueData != 0);
                        printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                               " valueDataType:bool, %s>>%s.\n", valueSize, valueRes0,
                               (*(sKeyString+entryKey)).c_str(), res ? "true":"false");
                    }
                    break;
                case 0x1c:
                    {
                        printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                           " valueDataType:#aarrggbb, %s>>#%08x.\n", valueSize, valueRes0,
                           (*(sKeyString+entryKey)).c_str(), valueData);
                    }
                    break;
                case 0x1d:
                    {
                        printf("\t\t__Res_Value >> (key::value) >> valueSize:%d, valueRes0:%d,"
                               " valueDataType:#rrggbb, %s>>#%08x.\n", valueSize, valueRes0,
                               (*(sKeyString+entryKey)).c_str(), valueData);
                    }
                    break;
                
            }
            bodySize = 8 + 8;
        }
    }
    
    //兼容
    if (mResTableType.id == 0x05) {
        preBoolIdStart = hasNum - 1;
    }
    
    offsetEntry += bodySize;
    printf("End Entry >> offset 0x%08x \n\n", offsetEntry);
    baseOffset += mResTableType.header.thunkSize;
}

int main(int argc, const char * argv[]) {
    FILE* mFile = fopen(RESOURCES_ARSC, "r");
    if (nullptr == mFile) {
        printf("Unable open %s", RESOURCES_ARSC);
        return 1;
    }
    
    printf("Start parse resources.arsc ...\n");
    parseResTableHeader(mFile);
    
    printf("Start parse globalStringPool ...\n");
    parseGlobalStringPool(mFile);
    
    printf("Start parse ResTable_Package ...\n");
    parseResTablePackage(mFile);
    
    printf("Start parse ResTypeStringPool ...\n");
    parseResTypeStringPool(mFile);
    
    printf("Start parse ResKeyStringPool ...\n");
    parseResKeyStringPool(mFile);
    
    specOffset = baseOffset;
    
    Header mSpecHeader;
    while (baseOffset < mResTableHeader.header.thunkSize) {
        printf(">> NewBaseOffset >> 0x%08x\n", baseOffset);
        
        fseek(mFile, baseOffset, SEEK_SET);
        char* header = new(nothrow) char[8];
        fread(header, 8, 1, mFile);
        
        mSpecHeader.type = *((uint16_t*) header);
        mSpecHeader.headerSize = *((uint16_t*) (header + 2));
        mSpecHeader.thunkSize = *((uint32_t*) (header + 4));
        delete [] header;
        
        //Spec
        if (mSpecHeader.type == 0x0202) {
            printf("Start parse ResTable_typeSpec ...\n");
            parseResTable_typeSpec(mFile, baseOffset);
        } else {
            printf("Start parse ResTable_type ...\n");
            parseResTable_type(mFile, baseOffset);
        }
    }
    
    fclose(mFile);
    return 0;
}

剩余的问题
1、解析Color多个版本重名的时候 color-v23/会id会错乱
2、float类型数据如何展示, 复数类型显示
3、剩下几种展示尚未遇到。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容