#include #include #include #include #define ULONG uint32_t #define USHORT uint16_t #define UCHAR uint8_t ULONG isdata[1000000]; /* each bit defines one byte as: code=0, data=1 */ extern void dis_asm(ULONG off, ULONG val, char *stg); int static inline le2int(unsigned char* buf) { int32_t res = (buf[3] << 24) | (buf[2] << 16) | (buf[1] << 8) | buf[0]; return res; } int main(int argc, char **argv) { FILE *in, *out; char *ptr, stg[256]; unsigned char buf[4]; ULONG pos, sz, val, loop; int offset, offset1; USHORT regid; if(argc == 1 || strcmp(argv[1], "--help") == 0) { printf("Usage: arm_disass [input file]\n"); printf(" disassembles input file to 'disasm.txt'\n"); exit(-1); } in = fopen(argv[1], "rb"); if(in == NULL) { printf("Cannot open %s", argv[1]); exit(-1); } out = fopen("disasm.txt", "w"); if(out == NULL) exit(-1); fseek(in, 0, SEEK_END); sz = ftell(in); /* first loop only sets data/code tags */ for(loop=0; loop<2; loop++) { for(pos=0; pos assume data */ if((isdata[pos>>5] & (0xf << (pos & 31))) || (val & 0xffff0000) == 0) { sprintf(stg, "%6x: %08x", pos, val); } else { dis_asm(pos, val, stg); /* check for instant mov operation */ if(memcmp(stg+17, "mov ", 4) == 0 && (ptr=strstr(stg, "0x")) != NULL) { regid = *(USHORT*)(stg+22); sscanf(ptr+2, "%x", &offset); if(ptr[-1] == '-') offset = -offset; } else /* check for add/sub operation */ if((ptr=strstr(stg, "0x")) != NULL && (memcmp(stg+17, "add ", 4) == 0 || memcmp(stg+17, "sub ", 4) == 0)) { if(regid == *(USHORT*)(stg+22) && regid == *(USHORT*)(stg+26)) { sscanf(ptr+2, "%x", &offset1); if(ptr[-1] == '-') offset1 = -offset1; if(memcmp(stg+17, "add ", 4) == 0) offset += offset1; else offset -= offset1; /* add result to disassembler string */ sprintf(stg+strlen(stg), " <- 0x%x", offset); } else regid = 0; } else regid = 0; /* check for const data */ if(memcmp(stg+26, "[pc, ", 5) == 0 && (ptr=strstr(stg, "0x")) != NULL) { sscanf(ptr+2, "%x", &offset); if(ptr[-1] == '-') offset = -offset; /* add data tag */ isdata[(pos+offset+8)>>5] |= 1 << ((pos+offset+8) & 31); /* add const data to disassembler string */ fseek(in, pos+offset+8, SEEK_SET); fread(&buf, 1, 4, in); offset = le2int(buf); sprintf(stg+strlen(stg), " <- 0x%x", offset); } } /* remove trailing spaces */ while(stg[strlen(stg)-1] == 32) stg[strlen(stg)-1] = 0; if(loop == 1) fprintf(out, "%s\n", stg); } } fclose(in); return 0; }