X-Git-Url: http://j8takagi.net/cgi-bin/gitweb.cgi?p=YACASL2.git;a=blobdiff_plain;f=src%2Fcmd.c;h=fc5da134a97bdfb0c59881d85409ce6917d8f221;hp=93ae8cde2a6b8182dd211812bd1fad3554afa2c6;hb=7f7bde5dd1305cfc598fe09030c69aad0dc01368;hpb=712486afe58b10ef37c5fa915de889ab8d1dd6cd diff --git a/src/cmd.c b/src/cmd.c index 93ae8cd..fc5da13 100644 --- a/src/cmd.c +++ b/src/cmd.c @@ -1,6 +1,14 @@ -#include "casl2.h" - -CMDCODEARRAY cmdcodearray[] = { +#include +#include +#include +#include "hash.h" +#include "struct.h" +#include "cmem.h" + +/** + * 機械語命令のリスト + */ +static CMD comet2cmd[] = { { "NOP", NONE, 0x0 }, { "LD", R_ADR_X_, 0x1000 }, { "ST", R_ADR_X, 0x1100 }, @@ -38,167 +46,185 @@ CMDCODEARRAY cmdcodearray[] = { { "POP", R_, 0x7100 }, { "CALL", ADR_X, 0x8000 }, { "SVC", ADR_X, 0xF000 }, - { "RET", NONE, 0x8100 } + { "RET", NONE, 0x8100 }, }; -int cmdcodesize = ARRAYSIZE(cmdcodearray); -CMDCODETAB *cmdtype_code[ARRAYSIZE(cmdcodearray)], *code_type[ARRAYSIZE(cmdcodearray)]; +/** + * 命令コード配列のサイズ + */ +static int comet2cmdsize = ARRAYSIZE(comet2cmd); + +/** + * 命令表のサイズ + */ +static int cmdtabsize; + +/** + * ハッシュ表 + */ +static CMDTAB **cmdtype_code, **code_type; + +/** + * 命令の名前とタイプからハッシュ値を生成する + */ +unsigned hash_cmdtype(const char *cmd, CMDTYPE type) +{ + HKEY *keys[2]; + unsigned hashval; + + /* 命令名を設定 */ + 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_chk(keys[0]->val.s, "keys[0].val.s"); + free_chk(keys[0], "keys[0]"); + free_chk(keys[1], "keys[1]"); + /* ハッシュ値を返す */ + return hashval; +} -/* 命令と命令タイプがキーのハッシュ表を作成する */ +/** + * 名前とタイプがキーの命令ハッシュ表を作成する + */ bool create_cmdtype_code() { - CMDCODETAB *np; + CMDTAB *np; unsigned hashval; int i; - for(i = 0; i < cmdcodesize; i++) { - np = (CMDCODETAB *) malloc(sizeof(*np)); - if(np == NULL) { - setcerr(122, NULL); /* cannot create hash table */ - return false; - } + cmdtabsize = comet2cmdsize; + cmdtype_code = calloc_chk(cmdtabsize, sizeof(CMDTAB *), "cmdtype_code"); + for(i = 0; i < cmdtabsize; i++) { + *(cmdtype_code + i) = NULL; + } + for(i = 0; i < comet2cmdsize; i++) { + np = malloc_chk(sizeof(CMDTAB), "cmdtype_code.next"); + np->cmd = NULL; + np->next = NULL; /* ハッシュ値の生成 */ - hashval = hash_cmdtype((&cmdcodearray[i])->cmd, (&cmdcodearray[i])->type); + hashval = hash_cmdtype(comet2cmd[i].name, comet2cmd[i].type); /* ハッシュ表に値を追加 */ np->next = cmdtype_code[hashval]; cmdtype_code[hashval] = np; - np->cca = &(cmdcodearray[i]); + np->cmd = &(comet2cmd[i]); } return true; } -/* 命令と命令タイプからハッシュ値を生成する */ -unsigned hash_cmdtype(const char *cmd, CMDTYPE type) { - HKEY *keys[2]; - - /* 命令をセット */ - keys[0] = malloc(sizeof(HKEY)); - keys[0]->type = CHARS; - keys[0]->val.s = strdup(cmd); - /* 命令タイプをセット */ - keys[1] = malloc(sizeof(HKEY)); - keys[1]->type = INT; - keys[1]->val.i = (int)(type & 070); - /* ハッシュ値を返す */ - return hash(2, keys, cmdcodesize); -} - -/* 命令と命令タイプから、命令コードを取得する */ -/* 無効な場合は0xFFFFを返す */ +/** + * 命令の名前とタイプから、命令コードを返す + * 無効な場合は0xFFFFを返す + */ WORD getcmdcode(const char *cmd, CMDTYPE type) { - CMDCODETAB *np; + CMDTAB *np; + assert(cmd != NULL); for(np = cmdtype_code[hash_cmdtype(cmd, type)]; np != NULL; np = np->next){ - if(strcmp(cmd, np->cca->cmd) == 0 && type == np->cca->type) { - return np->cca->code; + if(strcmp(cmd, np->cmd->name) == 0 && type == np->cmd->type) { + return np->cmd->code; } } return 0xFFFF; } -/* 命令と命令タイプがキーのハッシュ表を表示する */ -void print_cmdtype_code() -{ - int i; - CMDCODETAB *np; - for(i = 0; i < cmdcodesize; i++){ - np = cmdtype_code[i]; - while(np != NULL) { - fprintf(stdout, "(%2d) - %s\t0%02o\t#%04X\n", - i, np->cca->cmd, np->cca->type, np->cca->code); - np = np->next; - } - } -} - -/* 命令と命令タイプがキーのハッシュ表を解放する */ +/** + * 名前とタイプがキーの命令ハッシュ表を解放する + */ void free_cmdtype_code() { int i; - CMDCODETAB *np, *nq; - for(i = 0; i < cmdcodesize; i++){ + CMDTAB *np, *nq; + + for(i = 0; i < cmdtabsize; i++) { np = cmdtype_code[i]; while(np != NULL) { nq = np->next; - free(np); + free_chk(np, "free_cmdtype_code.np"); np = nq; } } + free_chk(cmdtype_code, "cmdtype_code"); +} + +/** + * 命令コードからハッシュ値を生成する + */ +unsigned hash_code(WORD code) +{ + HKEY *keys[1]; + + /* 命令コードを設定 */ + keys[0] = malloc_chk(sizeof(HKEY), "hash_code.key"); + keys[0]->type = INT; + keys[0]->val.i = (int)(code >> 8); + /* ハッシュ値を返す */ + return hash(1, keys, cmdtabsize); } -/* 命令コードがキーのハッシュ表を作成する */ +/** + * コードがキーの命令ハッシュ表を作成する + */ bool create_code_type() { - CMDCODETAB *np; + CMDTAB *np; unsigned hashval; int i; - for(i = 0; i < cmdcodesize; i++) { - if((np = (CMDCODETAB *) malloc(sizeof(*np))) == NULL) { - setcerr(122, NULL); /* cannot create hash table */ - return false; - } + cmdtabsize = comet2cmdsize; + code_type = calloc_chk(cmdtabsize, sizeof(CMDTAB *), "code_type"); + for(i = 0; i < cmdtabsize; i++) { + *(code_type + i) = NULL; + } + for(i = 0; i < comet2cmdsize; i++) { + np = malloc_chk(sizeof(CMDTAB), "code_type.next"); + np->cmd = NULL; + np->next = NULL; /* ハッシュ値の生成 */ - hashval = hash_code((&cmdcodearray[i])->code); + hashval = hash_code((&comet2cmd[i])->code); /* ハッシュ表に値を追加 */ np->next = code_type[hashval]; code_type[hashval] = np; - np->cca = &cmdcodearray[i]; + np->cmd = &comet2cmd[i]; } return true; } -/* 命令コードからハッシュ値を生成する */ -unsigned hash_code(WORD code) -{ - HKEY *keys[1]; - - /* 命令コードをセット */ - keys[0] = malloc(sizeof(HKEY)); - keys[0]->type = INT; - keys[0]->val.i = (int)(code >> 8); - /* ハッシュ値を返す */ - return hash(1, keys, cmdcodesize); -} - -/* 命令コードから命令タイプを取得する */ -/* 無効な場合はNONEを返す */ +/** + * 命令コードから命令タイプを返す + * 無効な場合はNONEを返す + */ CMDTYPE getcmdtype(WORD code) { - CMDCODETAB *np; + CMDTAB *np; for(np = code_type[hash_code(code)]; np != NULL; np = np->next) { - if(code == np->cca->code) { - return np->cca->type; + if(code == np->cmd->code) { + return np->cmd->type; } } return NONE; } -/* 命令コードがキーのハッシュ表を表示する */ -void print_code_type() -{ - int i; - CMDCODETAB *np; - for(i = 0; i < cmdcodesize; i++){ - for(np = code_type[i]; np != NULL; np = np->next) { - fprintf(stdout, "(%2d) - #%04X\t0%02o\t%s\n", - i, np->cca->code, np->cca->type, np->cca->cmd); - } - } -} - -/* 命令コードがキーのハッシュ表を解放する */ +/** + * コードがキーの命令ハッシュ表を解放する + */ void free_code_type() { int i; - CMDCODETAB *np, *nq; - for(i = 0; i < cmdcodesize; i++){ + CMDTAB *np, *nq; + for(i = 0; i < cmdtabsize; i++) { np = code_type[i]; while(np != NULL) { nq = np->next; - free(np); + free_chk(np, "np"); np = nq; } } + free_chk(code_type, "code_type"); }