mirror of
https://github.com/st3fan/ewm.git
synced 2025-01-07 19:29:52 +00:00
302 lines
8.2 KiB
C
302 lines
8.2 KiB
C
// The MIT License (MIT)
|
|
//
|
|
// Copyright (c) 2015 Stefan Arentz - http://github.com/st3fan/ewm
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
// copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
// SOFTWARE.
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
#include <getopt.h>
|
|
#include <string.h>
|
|
|
|
#include "cpu.h"
|
|
#include "mem.h"
|
|
#include "pia.h"
|
|
|
|
#include "a2p.h"
|
|
#include "scr.h"
|
|
#include "dsk.h"
|
|
#include "scr.h"
|
|
#include "sdl.h"
|
|
|
|
// TODO Get rid of these, use a struct ewm_t?
|
|
|
|
static struct cpu_t *cpu = NULL;
|
|
static struct a2p_t *a2p = NULL;
|
|
static struct scr_t *scr = NULL;
|
|
|
|
// Machine Setup
|
|
|
|
#define EWM_DISPLAY_TYPE_NONE 0
|
|
#define EWM_DISPLAY_TYPE_TTY 1
|
|
#define EWM_DISPLAY_TYPE_SDL 2
|
|
|
|
typedef int (*ewm_machine_create_f)();
|
|
|
|
struct ewm_machine_t {
|
|
char *name;
|
|
char *description;
|
|
int display_type;
|
|
ewm_machine_create_f create;
|
|
};
|
|
|
|
#define EWM_MEMORY_TYPE_RAM 0
|
|
#define EWM_MEMORY_TYPE_ROM 1
|
|
|
|
struct ewm_memory_t {
|
|
int type;
|
|
char *path;
|
|
uint16_t address;
|
|
struct ewm_memory_t *next;
|
|
};
|
|
|
|
// Apple 1 / 6502 / 8K RAM / WOZ Monitor
|
|
|
|
static int setup_apple1() {
|
|
cpu = cpu_create(EWM_CPU_MODEL_6502);
|
|
struct pia_t *pia = malloc(sizeof(struct pia_t));
|
|
pia_init(pia);
|
|
cpu_add_ram(cpu, 0x0000, 8 * 1024 - 1);
|
|
cpu_add_rom_file(cpu, 0xff00, "roms/apple1.rom");
|
|
cpu_add_iom(cpu, EWM_A1_PIA6820_ADDR, EWM_A1_PIA6820_ADDR + EWM_A1_PIA6820_LENGTH - 1, pia, pia_read, pia_write);
|
|
return 0;
|
|
}
|
|
|
|
// Replica 1 / 65C02 / 32K RAM / Krusader Assembler & Monitor
|
|
|
|
static int setup_replica1() {
|
|
cpu = cpu_create(EWM_CPU_MODEL_65C02);
|
|
struct pia_t *pia = malloc(sizeof(struct pia_t));
|
|
pia_init(pia);
|
|
cpu_add_ram(cpu, 0x0000, 32 * 1024 - 1);
|
|
cpu_add_rom_file(cpu, 0xe000, "roms/krusader.rom");
|
|
cpu_add_iom(cpu, EWM_A1_PIA6820_ADDR, EWM_A1_PIA6820_ADDR + EWM_A1_PIA6820_LENGTH - 1, pia, pia_read, pia_write);
|
|
return 0;
|
|
}
|
|
|
|
// Apple ][+ / 6502 / 48K RAM / Original ROMs
|
|
|
|
static int setup_apple2plus() {
|
|
cpu = cpu_create(EWM_CPU_MODEL_6502);
|
|
a2p = a2p_create(cpu);
|
|
scr = ewm_scr_create(a2p, renderer);
|
|
return 0;
|
|
}
|
|
|
|
struct ewm_memory_t *parse_memory(char *s) {
|
|
char *type = strsep(&s, ":");
|
|
if (type == NULL) { // || (strcmp(type, "ram") && strcmp(type, "rom"))) {
|
|
printf("type fail\n");
|
|
return NULL;
|
|
}
|
|
|
|
char *address = strsep(&s, ":");
|
|
if (address == NULL) {
|
|
printf("address fail\n");
|
|
return NULL;
|
|
}
|
|
|
|
char *path = strsep(&s, ":");
|
|
if (path == NULL) {
|
|
printf("path fail\n");
|
|
return NULL;
|
|
}
|
|
|
|
struct ewm_memory_t *m = (struct ewm_memory_t*) malloc(sizeof(struct ewm_memory_t));
|
|
m->type = strcmp(type, "ram") ? EWM_MEMORY_TYPE_RAM : EWM_MEMORY_TYPE_ROM;
|
|
m->path = path;
|
|
m->address = atoi(address);
|
|
m->next = NULL;
|
|
|
|
return m;
|
|
}
|
|
|
|
static struct ewm_machine_t machines[] = {
|
|
{ "apple1", "Apple 1", EWM_DISPLAY_TYPE_TTY, setup_apple1 },
|
|
{ "replica1", "Replica 1", EWM_DISPLAY_TYPE_TTY, setup_replica1 },
|
|
{ "apple2plus", "Apple ][+", EWM_DISPLAY_TYPE_SDL, setup_apple2plus },
|
|
{ NULL, NULL, EWM_DISPLAY_TYPE_NONE, NULL }
|
|
};
|
|
|
|
static struct option options[] = {
|
|
{ "machine", required_argument, NULL, 'm' },
|
|
{ "strict", no_argument, NULL, 's' },
|
|
{ "trace", optional_argument, NULL, 't' },
|
|
{ "memory", required_argument, NULL, 'x' },
|
|
{ "drive1", required_argument, NULL, 'a' },
|
|
{ "drive2", required_argument, NULL, 'b' },
|
|
{ NULL, 0, NULL, 0 }
|
|
};
|
|
|
|
static struct ewm_machine_t *machine_with_name(char *name) {
|
|
for (struct ewm_machine_t *m = machines; m->name != NULL; m++) {
|
|
if (strcmp(m->name, name) == 0) {
|
|
return m;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
int main(int argc, char **argv) {
|
|
struct ewm_machine_t *machine = &machines[0];
|
|
bool strict = false;
|
|
char *trace_path = NULL;
|
|
struct ewm_memory_t *memory = NULL;
|
|
char *drive1 = NULL;
|
|
char *drive2 = NULL;
|
|
|
|
char ch;
|
|
while ((ch = getopt_long(argc, argv, "m:", options, NULL)) != -1) {
|
|
switch (ch) {
|
|
case 'm':
|
|
machine = machine_with_name(optarg);
|
|
break;
|
|
case 's':
|
|
strict = true;
|
|
break;
|
|
case 't':
|
|
trace_path = optarg ? optarg : "/dev/stderr";
|
|
break;
|
|
case 'x': {
|
|
struct ewm_memory_t *m = parse_memory(optarg);
|
|
if (m == NULL) {
|
|
exit(1);
|
|
}
|
|
if (memory == NULL) {
|
|
memory = m;
|
|
} else {
|
|
memory->next = m;
|
|
}
|
|
break;
|
|
}
|
|
case 'a':
|
|
drive1 = optarg;
|
|
break;
|
|
case 'b':
|
|
drive2 = optarg;
|
|
break;
|
|
}
|
|
}
|
|
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
if (machine == NULL) {
|
|
fprintf(stderr, "Usage: ewm --machine apple1|replica1|apple2plus\n");
|
|
exit(1);
|
|
}
|
|
|
|
//
|
|
// Initialize the display
|
|
//
|
|
|
|
switch (machine->display_type) {
|
|
case EWM_DISPLAY_TYPE_NONE:
|
|
fprintf(stderr, "[EWM] TODO EWM_DISPLAY_TYPE_NONE\n");
|
|
exit(1);
|
|
break;
|
|
case EWM_DISPLAY_TYPE_TTY:
|
|
fprintf(stderr, "[EWM] TODO EWM_DISPLAY_TYPE_TTY\n");
|
|
break;
|
|
case EWM_DISPLAY_TYPE_SDL:
|
|
sdl_initialize();
|
|
break;
|
|
}
|
|
|
|
// Create the machine and initialize it
|
|
|
|
fprintf(stderr, "[EWM] Creating %s\n", machine->description);
|
|
|
|
machine->create(cpu);
|
|
|
|
// TODO This really does not belong here. it is probably better to
|
|
// pass arguments to setup_apple2plus so that it can handle its
|
|
// specific arguments like --drive1/drive2 and probably some more
|
|
// in the future.
|
|
|
|
if (a2p != NULL) {
|
|
if (drive1 != NULL) {
|
|
if (a2p_load_disk(a2p, EWM_DSK_DRIVE1, drive1) != 0) {
|
|
fprintf(stderr, "[A2P] Cannot load Drive 1 with %s\n", drive1);
|
|
exit(1);
|
|
}
|
|
}
|
|
if (drive2 != NULL) {
|
|
if (a2p_load_disk(a2p, EWM_DSK_DRIVE2, drive2) != 0) {
|
|
fprintf(stderr, "[A2P] Cannot load Drive 2 with %s\n", drive2);
|
|
exit(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
struct ewm_memory_t *m = memory;
|
|
while (m != NULL) {
|
|
fprintf(stderr, "[EWM] Adding %s $%.4X %s\n", m->type == EWM_MEMORY_TYPE_RAM ? "RAM" : "ROM", m->address, m->path);
|
|
if (m->type == EWM_MEMORY_TYPE_RAM) {
|
|
if (cpu_add_ram_file(cpu, m->address, m->path) == NULL) {
|
|
fprintf(stderr, "[EWM] Failed to add RAM from %s\n", m->path);
|
|
exit(1);
|
|
}
|
|
} else {
|
|
if (cpu_add_rom_file(cpu, m->address, m->path) == NULL) {
|
|
fprintf(stderr, "[EWM] Failed to add ROM from %s\n", m->path);
|
|
exit(1);
|
|
}
|
|
}
|
|
m = m->next;
|
|
}
|
|
|
|
cpu_strict(cpu, strict);
|
|
cpu_trace(cpu, trace_path);
|
|
|
|
cpu_reset(cpu);
|
|
|
|
switch (machine->display_type) {
|
|
case EWM_DISPLAY_TYPE_NONE:
|
|
fprintf(stderr, "[EWM] TODO EWM_DISPLAY_TYPE_NONE\n");
|
|
exit(1);
|
|
break;
|
|
case EWM_DISPLAY_TYPE_TTY:
|
|
fprintf(stderr, "[EWM] TODO EWM_DISPLAY_TYPE_TTY\n");
|
|
exit(1);
|
|
break;
|
|
case EWM_DISPLAY_TYPE_SDL:
|
|
sdl_main(cpu, a2p, scr);
|
|
break;
|
|
}
|
|
|
|
// Clean up
|
|
|
|
if (scr != NULL) {
|
|
ewm_scr_destroy(scr);
|
|
scr = NULL;
|
|
}
|
|
|
|
if (a2p != NULL) {
|
|
a2p_destroy(a2p);
|
|
a2p = NULL;
|
|
}
|
|
|
|
cpu_destroy(cpu);
|
|
|
|
return 0;
|
|
}
|