-#include "casl2.h"
-
-/* COMET IIのメモリ */
-WORD *memory;
-
-/* COMET IIのCPUレジスタ */
-WORD GR[REGSIZE], SP, PR, FR;
-
-CERRARRAY cerr[] = {
- { 101, "label already defined" },
- { 102, "label table is full" },
- { 103, "label not found" },
- { 104, "label length is too long" },
- { 105, "no command in the line" },
- { 106, "operand count mismatch" },
- { 107, "no label in START" },
- { 108, "not command of operand \"r\"" },
- { 109, "not command of operand \"r1,r2\"" },
- { 110, "not command of operand \"r,adr[,x]\"" },
- { 111, "not command of operand \"adr[,x]\"" },
- { 112, "not command of no operand" },
- { 113, "command not defined" },
- { 114, "not integer" },
- { 115, "not hex" },
- { 116, "out of hex range" },
- { 117, "operand is too many" },
- { 118, "operand length is too long" },
- { 119, "out of COMET II memory" },
- { 120, "GR0 in operand x" },
- { 121, "cannot get operand token" },
- { 122, "cannot create hash table" },
- { 201, "execute - out of COMET II memory" },
- { 202, "SVC input - out of Input memory" },
- { 203, "SVC output - out of COMET II memory" },
- { 204, "Program Register (PR) - out of COMET II memory" },
- { 205, "Stack Pointer (SP) - cannot allocate stack buffer" },
- { 206, "Address - out of COMET II memory" },
- { 207, "Stack Pointer (SP) - out of COMET II memory" },
- { 0, NULL },
+#include "struct.h"
+#include "exec.h"
+
+/**
+ * COMET IIの仮想実行マシンシステム
+ */
+SYSTEM *sys = NULL;
+
+/**
+ * プログラム実行時の開始と終了のアドレス
+ */
+EXECPTR *execptr = NULL;
+
+/**
+ * システムCOMET IIの命令表
+ */
+static const COMET2CMD comet2cmd[] = {
+ { "NOP", NONE, 0x0, nop },
+ { "LD", R_ADR_X, 0x1000, ld_r_adr_x },
+ { "ST", R_ADR_X, 0x1100, st },
+ { "LAD", R_ADR_X, 0x1200, lad },
+ { "LD", R1_R2, 0x1400, ld_r1_r2 },
+ { "ADDA", R_ADR_X, 0x2000, adda_r_adr_x },
+ { "SUBA", R_ADR_X, 0x2100, suba_r_adr_x },
+ { "ADDL", R_ADR_X, 0x2200, addl_r_adr_x },
+ { "SUBL", R_ADR_X, 0x2300, subl_r_adr_x },
+ { "ADDA", R1_R2, 0x2400, adda_r1_r2 },
+ { "SUBA", R1_R2, 0x2500, suba_r1_r2 },
+ { "ADDL", R1_R2, 0x2600, addl_r1_r2 },
+ { "SUBL", R1_R2, 0x2700, subl_r1_r2 },
+ { "AND", R_ADR_X, 0x3000, and_r_adr_x },
+ { "OR", R_ADR_X, 0x3100, or_r_adr_x },
+ { "XOR", R_ADR_X, 0x3200, xor_r_adr_x },
+ { "AND", R1_R2, 0x3400, and_r1_r2 },
+ { "OR", R1_R2, 0x3500, or_r1_r2 },
+ { "XOR", R1_R2, 0x3600, xor_r1_r2 },
+ { "CPA", R_ADR_X, 0x4000, cpa_r_adr_x },
+ { "CPL", R_ADR_X, 0x4100, cpl_r_adr_x },
+ { "CPA", R1_R2, 0x4400, cpa_r1_r2 },
+ { "CPL", R1_R2, 0x4500, cpl_r1_r2 },
+ { "SLA", R_ADR_X, 0x5000, sla },
+ { "SRA", R_ADR_X, 0x5100, sra },
+ { "SLL", R_ADR_X, 0x5200, sll },
+ { "SRL", R_ADR_X, 0x5300, srl },
+ { "JMI", ADR_X, 0x6100, jmi },
+ { "JNZ", ADR_X, 0x6200, jnz },
+ { "JZE", ADR_X, 0x6300, jze },
+ { "JUMP", ADR_X, 0x6400, jump },
+ { "JPL", ADR_X, 0x6500, jpl },
+ { "JOV", ADR_X, 0x6600, jov },
+ { "PUSH", ADR_X, 0x7000, push },
+ { "POP", R_, 0x7100, pop },
+ { "CALL", ADR_X, 0x8000, call },
+ { "SVC", ADR_X, 0xF000, svc },
+ { "RET", NONE, 0x8100, ret },
};
-/* レジストリの内容を表示する場合はtrue */
-bool tracemode = false;
+/**
+ * 命令表のサイズ
+ */
+static int comet2cmdsize = ARRAYSIZE(comet2cmd);
+
+/**
+ * ハッシュ表のサイズ
+ */
+enum {
+ CMDTABSIZE = 41,
+};
+
+/**
+ * ハッシュ表
+ */
+static CMDTAB *cmdtab[HASH_MAX][CMDTABSIZE] = {{NULL}};
+
+/**
+ * 命令の名前とタイプからハッシュ値を生成する
+ */
+unsigned hash_cmdtype(const char *cmd, CMDTYPE type);
+
+/**
+ * 命令コードからハッシュ値を生成する
+ */
+unsigned hash_code(WORD code);
+
+/**
+ * 命令の名前とタイプからハッシュ値を生成する
+ */
+unsigned hash_cmdtype(const char *cmd, CMDTYPE type)
+{
+ HKEY *keys[2] = {NULL};
+ unsigned hashval = 0;
+
+ /* 命令名を設定 */
+ keys[0] = malloc_chk(sizeof(HKEY), "hash_cmdtype.keys[0]");
+ keys[0]->type = CHARS;
+ keys[0]->val.s = strdup_chk(cmd, "keys[0].val.s");
+ /* 命令タイプを設定 */
+ keys[1] = malloc_chk(sizeof(HKEY), "hash_cmdtype.keys[1]");
+ keys[1]->type = INT;
+ keys[1]->val.i = (int)(type & 070);
+ /* ハッシュ値の計算 */
+ hashval = hash(2, keys, CMDTABSIZE);
+ FREE(keys[0]->val.s);
+ FREE(keys[0]);
+ FREE(keys[1]);
+ /* ハッシュ値を返す */
+ return hashval;
+}
+
+/**
+ * 命令ハッシュ表を作成する
+ */
+bool create_cmdtable(CMDTAB_HASH hash)
+{
+ CMDTAB *p = NULL;
+ unsigned hashval;
+
+ for(int i = 0; i < comet2cmdsize; i++) {
+ p = malloc_chk(sizeof(CMDTAB), "create_cmdtable.p");
+ p->cmd = &comet2cmd[i];
+ if(hash == HASH_CMDTYPE) {
+ hashval = hash_cmdtype(comet2cmd[i].name, comet2cmd[i].type);
+ } else if(hash == HASH_CODE) {
+ hashval = hash_code(comet2cmd[i].code);
+ }
+ p->next = cmdtab[hash][hashval];
+ cmdtab[hash][hashval] = p;
+ }
+ return true;
+}
+
+/**
+ * 命令ハッシュ表を解放する
+ */
+void free_cmdtable(CMDTAB_HASH hash)
+{
+ CMDTAB *p = NULL;
+ CMDTAB *q = NULL;
+
+ for(int i = 0; i < CMDTABSIZE; i++) {
+ for(p = cmdtab[hash][i]; p != NULL; p = q) {
+ q = p->next;
+ FREE(p);
+ }
+ cmdtab[hash][i] = NULL;
+ }
+}
+
+/**
+ * 命令の名前とタイプから、命令コードを返す\n
+ * 無効な場合は0xFFFFを返す
+ */
+WORD getcmdcode(const char *cmd, CMDTYPE type)
+{
+ CMDTAB *p = NULL;
+ WORD w = 0xFFFF;
+
+ assert(cmd != NULL);
+ for(p = cmdtab[HASH_CMDTYPE][hash_cmdtype(cmd, type)]; p != NULL; p = p->next) {
+ if(strcmp(cmd, p->cmd->name) == 0 && type == p->cmd->type) {
+ w = p->cmd->code;
+ break;
+ }
+ }
+ return w;
+}
+
+/**
+ * 命令コードからハッシュ値を生成する
+ */
+unsigned hash_code(WORD code)
+{
+ HKEY *keys[1] = {NULL};
+ unsigned h = 0;
+
+ /* 命令コードを設定 */
+ keys[0] = malloc_chk(sizeof(HKEY), "hash_code.key");
+ keys[0]->type = INT;
+ keys[0]->val.i = (int)(code >> 8);
+ h = hash(1, keys, CMDTABSIZE);
+ FREE(keys[0]);
+ return h;
+}
-/* レジストリの内容を論理値(0〜65535)で表示する場合はtrue */
-bool logicalmode = false;
+/**
+ * 命令コードから命令の関数ポインタを返す
+ */
+const void (*getcmdptr(WORD code))
+{
+ CMDTAB *t = NULL;
+ const void *ptr = NULL;
-/* メモリの内容を表示する場合はtrue */
-bool dumpmode = false;
+ for(t = cmdtab[HASH_CODE][hash_code(code)]; t != NULL; t = t->next) {
+ if(code == t->cmd->code) {
+ ptr = t->cmd->ptr;
+ break;
+ }
+ }
+ return ptr;
+}
-/* ソースを表示する場合はtrue */
-bool srcmode = false;
+/**
+ * 命令コードから命令のタイプを返す
+ */
+CMDTYPE getcmdtype(WORD code)
+{
+ CMDTAB *t = NULL;
+ CMDTYPE type = NONE;
-/* ラベル表を表示する場合はtrue */
-bool labelmode = false;
+ for(t = cmdtab[HASH_CODE][hash_code(code)]; t != NULL; t = t->next) {
+ if(code == t->cmd->code) {
+ type = t->cmd->type;
+ break;
+ }
+ }
+ return type;
+}
-/* ラベル表を表示して終了する場合はtrue */
-bool onlylabelmode = false;
+/**
+ * 命令コードから命令の名前を返す
+ */
+char *getcmdname(WORD code)
+{
+ CMDTAB *t = NULL;
+ char *cmd = NULL;
-/* アセンブラ詳細結果を表示する場合はtrue */
-bool asdetailmode = false;
+ for(t = cmdtab[HASH_CODE][hash_code(code)]; t != NULL; t = t->next) {
+ if(code == t->cmd->code) {
+ cmd = t->cmd->name;
+ break;
+ }
+ }
+ return cmd;
+}
-/* アセンブルだけを行う場合はtrue */
-bool onlyassemblemode = false;
+/**
+ * 汎用レジスタの番号からレジスタを表す文字列を返す
+ */
-/* メモリーサイズ */
-int memsize = DEFAULT_MEMSIZE;
+char *grstr(WORD word)
+{
+ assert(word <= 7);
+ char *str = NULL;
-/* クロック周波数 */
-int clocks = DEFAULT_CLOCKS;
+ str = malloc_chk(3 + 1, "grstr.str");
+ sprintf(str, "GR%d", word);
+ return str;
+}
-/* 実行開始番地 */
-WORD startptr = 0x0;
+/**
+ * COMET II仮想マシンのリセット
+ */
+void reset(int memsize, int clocks)
+{
+ sys = malloc_chk(sizeof(SYSTEM), "sys");
+ /* メモリサイズを設定 */
+ sys->memsize = memsize;
+ /* クロック周波数を設定 */
+ sys->clocks = clocks;
+ /* メモリを初期化 */
+ sys->memory = calloc_chk(sys->memsize, sizeof(WORD), "memory");
+ /* CPUを初期化 */
+ sys->cpu = malloc_chk(sizeof(CPU), "cpu");
+ for(int i = 0; i < GRSIZE; i++) { /* 汎用レジスタ */
+ sys->cpu->gr[i] = 0x0;
+ }
+ sys->cpu->sp = sys->memsize; /* スタックポインタ */
+ sys->cpu->pr = 0x0; /* プログラムレジスタ */
+ sys->cpu->fr = 0x0; /* フラグレジスタ */
+ /* CASL2プログラムの開始と終了のアドレスを初期化 */
+ execptr = malloc_chk(sizeof(EXECPTR), "execptr");
+ execptr->stop = false;
+}
-/* 実行終了番地 */
-WORD endptr = 0x0;
+/**
+ * COMET II仮想マシンのシャットダウン
+ */
+void shutdown()
+{
+ FREE(execptr);
+ FREE(sys->memory);
+ FREE(sys->cpu);
+ FREE(sys);
+}