diff system/system_config.cc @ 0:223b71206888

Initial import
author thib
date Fri, 01 Aug 2008 16:32:45 +0000
parents
children b6d6c0ffb423
line wrap: on
line diff
new file mode 100644
--- /dev/null
+++ b/system/system_config.cc
@@ -0,0 +1,1219 @@
+/*  system_config.cc
+ *      gameexe.ini ファイルの読み込み
+ */
+
+/*
+ *
+ *  Copyright (C) 2000-   Kazunori Ueno(JAGARL) <jagarl@creator.club.ne.jp>
+ *
+ *  This program is free software; you can redistribute it and/or modify
+ *  it under the terms of the GNU General Public License as published by
+ *  the Free Software Foundation; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This program is distributed in the hope that it will be useful,
+ *  but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *  GNU General Public License for more details.
+ *
+ *  You should have received a copy of the GNU General Public License
+ *  along with this program; if not, write to the Free Software
+ *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ *
+*/
+
+#include <stdio.h>
+#include <string.h>
+#include <stdlib.h>
+#include <stdarg.h>
+#include <ctype.h>
+#include <map>
+#include <string>
+#include "system_config.h"
+#include "../system/file.h"
+
+using namespace std;
+
+// #define DEBUG_CONFIG
+#ifdef DEBUG_CONFIG
+# define dprintf(X) printf X
+#else
+# define dprintf(X)
+#endif /* DEBUG_CONFIG */
+
+#define MAXTOKEN 10 /* = で区切られた領域の最大数 */
+#define MAXVARS 32 /* , で区切られた数値の最大数 */
+
+// 初期化ファイルの読み込み
+/* config は 文字列、数列、その複合など、いろいろな形式がありうる */
+/* 文字列と数列は一般に AyuSys_Config クラスに含める */
+
+
+/**********************************************************/
+/* とりあえずハッシュ比較付き文字列 */
+class HashStr {
+	const char* str;
+	unsigned int hash;
+public:
+	HashStr(const char*);
+	HashStr(const HashStr& orig);
+	~HashStr() {
+		if (str) delete[] str;
+	}
+	const char* c_str(void) const { return str; }
+	friend inline int operator<(const HashStr& a, const HashStr& b) {
+		if (a.hash == b.hash) {
+			if (a.str == 0) return 1;
+			else if (b.str == 0) return 0;
+			else return strcmp(a.str, b.str);
+		}
+		else return a.hash < b.hash;
+	}
+};
+HashStr::HashStr(const char* s ) {
+	if (s == 0 || s[0] == '\0') {
+		str = 0; hash = 0; return; /* invalid string */
+	}
+	char* new_str = new char[strlen(s)+1];
+	strcpy(new_str, s);
+	str = new_str;
+	/* calc hash... 適当 */
+	int h = strlen(s);
+	while(*s != 0) {
+		h = *s + ((h * (0x9449+*s))>>7);
+		s++;
+	}
+	hash = (unsigned int)h;
+}
+HashStr::HashStr(const HashStr& orig) {
+	if (orig.str == 0 || orig.str[0] == '\0') {
+		str = 0; hash = 0; return; /* invalid */
+	}
+	char* new_str = new char[strlen(orig.str)+1];
+	strcpy(new_str, orig.str);
+	str = new_str;
+	hash = orig.hash;
+}
+
+/**********************************************************
+**AyuSys_Config_[String | Intlist] :
+** 設定の本体
+**	original : 元設定
+**	old_data : 前回 ClearDiff() したときの設定
+**	new_data : ClearDiff() 以降に設定した内容を保存
+**	データ設定:
+**	Init() : 元設定を作成
+**	Set() : 設定を変更
+**	Get() : 最も新しい設定を得る
+**
+**	変更の記録:
+**	Diff() : 前回のClearDiff() から変更した内容を得る
+**	DiffLen() : Diff() で必要な文字列長を得る
+**	ClearDiff() : 変更記録を消す
+**	PatchOld() : Diff() で得た記録に基づき、変更前の状態に戻す
+**	PatchNew() : Diff() で得た記録に基づき、変更後の状態に戻す
+**
+**	元設定からの変更の記録:
+**	SetOriginal() : 元設定に戻す
+**	DiffOriginal() : 元設定から現在の設定の変更を得る
+**	DiffOriginalLen() : DiffOriginal() で必要な文字列長を得る
+**	PatchOriginal() : DiffOriginal() で得た記録に基づき、設定を復旧する
+*/
+
+/************************************************
+** AyuSysConfigStringItem
+** 文字列をデータとしてもつ設定項目
+*/
+class AyuSysConfigStringItem {
+	char* original_data;
+	char* old_data;
+	char* new_data;
+public:
+	AyuSysConfigStringItem(void) {
+		original_data = 0;
+		old_data = 0;
+		new_data = 0;
+	}
+	AyuSysConfigStringItem(const AyuSysConfigStringItem& o) {
+		original_data = 0; old_data = 0; new_data = 0;
+		if (o.original_data) {
+			original_data = new char[strlen(o.original_data)+1];
+			strcpy(original_data, o.original_data);
+		}
+		if (o.old_data) {
+			old_data = new char[strlen(o.old_data)+1];
+			strcpy(old_data, o.old_data);
+		}
+		if (o.new_data) {
+			new_data = new char[strlen(o.new_data)+1];
+			strcpy(new_data, o.new_data);
+		}
+	}
+	/* 設定:Init で初期化、Set で変更、Get で変更を優先して取り出す */
+	void Init(int deal, const char* str) { /* deal は無視 */
+		if (original_data) delete[] original_data;
+		int len = strlen(str);
+		original_data = new char[len+1];
+		strcpy(original_data, str);
+		original_data[len] = '\0';
+	}
+	void Set(int deal, const char* str) { /* deal は無視 */
+		if (new_data) delete[] new_data;
+		int len = strlen(str);
+		new_data = new char[len+1];
+		strcpy(new_data, str);
+		new_data[len] = '\0';
+	}
+	const char* Get(int deal) const {/* deal は無視 */
+		if (new_data) return new_data;
+		else if (old_data) return old_data;
+		return original_data;
+	}
+	const char* GetOriginal(int deal) const {
+		return original_data;
+	}
+	int Deal(void) const {
+		return 1;
+	}
+	/* オリジナルからの変化の調査 :
+	** DiffOriginal で変化を文字列で取り出し、PatchOriginal で
+	** 変化を反映
+	*/
+	int DiffOriginalLen(void) {
+		if (new_data == 0) return 0;
+		return strlen(new_data)+1;
+	}
+	void DiffOriginal(string& data) {
+		if (new_data == 0) { /* あり得ない */
+			fprintf(stderr,"AyuSysConfigStringItem::DiffOriginal : this method must not called if not required!\n");
+			return;
+		}
+		char* out_data = new char[strlen(new_data)*2+1];
+		char* buf = out_data;
+		int i;
+		for (i=0; new_data[i]!=0; i++) {
+			switch(new_data[i]) {
+				case '?': *buf++ = '?'; *buf++ = '0'; break;
+				case '"': *buf++ = '?'; *buf++ = '1'; break;
+				case '\'': *buf++ = '?'; *buf++ = '2'; break;
+				case ',': *buf++ = '?'; *buf++ = '3'; break;
+				case '.': *buf++ = '?'; *buf++ = '4'; break;
+				case ':': *buf++ = '?'; *buf++ = '5'; break;
+				case ';': *buf++ = '?'; *buf++ = '6'; break;
+				case '=': *buf++ = '?'; *buf++ = '7'; break;
+				case '<': *buf++ = '?'; *buf++ = '8'; break;
+				case '>': *buf++ = '?'; *buf++ = '9'; break;
+				default: *buf++ = new_data[i]; break;
+			}
+		}
+		*buf++ = 0;
+		data += out_data;
+		delete[] out_data;
+		return;
+	}
+	const char* PatchOriginal(const char* data) {
+		static char* table = "?\"',.:;=<>";
+		if (new_data) delete[] new_data;
+		if (old_data) delete[] old_data;
+		new_data = 0; old_data = 0;
+		new_data = new char[1024];
+		int i,j = 0;
+		for (i=0; i<1020; i++) {
+			switch(data[j]) {
+			case '?':
+				if (data[j+1] >= '0' && data[j+1] <= '9') {
+					new_data[i] = table[ data[j+1] - '0'];
+					j += 2;
+					break;
+				}
+			case '"': case '\'': case ',': case '.': case ':':
+			case ';': case '=':  case '<': case '>':
+				goto for_end;
+			default: new_data[i] = data[j++]; break;
+			}
+		}
+	for_end:
+		new_data[i] = 0;
+		return data;
+	}
+	void SetOriginal(void) {
+		if (new_data) delete[] new_data;
+		if (old_data) delete[] old_data;
+		new_data = 0; old_data = 0;
+	}
+	void Dump(FILE* f) const {
+		if (original_data) fprintf(f, "original %s ",original_data);
+		if (old_data) fprintf(f, "old_data %s ",old_data);
+		if (new_data) fprintf(f, "new_data %s ",new_data);
+		fprintf(f, "\n");
+	}
+};
+
+/************************************************
+** AyuSysConfigIntlistItem
+** 数値列をデータとしてもつ設定項目
+*/
+class AyuSysConfigIntlistItem {
+	int item_deal;
+	int* original_data;
+	int* old_data;
+	int* new_data;
+public:
+	AyuSysConfigIntlistItem(void) {
+		item_deal = 0;
+		original_data = 0;
+		old_data = 0;
+		new_data = 0;
+	}
+	AyuSysConfigIntlistItem(const AyuSysConfigIntlistItem& o) {
+		item_deal = o.item_deal;
+		original_data = 0; old_data = 0; new_data = 0;
+		if (o.original_data) {
+			original_data = new int[item_deal];
+			memcpy(original_data, o.original_data, sizeof(int)*item_deal);
+		}
+		if (o.old_data) {
+			old_data = new int[item_deal];
+			memcpy(old_data, o.old_data, sizeof(int)*item_deal);
+		}
+		if (o.new_data) {
+			new_data = new int[item_deal];
+			memcpy(new_data, o.new_data, sizeof(int)*item_deal);
+		}
+	}
+	/* 設定:Init で初期化、Set で変更、Get で変更を優先して取り出す */
+	void Init(int deal, const int* list) { /* deal は無視 */
+		if (original_data) delete[] original_data;
+		original_data = 0;
+		if (deal <= 0) {
+			item_deal = 0; return;
+		}
+		item_deal = deal;
+		original_data = new int[item_deal];
+		memcpy(original_data, list, sizeof(int)*deal);
+	}
+	void Set(int deal, const int* list) { /* deal は無視 */
+		item_deal = deal;
+		if (new_data) delete[] new_data;
+		new_data = new int[item_deal];
+		memcpy(new_data, list, sizeof(int)*item_deal);
+	}
+	const int* Get(int deal) const {/* deal は無視 */
+		if (item_deal == 0) return 0;
+		if (deal > item_deal) {
+			fprintf(stderr,"AyuSysConfigIntlistItem::Get : invalid items deal %d (correct: %d)\n",deal,item_deal);
+			return 0;
+		}
+		if (new_data) return new_data;
+		else if (old_data) return old_data;
+		return original_data;
+	}
+	const int* GetOriginal(int deal) const {/* deal は無視 */
+		if (item_deal == 0) return 0;
+		if (deal > item_deal) {
+			fprintf(stderr,"AyuSysConfigIntlistItem::Get : invalid items deal %d (correct: %d)\n",deal,item_deal);
+			return 0;
+		}
+		return original_data;
+	}
+	int Deal(void) const {
+		return item_deal;
+	}
+	/* オリジナルからの変化の調査 :
+	** DiffOriginal で変化を文字列で取り出し、PatchOriginal で
+	** 変化を反映
+	*/
+	int DiffOriginalLen(void) {
+		if (new_data == 0) return 0;
+		return  12 * item_deal + 1;
+	}
+	void DiffOriginal(string& data) {
+		if (new_data == 0) { /* あり得ない */
+			fprintf(stderr,"AyuSysConfigStringItem::DiffOriginal : this method must not called if not required!\n");
+			return;
+		}
+		int i; char buf[1024];
+		for (i=0; i<item_deal; i++) {
+			sprintf(buf, "%d,",new_data[i]);
+			data += buf;
+		}
+		return;
+	}
+	const char* PatchOriginal(const char* data) {
+		if (old_data) delete[] old_data;
+		if (new_data) delete[] new_data;
+		old_data = 0; new_data = 0;
+		new_data = new int[item_deal];
+		int i;
+		for (i=0; i<item_deal; i++) {
+			new_data[i] = atoi(data);
+			if (strchr(data, ',') == 0) break;
+			data = strchr(data, ',') + 1;
+		}
+		return data;
+	}
+	void SetOriginal(void) {
+		if (new_data) delete[] new_data;
+		if (old_data) delete[] old_data;
+		new_data = 0; old_data = 0;
+	}
+	void Dump(FILE* f) const {
+		fprintf(f, "item deal %d, ",item_deal);
+		if (original_data) {
+			fprintf(f, "(%d", original_data[0]);
+			int i;for (i=1; i<item_deal; i++) {
+				fprintf(f, ",%d",original_data[i]);
+			}
+			fprintf(f, ") ");
+		}
+		if (old_data) {
+			fprintf(f, "old %08x(%d", (unsigned int)(old_data), old_data[0]);
+			int i;for (i=1; i<item_deal; i++) {
+				fprintf(f, ",%d",old_data[i]);
+			}
+			fprintf(f, ") ");
+		}
+		if (new_data) {
+			fprintf(f, "new %08x(%d", (unsigned int)(new_data), new_data[0]);
+			int i;for (i=1; i<item_deal; i++) {
+				fprintf(f, ",%d",new_data[i]);
+			}
+			fprintf(f, ") ");
+		}
+		fprintf(f, "\n");
+	}
+};
+
+// template map<HashStr, AyuSysConfigStringItem>;
+// template map<HashStr, AyuSysConfigIntlistItem>;
+
+/************************************************
+** AyuSysConfigItem
+** データ名 -> データ本体の map と、map 全体に
+** 様々な操作を行うためのメソッド
+*/
+
+template<class ItemType, class DataType> class AyuSysConfigItem {
+	typedef map<HashStr,ItemType> maptype;
+	typedef typename maptype::iterator mapiterator;
+	typedef typename maptype::const_iterator const_mapiterator;
+	maptype data;
+public:
+	void SetOrig(HashStr& name, int deal, const DataType* str) {
+		if (str == 0) return; /* 無効 */
+		data[name].Init(deal, str);
+	}
+	void Set(HashStr& name, int deal, const DataType* new_data) {
+		if (new_data == 0) return; /* 無効 */
+		/* 設定を検索 */
+		mapiterator it = data.find(name);
+		/* 設定が元設定に見つからないなら失敗 */
+		if (it == data.end()) {
+			fprintf(stderr,"AyuSysConfigItem::Set : there is no '%s' parameter\n",name.c_str());
+			return;
+		}
+		/* 設定を変更 */
+		it->second.Set(deal, new_data);
+	}
+	/* 新しい設定を優先して返す */
+	const DataType* Get(int deal, HashStr& name) const {
+		const_mapiterator it = data.find(name);
+		if (it == data.end()) return 0;
+		return it->second.Get(deal);
+	}
+	const DataType* GetOriginal(int deal, HashStr& name) const {
+		const_mapiterator it = data.find(name);
+		if (it == data.end()) return 0;
+		return it->second.GetOriginal(deal);
+	}
+	int Deal(HashStr& name) const {
+		const_mapiterator it = data.find(name);
+		if (it == data.end()) return 0;
+		return it->second.Deal();
+	}
+	/* オリジナルからの変化の調査 :
+	** DiffOriginal で変化を文字列で取り出し、PatchOriginal で
+	** 変化を反映
+	*/
+	void DiffOriginal(string& ret_str) {
+		mapiterator it = data.begin();
+		for (; it != data.end(); it++) {
+			int len = it->second.DiffOriginalLen();
+			if (len) {
+				ret_str += it->first.c_str();
+				ret_str += "=";
+				it->second.DiffOriginal(ret_str);
+				ret_str += ";";
+			}
+		}
+		ret_str += ";";
+		return;
+	}
+	const char* PatchOriginal(const char* diff_data) {
+		while(*diff_data != ';') {
+			char name[1024];
+			const char* data_start = strchr(diff_data, '=');
+			if (data_start == 0) break;
+			strncpy(name, diff_data, data_start-diff_data);
+			name[data_start-diff_data] = 0;
+			data_start++;
+			mapiterator it = data.find(name);
+			if (it != data.end()) {
+				diff_data = data_start;
+				it->second.PatchOriginal(diff_data);
+			}
+			diff_data = strchr(diff_data, ';');
+			if (diff_data) diff_data++;
+		}
+		if (*diff_data == ';') {
+			diff_data++;
+		} else {
+			fprintf(stderr,"AyusysConfigItem::PatchOriginal: invalid data %s\n",diff_data);
+		}
+		return diff_data;
+	}
+	void SetOriginal(void) {
+		mapiterator it = data.begin();
+		for (; it != data.end(); it++) {
+			it->second.SetOriginal();
+		}
+	}
+	void Dump(FILE* f) const {
+		const_mapiterator it = data.begin();
+		for (; it != data.end(); it++) {
+			fprintf(f, "name %s: ",it->first.c_str());
+			it->second.Dump(f);
+		}
+	}
+};
+// template AyuSysConfigItem<AyuSysConfigStringItem, char>;
+// template AyuSysConfigItem<AyuSysConfigIntlistItem, int>;
+
+/************************************************/
+/* ラッパ */
+struct AyuSysConfigString {
+	AyuSysConfigItem<AyuSysConfigStringItem,char> orig;
+	void Dump(FILE* f) const {
+		fprintf(f, "string config:\n");
+		orig.Dump(f);
+	}
+};
+struct AyuSysConfigIntlist {
+	AyuSysConfigItem<AyuSysConfigIntlistItem, int> orig;
+	void Dump(FILE* f) const {
+		fprintf(f, "integer array config:\n");
+		orig.Dump(f);
+	}
+};
+
+/************************************************/
+/* AyuSysConfig クラス */
+int AyuSysConfig::SearchParam(const char* name) const{
+	HashStr str(name);
+	if (str_config->orig.Get(1, str)) return 1; /* char* のパラメータ */
+	else if (int_config->orig.Get(1, str)) return 2; /* int のパラメータ */
+	/* XXX.015.XXX の類のキー名を XXX.000.XXX の形に規格化して再検索 */
+	char name_copy[1024];
+	strncpy(name_copy, name, 1000);
+	name_copy[1000] = 0;
+	char* s;
+	for (s=name_copy; s != 0; s = strchr(s,'.')) {
+		if (isdigit(s[1]) && isdigit(s[2]) && isdigit(s[3])) {
+			s[1] = '0'; s[2] = '0'; s[3] = '0';
+		}
+		s++;
+	}
+	HashStr str2(name_copy);
+	if (str_config->orig.Get(1, str2)) return 1; /* char* のパラメータ */
+	else if (int_config->orig.Get(1, str2)) return 2; /* int のパラメータ */
+	else return 0;
+}
+const char* AyuSysConfig::GetParaStr(const char* name) const{
+	HashStr str(name);
+	const char* ret = str_config->orig.Get(1,str);
+	if (ret == 0) {
+		// fprintf(stderr,"Cannot find config name '%s'\n",name);
+	}
+	return ret;
+}
+int AyuSysConfig::GetParam(const char* name, int deal, ...) const{
+	HashStr str(name);
+	va_list va; int i;
+	const int* vars = int_config->orig.Get(deal, str);
+	if (vars == 0) {
+		// fprintf(stderr,"Cannot find config name '%s'\n",name);
+		va_start(va, deal);
+		for (i=0; i<deal; i++) {
+			int* var = va_arg(va, int*);
+			if (var) *var = 0;
+		}
+		return -1;
+	} else {
+		va_start(va, deal);
+		for (i=0; i<deal; i++) {
+			int* var = va_arg(va, int*);
+			if (var) *var = vars[i];
+		}
+	}
+	return 0;
+}
+int AyuSysConfig::GetOriginalParam(const char* name, int deal, ...) const{
+	HashStr str(name);
+	va_list va; int i;
+	const int* vars = int_config->orig.GetOriginal(deal, str);
+	if (vars == 0) {
+		// fprintf(stderr,"Cannot find config name '%s'\n",name);
+		va_start(va, deal);
+		for (i=0; i<deal; i++) {
+			int* var = va_arg(va, int*);
+			if (var) *var = 0;
+		}
+		return -1;
+	} else {
+		va_start(va, deal);
+		for (i=0; i<deal; i++) {
+			int* var = va_arg(va, int*);
+			if (var) *var = vars[i];
+		}
+	}
+	return 0;
+}
+const int* AyuSysConfig::GetParamArray(const char* name, int& deal) const{
+	HashStr str(name);
+	if (int_config->orig.Deal(str) == 0) { deal = 0; return 0; }
+	deal = int_config->orig.Deal(str);
+	const int* vars = int_config->orig.Get(deal, str);
+	if (vars == 0) { deal = 0; return 0; }
+	return vars;
+}
+void AyuSysConfig::SetParaStr(const char* name, const char* var) {
+	HashStr str(name);
+	dirty_flag = 1; change_flag = 1;
+	str_config->orig.Set(str, 1, var);
+}
+void AyuSysConfig::SetParam(const char* name, int deal, ...) {
+	if (deal >= MAXVARS) return ;
+	HashStr str(name);
+	int vars[MAXVARS]; va_list va; int i;
+	va_start(va, deal);
+	for (i=0; i<deal; i++) vars[i] = va_arg(va, int);
+	int_config->orig.Set(str, deal, vars);
+	dirty_flag = 1; change_flag = 1;
+	return;
+}
+void AyuSysConfig::SetOrigParaStr(const char* name, const char* var) {
+	HashStr str(name);
+	str_config->orig.SetOrig(str, 1, var);
+	change_flag = 1;
+}
+void AyuSysConfig::SetOrigParam(const char* name, int deal, ...) {
+	if (deal >= MAXVARS) return;
+	HashStr str(name);
+	int vars[MAXVARS]; va_list va; int i;
+	va_start(va, deal);
+	for(i=0; i<deal; i++) vars[i] = va_arg(va, int);
+	int_config->orig.SetOrig(str, deal, vars);
+	change_flag = 1;
+}
+void AyuSysConfig::SetOrigParamArray(const char* name, int deal, int* array) {
+	HashStr str(name);
+	int_config->orig.SetOrig(str, deal, array);
+}
+void AyuSysConfig::SetOriginal(void) {
+	/* 全ての設定を元に戻す */
+	str_config->orig.SetOriginal();
+	int_config->orig.SetOriginal();
+	change_flag = 1;
+}
+void AyuSysConfig::DiffOriginal(string& data) {
+	str_config->orig.DiffOriginal(data);
+	int_config->orig.DiffOriginal(data);
+	return;
+}
+const char* AyuSysConfig::PatchOriginal(const char* data) {
+	data = str_config->orig.PatchOriginal(data);
+	data = int_config->orig.PatchOriginal(data);
+	return data;
+}
+
+void AyuSysConfig::Dump(FILE* f) const {
+	str_config->Dump(f);
+	int_config->Dump(f);
+}
+
+/************************************************
+** AyuSysConfig のコンストラクタ:
+** 全ての config 項目を初期化する
+*/
+AyuSysConfig::AyuSysConfig(void) {
+	int i;
+
+	change_flag = 1; dirty_flag = 0;
+	str_config = new AyuSysConfigString;
+	int_config = new AyuSysConfigIntlist;
+
+	/****** 文字列 *******/
+	SetOrigParaStr("#WAKUPDT", "GRDAT");	/* 枠、マウスカーソルなどの画像ファイル */
+	SetOrigParaStr("#REGNAME", "xclannad");	/* レジストリ名。セーブファイルの作成に使う */
+	SetOrigParaStr("#CAPTION", "xclannad");	/* ウィンドウのタイトル */
+	SetOrigParaStr("#SAVENAME","SAVE.INI");	/* セーブファイルの名前 */
+	SetOrigParaStr("#SAVETITLE", "This is save file"); /* セーブファイルの先頭の文字列 */
+	SetOrigParaStr("#SAVENOTITLE", "-----------------"); /* 使われてないセーブデータの名前 */
+	SetOrigParaStr("#CGM_FILE", "MODE.CGM");/* CG mode の設定が保存されたファイル名 */
+	SetOrigParaStr("#CGTABLE_FILE", "MODE.CGM");/* CG mode の設定が保存されたファイル名 */
+
+	SetOrigParaStr("#WAKU.000.000.NAME", ""); // テキストウィンドウの窓飾り画像名
+	SetOrigParaStr("#WAKU.000.000.BACK", ""); // テキストウィンドウのテキスト背景画像名
+	SetOrigParaStr("#WAKU.000.000.BTN", ""); // テキストウィンドウのボタン画像名
+
+	SetOrigParaStr("#MOUSE_CURSOR.000.NAME", ""); // マウスカーソルのファイル名
+	SetOrigParaStr("#CURSOR.000.NAME", ""); // リターンカーソルのファイル名
+	SetOrigParaStr("#SELBTN.000.NAME", ""); // 選択肢背景
+	SetOrigParaStr("#SELBTN.000.BACK", ""); // 選択肢背景
+
+	char name_str[8] = "#NAME.A";
+	for (i='A'; i<='Z'; i++) {
+		name_str[6] = i;
+		SetOrigParaStr(name_str, "");
+	}
+
+	/****** 数値列 *******/
+	SetOrigParam("#CANCELCALL", 2, 0,0); /* キャンセルボタン(右クリック)したときに呼び出されるサブルーチン番号(メニュー) */
+	SetOrigParam("#COM2_TITLE", 1, 1); /* ? */
+	SetOrigParam("#COM2_TITLE_COLOR", 1, 2); /* 選択肢タイトルの色 */
+	SetOrigParam("#COM2_TITLE_INDENT", 1, 2); /* ? */
+	SetOrigParam("#SAVEFILETIME", 1, 24); /* セーブする場所の数 */
+	SetOrigParam("#SEEN_START", 1, 0); /* ゲームを開始するシナリオ番号 */
+	SetOrigParam("#SEEN_SRT", 1, 0); /* ゲームを開始するシナリオ番号(好き好き大好き) */
+	SetOrigParam("#SEEN_MENU",  1, 0); /* メニューのシナリオ番号 */
+	SetOrigParam("#SEEN_TEXT_CURRENT",  1, 0); /* seen.txt を root directory に置くか */
+	SetOrigParam("#FADE_TIME", 1, 40); /* 画面のフェード・アウトの速度 */
+	SetOrigParam("#NVL_SYSTEM",1, 0); /* テキストウィンドウが全画面か否か */
+	SetOrigParam("#WINDOW_ATTR", 5, -1, 128,128, 190, 0); /* テキストウィンドウの色 */
+	SetOrigParam("#WINDOW_ATTR_AREA", 4, 4,4,4,4); /* テキストウィンドウの範囲 */
+	SetOrigParam("#WINDOW_ATTR_TYPE", 1, 0); /* テキストウィンドウを半透明にするか */
+	SetOrigParam("#WINDOW_MSG_POS", 2, 22, 350); /* テキストウィンドウの位置 */
+	SetOrigParam("#WINDOW_COM_POS", 2,450, 250); /* 選択ウィンドウの位置 */
+	SetOrigParam("#WINDOW_GRP_POS", 2, 16, 100); /* なにかのウィンドウの位置 */
+	SetOrigParam("#WINDOW_SUB_POS", 2, 48, 100); /* なにかのウィンドウの位置 */
+	SetOrigParam("#WINDOW_SYS_POS", 2, 32, 100); /* なにかのウィンドウの位置 */
+	SetOrigParam("#WINDOW_WAKU_TYPE", 1, 0); /* テキストウィンドウの枠の種類。xkanon 独自設定 */
+	SetOrigParam("#RETN_CONT", 1, 16); /* リターンカーソルの数 */
+	SetOrigParam("#RETN_SPEED",1,100); /* リターンカーソルの動く速度 */
+	SetOrigParam("#RETN_XSIZE", 1, 16); /* リターンカーソルの大きさ */
+	SetOrigParam("#RETN_YSIZE", 1, 16); /* リターンカーソルの大きさ */
+	SetOrigParam("#FONT_SIZE", 1, 26); /* フォントの大きさ */
+	SetOrigParam("#FONT_WEIGHT", 1, 100); /* フォントの weight */
+	SetOrigParam("#MSG_MOJI_SIZE", 2, 12, 29); /* 文字の大きさ(半角) */
+	SetOrigParam("#MESSAGE_SIZE", 2, 23, 3); /* メッセージウィンドウの文字数 */
+	SetOrigParam("#COM_MESSAGE_SIZE", 2, 23, 3); /* メッセージウィンドウの文字数 */
+	SetOrigParam("#INIT_MESSAGE_SPEED", 1, 30); /* テキスト表示速度 */
+	SetOrigParam("#INIT_MESSAGE_SPEED_MOD", 1, 0); /* テキスト表示 no wait */
+	SetOrigParam("#MESSAGE_KEY_WAIT_USE", 1, 0); /* テキスト進行オートモード */
+	SetOrigParam("#MESSAGE_KEY_WAIT_TIME", 1, 1500); /* オートモードでのキー待ち時間 */
+
+	SetOrigParam("#GRP_DC_TIMES", 1, 4); /* 裏画面の数 */
+	SetOrigParam("#MUSIC_LINEAR_PAC",1,0); /* PCM データの 8bit -> 16bit 変換を行うか */
+	SetOrigParam("#MUSIC_TYPE",1,0); /* PCM データの種類 */
+	SetOrigParam("#WINDOW_MSGBK_BOX",1,0); /* バックログ用のボタン */
+	SetOrigParam("#WINDOW_MSGBK_LBOX_POS",4,15,7,8,0); /* バックログ用のボタン(左)の位置 */
+	SetOrigParam("#WINDOW_MSGBK_RBOX_POS",4,7,7,0,0); /* バックログ用のボタン(左)の位置 */
+	SetOrigParam("#MSGBK_MOD",1,0); /* バックログ用のボタンを使用するか */
+
+	SetOrigParam("#WAKU.000.000.TYPE", 1, 5);
+	SetOrigParam("#WAKU.000.000.MOVE_BOX", 5, 0, 0, 0, 0, 0); // テキストウィンドウの移動用ボタン位置
+	SetOrigParam("#WAKU.000.000.CLEAR_BOX", 5, 0, 0, 0, 0, 0); //    一時消去用ボタン位置
+	SetOrigParam("#WAKU.000.000.READJUMP_BOX", 5, 0, 0, 0, 0, 0); // スキップ用ボタン位置
+	SetOrigParam("#WAKU.000.000.AUTOMODE_BOX", 5, 0, 0, 0, 0, 0); // オート用ボタン位置
+	SetOrigParam("#WAKU.000.000.MSGBK_BOX", 5, 0, 0, 0, 0, 0); // バックログボタン位置
+	SetOrigParam("#WAKU.000.000.MSGBKLEFT_BOX", 5, 0, 0, 0, 0, 0); // バックログ(進める)ボタン位置
+	SetOrigParam("#WAKU.000.000.MSGBKRIGHT_BOX", 5, 0, 0, 0, 0, 0); // バックログ(戻る)ボタン位置
+	SetOrigParam("#WAKU.000.000.EXBTN_000_BOX", 5, 0, 0, 0, 0, 0); // その他ボタン0位置
+	SetOrigParam("#WAKU.000.000.EXBTN_001_BOX", 5, 0, 0, 0, 0, 0); // その他ボタン1位置
+	SetOrigParam("#WAKU.000.000.EXBTN_002_BOX", 5, 0, 0, 0, 0, 0); // その他ボタン2位置
+
+	SetOrigParam("#WINDOW.000.MOJI_SIZE", 1, 21); // 文字サイズ
+	SetOrigParam("#WINDOW.000.MOJI_REP", 2, -1, 2); // 文字の余裕
+	SetOrigParam("#WINDOW.000.MOJI_CNT", 2, 20, 3); // ウィンドウ内の文字数
+	SetOrigParam("#WINDOW.000.MOJI_POS", 4, 100, 0, 180, 40); // テキスト位置。3つ目がx で1つ目がyらしい
+	SetOrigParam("#WINDOW.000.MOJI_SHADOW", 1, 0); // 文字に影を付けるか
+	SetOrigParam("#WINDOW.000.LUBY_SIZE", 1, 8); // ルビの文字サイズ
+	SetOrigParam("#WINDOW.000.MOJI_MIN", 2, 8, 1); // 文字同士の隙間?
+	SetOrigParam("#WINDOW.000.SELCOM_USE", 1, 0); // 選択肢の実装方法
+	SetOrigParam("#WINDOW.000.POS", 4, 100, 0, 0, 260); // ウィンドウ位置
+	SetOrigParam("#WINDOW.000.ATTR_MOD", 1, 0); // ウィンドウ色
+	SetOrigParam("#WINDOW.000.ATTR", 5, -1, -1, -1, -1, -1); // ウィンドウ色
+	/* SELCOM はよくわからんので無視 */
+	SetOrigParam("#WINDOW.000.OPEN_ANM_MOD", 1, 0); // ウィンドウを開くときの効果らしい
+	SetOrigParam("#WINDOW.000.OPEN_ANM_TIME", 1, 500);
+	SetOrigParam("#WINDOW.000.CLOSE_ANM_MOD", 1, 0); // ウィンドウを閉じるときの効果らしい
+	SetOrigParam("#WINDOW.000.CLOSE_ANM_TIME", 1, 500);
+	SetOrigParam("#WINDOW.000.WAKU_SETNO", 1, 0); // 枠の種類
+	SetOrigParam("#WINDOW.000.MOVE_USE", 1, 0); // ウィンドウ枠移動ボタン使用の可否
+	SetOrigParam("#WINDOW.000.CLEAR_USE", 1, 0); // ウィンドウ枠消去ボタン使用の可否
+	SetOrigParam("#WINDOW.000.READJUMP_USE", 1, 0); // スキップボタン使用の可否
+	SetOrigParam("#WINDOW.000.AUTOMODE_USE", 1, 0); // スキップボタン使用の可否
+	SetOrigParam("#WINDOW.000.MSGBK_USE", 1, 0); // バックログボタン使用の可否
+	SetOrigParam("#WINDOW.000.MSGBKLEFT_USE", 1, 0); // バックログ(進む)ボタン使用の可否
+	SetOrigParam("#WINDOW.000.MSGBKRIGHT_USE", 1, 0); // バックログ(戻る)ボタン使用の可否
+	SetOrigParam("#WINDOW.000.EXBTN_000_USE", 1, 0); // その他ボタン0使用の可否
+	SetOrigParam("#WINDOW.000.EXBTN_001_USE", 1, 0); // その他ボタン1使用の可否
+	SetOrigParam("#WINDOW.000.EXBTN_002_USE", 1, 0); // その他ボタン2使用の可否
+	SetOrigParam("#WINDOW.000.NAME_MOD", 1, 0); // 名前ウィンドウを別途使用するか
+	SetOrigParam("#WINDOW.000.NAME_MOJI_SIZE", 1, 20); // 名前フォントのサイズ
+	SetOrigParam("#WINDOW.000.NAME_MOJI_POS", 2, 0, 0); // 名前ウィンドウの文字の位置
+	SetOrigParam("#WINDOW.000.NAME_MOJI_MIN", 1, 0); // 名前ウィンドウの幅
+	SetOrigParam("#WINDOW.000.NAME_CENTERING", 1, 1); // 名前のセンタリングの有無
+	SetOrigParam("#WINDOW.000.NAME_POS", 2, 159, 78); // 名前ウィンドウ位置(左下位置らしい)
+	SetOrigParam("#WINDOW.000.NAME_WAKU_SETNO", 1, -1); // 名前ウィンドウ位置(左下位置らしい)
+	SetOrigParam("#WINDOW.000.FACE.000", 5, 0, 0, 1, 1, 1); // 顔ウィンドウ位置(始め2つがx,y、MOJI_POSからの相対位置なのに注意)
+	SetOrigParam("#WINDOW.000.KEYCUR_MOD", 3, 0, 0, 0); // リターンカーソルの位置
+
+
+	SetOrigParam("#CURSOR.000.SIZE", 2, 0, 0); // リターンカーソルの大きさ
+	SetOrigParam("#CURSOR.000.CONT", 1, 50); // リターンカーソルの繰り返し数
+	SetOrigParam("#CURSOR.000.SPEED", 1, 1000); // ブリンクする速さ
+
+	SetOrigParam("#SELBTN.000.BASEPOS", 2, 0, 0); // 選択肢ウィンドウの位置
+	SetOrigParam("#SELBTN.000.REPPOS", 2, 0, 50); // 選択肢ウィンドウの次の位置(相対)
+	SetOrigParam("#SELBTN.000.MOJISIZE", 4, 26, 0,0,0); // 文字の大きさ
+	SetOrigParam("#SELBTN.000.MOJIDEFAULTCOL", 1, 0); // 非選択時の文字色
+	SetOrigParam("#SELBTN.000.MOJISELECTCOL", 1, 0); // 選択時の文字色
+
+	SetOrigParam("#COLOR_TABLE.000", 3, 255,255,255);
+	SetOrigParam("#SHAKE.000", 3, 0,0,0);
+
+	SetOrigParam("#SELR.000",16,0,0,640,480,0,0,500,50,0,0,0,0,0,0,255,0);
+	SetOrigParam("#SEL.000", 15,0,0,639,479,0,0, 32, 4,0,0,0,0,0,0,0); 
+
+	SetOrigParam("#SCREENSIZE_MOD", 1, 0); /* 0 = 640x480; 1 = 800x600 */
+}
+
+static int SplitVar(const char* str, int* ret_var, int ret_size) {
+	/* , あるいは ),:( をセパレータとして、-?[0-9]+ の
+	** フォーマットの数値列を読み込む。先頭に (、末尾に ) が付きうる。
+	** (),-[0-9] 以外の文字があったらそこで終了
+	** 得られたデータ数を返す
+	*/
+	if (*str == '(') str++;
+	int i; for (i=0; i<ret_size; i++) {
+		int c; int is_positive = 1;
+		/* セパレータの読み飛ばし */
+		c = *str;
+		if (c == ',' || c == ':') {
+			str++;
+		} else if (c == ')' && str[1] == '(') {
+			str += 2;
+		}
+		/* - を parse */
+		c = *str;
+		if (c == '-' && isdigit(str[1])) {
+			is_positive = -1; str++;
+		} else if (! isdigit(c)) {
+			return i; /* 異常な文字を見つけた:終了 */
+		}
+		int number = 0;
+		/* 数字読み込み */
+		while(isdigit( (c=*str) )) {
+			number *= 10;
+			number += c-'0';
+			str++;
+		}
+		ret_var[i] = is_positive * number;
+	}
+	return i;
+}
+/* 決められた数の引数を得る。-1 ならエラーが生じた */
+static inline int SplitVar(const char* str, int& var1) {
+	if (SplitVar(str, &var1, 1) != 1) return -1;
+	return 0;
+}
+static inline int SplitVar(const char* str, int& var1, int& var2) {
+	int vars[2];
+	if (SplitVar(str, vars, 2) != 2) return -1;
+	var1 = vars[0]; var2 = vars[1];
+	return 0;
+}
+static inline int SplitVar(const char* str, int& var1, int& var2, int& var3) {
+	int vars[3];
+	if (SplitVar(str, vars, 3) != 3) return -1;
+	var1 = vars[0]; var2 = vars[1]; var3 = vars[2];
+	return 0;
+}
+static inline int SplitVar(const char* str, int& var1, int& var2, int& var3, int& var4) {
+	int vars[4];
+	if (SplitVar(str, vars, 4) != 4) return -1;
+	var1 = vars[0]; var2 = vars[1]; var3 = vars[2]; var4 = vars[3];
+	return 0;
+}
+
+bool AyuSysConfig::LoadInitFile(void)
+{
+	char buf[1024]; int i;
+	char* tokens[MAXTOKEN]; int token_deal; int buf_ptr;
+	int numbers[MAXVARS];
+
+	ARCINFO* info = file_searcher.Find(FILESEARCH::ROOT, "gameexe.ini");
+	if (info == NULL) return false;
+	int size = info->Size();
+	unsigned char* buf_orig = (unsigned char*)info->Read();
+	if (size <= 0 || buf_orig == 0) {
+		delete info; return false;
+	}
+	unsigned char* buf_end = buf_orig + size;
+	int line_count = 0;
+	while(buf_orig < buf_end) {
+		/* buf_orig から一行読み込む */
+		/* その際に、
+		** ・頭が # 以外なら次の行までとばす
+		** ・"" 外のスペース、TABを初めとする制御文字 (0x20 以下のASCIIコード)を削除
+		** ・= で区切る。区切りは最大で10個で、tokens に代入される
+		** などの操作を行う
+		*/
+
+		/* # チェック */
+		if (*buf_orig != '#') {
+			/* 次の '\n' まで読み飛ばし */
+			while(buf_orig < buf_end &&
+				*buf_orig != '\n' && *buf_orig != '\r') buf_orig++;
+			if (buf_orig < buf_end-1 && *buf_orig == '\r' && buf_orig[1] == '\n') buf_orig += 2;
+			else if (*buf_orig == '\r' || *buf_orig == '\n') buf_orig++;
+			line_count++;
+			continue;
+		}
+		/* 初期化 */
+		token_deal = 1; tokens[0] = buf; buf_ptr = 0;
+		int in_quote = 0;
+
+		while(buf_orig < buf_end && buf_ptr < 1023) {
+			if (in_quote) {
+				/* "" の中 */
+				int c = *buf_orig;
+				if (c == '\n' || c == '\r') {
+					break;
+				} else if (c == '\"') {
+					in_quote = 0;
+				} else {
+					buf[buf_ptr++] = c;
+				}
+				buf_orig++;
+			} else { /* quote されてない */
+				/* 制御文字を読み飛ばす */
+				while(*buf_orig <= 0x20 && buf_orig < buf_end &&
+					*buf_orig != '\n' && *buf_orig != '\r') buf_orig++;
+				int c = *buf_orig;
+				if (c == '\n' || c == '\r') break;
+				/* = なら次の token */
+				if (c == '=') {
+					c = 0; tokens[token_deal++] = buf+buf_ptr+1;
+					if (token_deal >= MAXTOKEN) break;
+				} else if (c == '\"') {
+					in_quote = 1; buf_orig++; continue;
+				}
+				buf[buf_ptr++] = c;
+				buf_orig++;
+			}
+		}
+		buf[buf_ptr] = '\0';
+		/* 末尾の \r\n を消去 */
+		if (buf_orig < buf_end-1 && buf_orig[0] == '\r' && buf_orig[1] == '\n') buf_orig += 2;
+		else if (buf_orig < buf_end && (buf_orig[0] == '\r' || buf_orig[0] == '\n')) buf_orig++;
+		/* 必要なら parse 内容を出力 */
+		dprintf(("line %3d ",line_count));
+		for (i=0; i<token_deal; i++) {
+			dprintf(("%d:\"%s\", ",i,tokens[i]));
+		}
+		dprintf(("\n"));
+		if (in_quote) {
+			fprintf(stderr, "Warning : open quote is found while parsing gameexe.ini, line %d\n",line_count);
+		}
+
+
+		/* 得られた内容を parse */
+
+		/* #NAME=<文字列> */
+		int type = SearchParam(tokens[0]);
+		if (type == 1) { /* #NAME=<文字列> */
+			if (token_deal != 2) {
+				dprintf(("Parse error, line %d, %s\n",line_count, tokens[0]));
+				goto parse_error;
+			}
+			SetOrigParaStr(tokens[0], tokens[1]);
+			goto parse_end;
+		} else if (type == 2) { /* #NAME=<数値列> */
+			if (token_deal != 2) {
+				dprintf(("Parse error, line %d, %s\n",line_count, tokens[0]));
+				goto parse_error;
+			}
+			int number_deal = SplitVar(tokens[1], numbers, MAXVARS);
+			SetOrigParamArray(tokens[0], number_deal, numbers);
+			goto parse_end;
+		}
+		/* 一般的な設定以外 : cdrom track など */
+		if (strncmp(tokens[0],"#NAME.", 6) == 0) {
+			if (token_deal != 2) goto parse_error;
+			SetOrigParaStr(tokens[0], tokens[1]);
+			goto parse_end;
+		} else if (strncmp(tokens[0],"#DIRC.",6) == 0) {
+			if (token_deal != 3) goto parse_error;
+			/* ファイル形式の指定 */
+			FILESEARCH::FILETYPE type;
+			char* name = tokens[0]+6;
+			if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN;
+			else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM; 
+			else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD;
+			else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR;
+			else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV;
+			else if (strcmp(name, "KOE") == 0) type = FILESEARCH::KOE;
+			else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN;
+			else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */
+			if (tokens[2][0] == 'N') { /* directory */
+				file_searcher.SetFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]);
+				dprintf(("set file directory; type %s, directory %s\n",name,tokens[1]));
+			} else if (tokens[2][0] == 'P' && tokens[2][1] == ':') { /* アーカイブ */
+				file_searcher.SetFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2);
+				dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2));
+			} else goto parse_error;
+			goto parse_end;
+		}
+		if (strncmp(tokens[0],"#ADRC.",6) == 0) {
+			if (token_deal != 3) goto parse_error;
+			/* ファイル形式の指定 */
+			FILESEARCH::FILETYPE type;
+			char* name = tokens[0]+6;
+			if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN;
+			else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM; 
+			else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD;
+			else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR;
+			else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV;
+			else if (strcmp(name, "KOE") == 0) type = FILESEARCH::KOE;
+			else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN;
+			else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */
+			if (tokens[2][0] == 'N') { /* directory */
+				file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]);
+				dprintf(("set file directory; type %s, directory %s\n",name,tokens[1]));
+			} else if (tokens[2][0] == 'P' && tokens[2][1] == ':') { /* アーカイブ */
+				file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2);
+				dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2));
+			} else if (tokens[2][0] == 'R' && tokens[2][1] == ':') { /* それ散るアーカイブ */
+				file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2);
+				dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2));
+			} else goto parse_error;
+			goto parse_end;
+		}
+		if (strncmp(tokens[0],"#FOLDNAME.",10) == 0) {
+			if (token_deal != 3) goto parse_error;
+			/* ファイル形式の指定 */
+			FILESEARCH::FILETYPE type;
+			char* name = tokens[0]+10;
+			if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT;
+			else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN;
+			else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM; 
+			else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD;
+			else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR;
+			else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV;
+			else if (strcmp(name, "BGM") == 0) type = FILESEARCH::BGM;
+			else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN;
+			else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */
+			if (tokens[2][0] == '0') { /* directory */
+				file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]);
+				dprintf(("set file directory; type %s, directory %s\n",name,tokens[1]));
+			} else if (tokens[2][0] == '1' && tokens[2][1] == ':') { /* アーカイブ */
+				file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_SCN2k, tokens[2]+2);
+				dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2));
+			} else goto parse_error;
+			goto parse_end;
+		}
+		if (strcmp(tokens[0], "#CDTRACK") == 0) {
+			if (token_deal != 3) goto parse_error;
+			track_name.AddCDROM(tokens[2], atoi(tokens[1]));
+			dprintf(("Set CDTRACK, name %s, track %d\n",tokens[2], atoi(tokens[1])));
+			goto parse_end;
+		}
+		if (strcmp(tokens[0], "#DSTRACK") == 0) {
+			/* #DSTRACK=00000000-99999000-00782556="filename"   ="name"       */
+			/* #DSTRACK=00000000-99999000-00782556="name"       */
+			/* 第二トークンの3つめのパラメータを得る(繰り返しの時の再生開始位置) */
+			int start_pt = 0;
+			const char* tk1 = strchr(tokens[1], '-');
+			const char* tk2 = 0;
+			if (tk1 && *tk1) tk2 = strchr(tk1+1, '-');
+			if (tk2 && *tk2) start_pt = atoi(tk2+1);
+			if (token_deal == 3) {
+				track_name.AddWave(tokens[2], tokens[2], start_pt);
+				dprintf(("Set Wave track, name %s\n",tokens[2]));
+			} else if (token_deal == 4) {
+				track_name.AddWave(tokens[3], tokens[2], start_pt);
+				dprintf(("Set Wave track, name %s, file %s\n",tokens[3], tokens[2]));
+			} else goto parse_error;
+			goto parse_end;
+		}
+		if (strncmp(tokens[0], "#SE.", 4) == 0) {
+			/* SE.XXX="XXX"=X */
+			if (token_deal == 2) {
+				track_name.AddSE(atoi(tokens[0]+4), tokens[1]);
+			} else if (token_deal == 3) {
+				if (atoi(tokens[2]) != 0) {
+					track_name.AddSE(atoi(tokens[0]+4), tokens[1]);
+				}
+			}
+			dprintf(("Set SE %d, name %s\n",atoi(tokens[0]+4), tokens[1]));
+			goto parse_end;
+		}
+		/* 設定項目が見つからなかった */
+		dprintf(("Cannot find configuration name: %s\n",tokens[0]));
+	parse_error:
+	parse_end:
+		line_count++;
+	}
+	delete info;
+	/* デフォルトのオプションを指定する */
+	// set_game(GetParaStr("#REGNAME"), *this);
+	return true;
+}
+
+TrackName::TrackName(void) {
+	deal = 1;
+	track = new char*[deal];
+	track_wave = new char*[deal];
+	track_num = new int[deal];
+	track_start = new int[deal];
+	int i; for (i=0; i<deal; i++) track[i] = 0;
+	for (i=0; i<deal; i++) track_wave[i] = 0;
+	se_deal = 10;
+	se_track = new char*[se_deal];
+	for (i=0; i<se_deal; i++) se_track[i] = 0;
+}
+
+TrackName::~TrackName() {
+	int i; for (i=0; i<deal; i++) {
+		if (track[i] != 0) delete[] track[i];
+		if (track_wave[i] != 0) delete[] track_wave[i];
+	}
+	for (i=0; i<se_deal; i++) {
+		if (se_track[i]) delete[] se_track[i];
+	}
+	delete[] track;
+	delete[] track_wave;
+	delete[] track_num;
+	delete[] track_start;
+	delete[] se_track;
+}
+void TrackName::Expand(void) {
+	int new_deal = deal * 2;
+	int* new_track_num = new int[new_deal];
+	int* new_track_start = new int[new_deal];
+	char** new_track = new char*[new_deal];
+	char** new_track_wave = new char*[new_deal];
+	int i; for (i=0; i<deal; i++) {
+		new_track_num[i] = track_num[i];
+		new_track_start[i] = track_start[i];
+		new_track[i] = track[i];
+		new_track_wave[i] = track_wave[i];
+	}
+	for (; i<new_deal; i++) { 
+		new_track_num[i] = 0;
+		new_track_start[i] = 0;
+		new_track[i] = 0;
+		new_track_wave[i] = 0;
+	}
+	deal = new_deal;
+	delete[] track; track = new_track;
+	delete[] track_num; track_num= new_track_num;
+	delete[] track_start; track_start= new_track_start;
+	delete[] track_wave; track_wave = new_track_wave;
+}
+void TrackName::ExpandSE(int n) {
+	if (n < 0) return;
+	n += 10;
+	if (se_deal >= n) return;
+	char** new_se = new char*[n];
+	int i; for (i=0; i<se_deal; i++) new_se[i] = se_track[i];
+	for (; i<n; i++) new_se[i] = 0;
+	delete[] se_track;
+	se_deal = n; se_track = new_se;
+}
+void TrackName::AddCDROM(char* name, int tk) {
+	if (CDTrack(name) != -1) return;
+	int i; for (i=0; i<deal; i++) {
+		if (track[i] == 0) break;
+	}
+	int num = i;
+	if (i == deal) Expand();
+	track[num] = new char[strlen(name)+1];
+	for (i=0; name[i] != 0; i++) track[num][i] = tolower(name[i]);
+	track[num][i] = 0;
+	track_num[num] = tk;
+}
+void TrackName::AddWave(char* name, char* file, int pt) {
+	if (CDTrack(name) != -1) return;
+	int i; for (i=0; i<deal; i++) {
+		if (track[i] == 0) break;
+	}
+	int num = i;
+	if (i == deal) Expand();
+	track_num[num] = 0;
+	track_start[num] = pt;
+	track[num] = new char[strlen(name)+1];
+	for (i=0; name[i] != 0; i++) track[num][i] = tolower(name[i]);
+	track[num][i] = 0;
+	track_wave[num] = new char[strlen(file)+1]; strcpy(track_wave[num], file);
+}
+int TrackName::CDTrack(char* name) {
+	char buf[1024];
+	int i;
+	for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]);
+	buf[i]=0;
+	for (i=0; i<deal; i++) {
+		if (track[i] == 0) return -1;
+		if (strcmp(track[i],  buf) == 0) {
+			return track_num[i];
+		}
+	}
+	return -1;
+}
+int TrackName::TrackStart(char* name) {
+	char buf[1024];
+	int i;
+	for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]);
+	buf[i]=0;
+	for (i=0; i<deal; i++) {
+		if (track[i] == 0) return -1;
+		if (strcmp(track[i],  buf) == 0) {
+			return track_start[i];
+		}
+	}
+	return 0;
+}
+const char* TrackName::WaveTrack(char* name) {
+	char buf[1024];
+	int i;
+	for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]);
+	buf[i]=0;
+	for (i=0; i<deal; i++) {
+		if (track[i] == 0) return 0;
+		if (strcmp(track[i],  buf) == 0) {
+			return track_wave[i];
+		}
+	}
+	return 0;
+}
+const char* TrackName::SETrack(int n) {
+	if (n < 0 || n >= se_deal) return 0;
+	return se_track[n];
+}
+void TrackName::AddSE(int n, char* file) {
+	if (se_deal <= n) ExpandSE(n);
+	if (se_track[n]) delete[] se_track[n];
+	se_track[n] = new char[strlen(file)+1];
+	strcpy(se_track[n], file);
+}
+