mirror of
https://github.com/mrkite/regs.git
synced 2025-01-01 09:29:33 +00:00
121 lines
2.6 KiB
C
121 lines
2.6 KiB
C
|
/**
|
||
|
* @copyright 2018 Sean Kasun
|
||
|
* The main disassembler
|
||
|
*/
|
||
|
|
||
|
#include <argp.h>
|
||
|
#include "handle.h"
|
||
|
#include "scan.h"
|
||
|
#include "disasm.h"
|
||
|
|
||
|
const char *argp_program_version = "regs 0.1";
|
||
|
const char *argp_program_bug_address = "sean@seancode.com";
|
||
|
static char doc[] = "Disassemble Apple IIgs software";
|
||
|
static char args_doc[] = "FILE or MAP";
|
||
|
static struct argp_option options[] = {
|
||
|
{"org", 'o', "ADDRESS", 0,
|
||
|
"Starting address of the binary file, in hex"},
|
||
|
{"m", 'm', 0, OPTION_ARG_OPTIONAL,
|
||
|
"Start with 8-bit accumulator"},
|
||
|
{"x", 'x', 0, OPTION_ARG_OPTIONAL,
|
||
|
"Start with 8-bit indices"},
|
||
|
{"e", 'e', 0, OPTION_ARG_OPTIONAL,
|
||
|
"Start in emulation mode"},
|
||
|
{ 0 }
|
||
|
};
|
||
|
|
||
|
struct arguments {
|
||
|
char *filename;
|
||
|
uint32_t org;
|
||
|
MapFlags flags;
|
||
|
};
|
||
|
|
||
|
static inline uint32_t parseNum(const char *s) {
|
||
|
uint32_t res = 0;
|
||
|
while (isspace(*s)) {
|
||
|
s++;
|
||
|
}
|
||
|
while (isxdigit(*s)) {
|
||
|
res <<= 4;
|
||
|
if (*s >= '0' && *s <= '9') {
|
||
|
res |= *s - '0';
|
||
|
} else if (*s >= 'a' && *s <= 'f') {
|
||
|
res |= *s - 'a' + 10;
|
||
|
} else if (*s >= 'A' && *s <= 'F') {
|
||
|
res |= *s - 'A' + 10;
|
||
|
}
|
||
|
s++;
|
||
|
}
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
static error_t parse_opt(int key, char *arg, struct argp_state *state) {
|
||
|
struct arguments *arguments = state->input;
|
||
|
switch (key) {
|
||
|
case 'o':
|
||
|
if (arg) {
|
||
|
arguments->org = parseNum(arg);
|
||
|
}
|
||
|
break;
|
||
|
case 'm':
|
||
|
if (arg) {
|
||
|
arguments->flags |= IsM8;
|
||
|
}
|
||
|
break;
|
||
|
case 'x':
|
||
|
if (arg) {
|
||
|
arguments->flags |= IsX8;
|
||
|
}
|
||
|
break;
|
||
|
case 'e':
|
||
|
if (arg) {
|
||
|
arguments->flags |= IsEmu;
|
||
|
}
|
||
|
break;
|
||
|
case ARGP_KEY_ARG:
|
||
|
if (state->arg_num >= 1) {
|
||
|
argp_usage(state);
|
||
|
}
|
||
|
arguments->filename = arg;
|
||
|
break;
|
||
|
case ARGP_KEY_END:
|
||
|
if (state->arg_num < 1) {
|
||
|
argp_usage(state);
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
return ARGP_ERR_UNKNOWN;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static struct argp argp = { options, parse_opt, args_doc, doc };
|
||
|
|
||
|
int main(int argc, char **argv) {
|
||
|
struct arguments arguments;
|
||
|
arguments.filename = "";
|
||
|
arguments.org = 0x300;
|
||
|
arguments.flags = IsOpcode;
|
||
|
argp_parse(&argp, argc, argv, 0, 0, &arguments);
|
||
|
|
||
|
// load the map, if it exists
|
||
|
Map *map = loadMap(arguments.filename, arguments.org, arguments.flags);
|
||
|
|
||
|
// load the file
|
||
|
FILE *f = fopen(map->filename, "rb");
|
||
|
if (!f) {
|
||
|
fprintf(stderr, "Failed to open '%s'\n", map->filename);
|
||
|
return -1;
|
||
|
}
|
||
|
fseek(f, 0, SEEK_END);
|
||
|
size_t len = ftell(f);
|
||
|
fseek(f, 0, SEEK_SET);
|
||
|
uint8_t *data = malloc(len);
|
||
|
fread(data, len, 1, f);
|
||
|
fclose(f);
|
||
|
|
||
|
scan(data, len, map);
|
||
|
disassemble(data, len, map);
|
||
|
free(data);
|
||
|
}
|