* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Mayaqua / Table.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Table.c b/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Table.c
new file mode 100644 (file)
index 0000000..1735784
--- /dev/null
@@ -0,0 +1,865 @@
+// SoftEther UT-VPN SourceCode\r
+// \r
+// Copyright (C) 2004-2010 SoftEther Corporation.\r
+// Copyright (C) 2004-2010 University of Tsukuba, Japan.\r
+// Copyright (C) 2003-2010 Daiyuu Nobori.\r
+// All Rights Reserved.\r
+// \r
+// http://utvpn.tsukuba.ac.jp/\r
+// \r
+// This program is free software; you can redistribute it and/or\r
+// modify it under the terms of the GNU General Public License\r
+// version 2 as published by the Free Software Foundation.\r
+// \r
+// This program is distributed in the hope that it will be useful,\r
+// but WITHOUT ANY WARRANTY; without even the implied warranty of\r
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\r
+// GNU General Public License for more details.\r
+// \r
+// You should have received a copy of the GNU General Public License version 2\r
+// along with this program; if not, write to the Free Software\r
+// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.\r
+// \r
+// このファイルは GPL バージョン 2 ライセンスで公開されています。\r
+// 誰でもこのファイルの内容を複製、改変したり、改変したバージョンを再配布\r
+// することができます。ただし、原著作物を改変した場合は、原著作物の著作権表示\r
+// を除去することはできません。改変した著作物を配布する場合は、改変実施者の\r
+// 著作権表示を原著作物の著作権表示に付随して記載するようにしてください。\r
+// \r
+// この SoftEther UT-VPN オープンソース・プロジェクトは、日本国の\r
+// ソフトイーサ株式会社 (SoftEther Corporation, http://www.softether.co.jp/ )\r
+// および筑波大学 (University of Tsukuba, http://www.tsukuba.ac.jp/ ) によって\r
+// ホストされています。\r
+// 本プログラムの配布者は、本プログラムを、業としての利用以外のため、\r
+// および、試験または研究のために利用が行われることを想定して配布\r
+// しています。\r
+// SoftEther UT-VPN プロジェクトの Web サイトは http://utvpn.tsukuba.ac.jp/ に\r
+// あります。\r
+// 本ソフトウェアの不具合の修正、機能改良、セキュリティホールの修復などのコード\r
+// の改変を行った場合で、その成果物を SoftEther UT-VPN プロジェクトに提出して\r
+// いただける場合は、 http://utvpn.tsukuba.ac.jp/ までソースコードを送付して\r
+// ください。SoftEther UT-VPN プロジェクトの本体リリースまたはブランチリリース\r
+// に組み込みさせていただきます。\r
+// \r
+// GPL に基づいて原著作物が提供される本ソフトウェアの改良版を配布、販売する\r
+// 場合は、そのソースコードを GPL に基づいて誰にでも開示する義務が生じます。\r
+// \r
+// 本ソフトウェアに関連する著作権、特許権、商標権はソフトイーサ株式会社\r
+// (SoftEther Corporation) およびその他の著作権保持者が保有しています。\r
+// ソフトイーサ株式会社等はこれらの権利を放棄していません。本ソフトウェアの\r
+// 二次著作物を配布、販売する場合は、これらの権利を侵害しないようにご注意\r
+// ください。\r
+// \r
+// お願い: どのような通信ソフトウェアにも通常は必ず未発見の\r
+// セキュリティホールが潜んでいます。本ソースコードをご覧いただいた結果、\r
+// UT-VPN にセキュリティホールを発見された場合は、当該セキュリティホールの\r
+// 情報を不特定多数に開示される前に、必ず、ソフトイーサ株式会社\r
+// および脆弱性情報の届出を受け付ける公的機関まで通報いただき、\r
+// 公益保護にご協力いただきますようお願い申し上げます。\r
+// \r
+// ソフトイーサ株式会社は、当該セキュリティホールについて迅速に対処を\r
+// 行い、UT-VPN および UT-VPN に関連するソフトウェアのユーザー・顧客\r
+// を保護するための努力を行います。\r
+// \r
+// ソフトイーサへの届出先: http://www.softether.co.jp/jp/contact/\r
+// 日本国内の脆弱性情報届出受付公的機関:\r
+//         独立行政法人 情報処理推進機構\r
+//         http://www.ipa.go.jp/security/vuln/report/\r
+// \r
+// 上記各事項について不明な点は、ソフトイーサ株式会社までご連絡ください。\r
+// 連絡先: http://www.softether.co.jp/jp/contact/\r
+\r
+// -----------------------------------------------\r
+// [ChangeLog]\r
+// 2010.05.20\r
+//  新規リリース by SoftEther\r
+// -----------------------------------------------\r
+\r
+// Table.c\r
+// 文字列テーブル読み込み & 管理ルーチン\r
+\r
+#include <stdio.h>\r
+#include <stdlib.h>\r
+#include <string.h>\r
+#include <wchar.h>\r
+#include <stdarg.h>\r
+#include <time.h>\r
+#include <errno.h>\r
+#include <Mayaqua/Mayaqua.h>\r
+\r
+// TABLE のリスト\r
+static LIST *TableList = NULL;\r
+static wchar_t old_table_name[MAX_SIZE] = {0};         // 古いテーブル名\r
+\r
+// エラー文字列を Unicode で取得する\r
+wchar_t *GetUniErrorStr(UINT err)\r
+{\r
+       wchar_t *ret;\r
+       char name[MAX_SIZE];\r
+       Format(name, sizeof(name), "ERR_%u", err);\r
+\r
+       ret = GetTableUniStr(name);\r
+       if (UniStrLen(ret) != 0)\r
+       {\r
+               return ret;\r
+       }\r
+       else\r
+       {\r
+               return _UU("ERR_UNKNOWN");\r
+       }\r
+}\r
+\r
+// エラー文字列を取得する\r
+char *GetErrorStr(UINT err)\r
+{\r
+       char *ret;\r
+       char name[MAX_SIZE];\r
+       Format(name, sizeof(name), "ERR_%u", err);\r
+\r
+       ret = GetTableStr(name);\r
+       if (StrLen(ret) != 0)\r
+       {\r
+               return ret;\r
+       }\r
+       else\r
+       {\r
+               return _SS("ERR_UNKNOWN");\r
+       }\r
+}\r
+\r
+// テーブルから整数値をロードする\r
+UINT GetTableInt(char *name)\r
+{\r
+       char *str;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       str = GetTableStr(name);\r
+       return ToInt(str);\r
+}\r
+\r
+// テーブルから Unicode 文字列をロードする\r
+wchar_t *GetTableUniStr(char *name)\r
+{\r
+       TABLE *t;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+//             Debug("%s: ************\n", name);\r
+               return L"";\r
+       }\r
+\r
+       // 検索\r
+       t = FindTable(name);\r
+       if (t == NULL)\r
+       {\r
+//             Debug("%s: UNICODE STRING NOT FOUND\n", name);\r
+               return L"";\r
+       }\r
+\r
+       return t->unistr;\r
+}\r
+\r
+// テーブルから文字列をロードする\r
+char *GetTableStr(char *name)\r
+{\r
+       TABLE *t;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return "";\r
+       }\r
+\r
+#ifdef OS_WIN32\r
+       if (StrCmpi(name, "DEFAULT_FONT") == 0)\r
+       {\r
+               if (_II("LANG") == 2)\r
+               {\r
+                       UINT os_type = GetOsType();\r
+                       if (OS_IS_WINDOWS_9X(os_type) ||\r
+                               GET_KETA(os_type, 100) <= 4)\r
+                       {\r
+                               // Windows 9x, Windows NT 4.0, Windows 2000, Windows XP, Windows Server 2003 の場合は SimSun フォントを利用する\r
+                               return "SimSun";\r
+                       }\r
+               }\r
+       }\r
+#endif // OS_WIN32\r
+\r
+       // 検索\r
+       t = FindTable(name);\r
+       if (t == NULL)\r
+       {\r
+//             Debug("%s: ANSI STRING NOT FOUND\n", name);\r
+               return "";\r
+       }\r
+\r
+       return t->str;\r
+}\r
+\r
+// 指定した名前で始まる文字列名を取得する\r
+TOKEN_LIST *GetTableNameStartWith(char *str)\r
+{\r
+       UINT i;\r
+       UINT len;\r
+       LIST *o;\r
+       TOKEN_LIST *t;\r
+       char tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (str == NULL)\r
+       {\r
+               return NullToken();\r
+       }\r
+\r
+       StrCpy(tmp, sizeof(tmp), str);\r
+       StrUpper(tmp);\r
+\r
+       len = StrLen(tmp);\r
+\r
+       o = NewListFast(NULL);\r
+\r
+       for (i = 0;i < LIST_NUM(TableList);i++)\r
+       {\r
+               TABLE *t = LIST_DATA(TableList, i);\r
+               UINT len2 = StrLen(t->name);\r
+\r
+               if (len2 >= len)\r
+               {\r
+                       if (Cmp(t->name, tmp, len) == 0)\r
+                       {\r
+                               Insert(o, CopyStr(t->name));\r
+                       }\r
+               }\r
+       }\r
+\r
+       t = ZeroMalloc(sizeof(TOKEN_LIST));\r
+       t->NumTokens = LIST_NUM(o);\r
+       t->Token = ZeroMalloc(sizeof(char *) * t->NumTokens);\r
+\r
+       for (i = 0;i < t->NumTokens;i++)\r
+       {\r
+               t->Token[i] = LIST_DATA(o, i);\r
+       }\r
+\r
+       ReleaseList(o);\r
+\r
+       return t;\r
+}\r
+\r
+// テーブルを検索する\r
+TABLE *FindTable(char *name)\r
+{\r
+       TABLE *t, tt;\r
+       // 引数チェック\r
+       if (name == NULL || TableList == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       tt.name = CopyStr(name);\r
+       t = Search(TableList, &tt);\r
+       Free(tt.name);\r
+\r
+       return t;\r
+}\r
+\r
+// テーブル名を比較する関数\r
+int CmpTableName(void *p1, void *p2)\r
+{\r
+       TABLE *t1, *t2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       t1 = *(TABLE **)p1;\r
+       t2 = *(TABLE **)p2;\r
+       if (t1 == NULL || t2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return StrCmpi(t1->name, t2->name);\r
+}\r
+\r
+// 1 行を解釈する\r
+TABLE *ParseTableLine(char *line, char *prefix, UINT prefix_size)\r
+{\r
+       UINT i, len;\r
+       UINT len_name;\r
+       UINT string_start;\r
+       char *name;\r
+       char *name2;\r
+       UINT name2_size;\r
+       wchar_t *unistr;\r
+       char *str;\r
+       UINT unistr_size, str_size;\r
+       TABLE *t;\r
+       // 引数チェック\r
+       if (line == NULL || prefix == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       TrimLeft(line);\r
+\r
+       // 行無し\r
+       len = StrLen(line);\r
+       if (len == 0)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // コメント\r
+       if (line[0] == '#' || (line[0] == '/' && line[1] == '/'))\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 名前の終了位置まで検索\r
+       len_name = 0;\r
+       for (i = 0;;i++)\r
+       {\r
+               if (line[i] == 0)\r
+               {\r
+                       // トークンが 1 つしか無い\r
+                       return NULL;\r
+               }\r
+               if (line[i] == ' ' || line[i] == '\t')\r
+               {\r
+                       break;\r
+               }\r
+               len_name++;\r
+       }\r
+\r
+       name = Malloc(len_name + 1);\r
+       StrCpy(name, len_name + 1, line);\r
+\r
+       string_start = len_name;\r
+       for (i = len_name;i < len;i++)\r
+       {\r
+               if (line[i] != ' ' && line[i] != '\t')\r
+               {\r
+                       break;\r
+               }\r
+               string_start++;\r
+       }\r
+       if (i == len)\r
+       {\r
+               Free(name);\r
+               return NULL;\r
+       }\r
+\r
+       // アンエスケープ\r
+       UnescapeStr(&line[string_start]);\r
+\r
+       // Unicode に変換する\r
+       unistr_size = CalcUtf8ToUni(&line[string_start], StrLen(&line[string_start]));\r
+       if (unistr_size == 0)\r
+       {\r
+               Free(name);\r
+               return NULL;\r
+       }\r
+       unistr = Malloc(unistr_size);\r
+       Utf8ToUni(unistr, unistr_size, &line[string_start], StrLen(&line[string_start]));\r
+\r
+       // ANSI に変換する\r
+       str_size = CalcUniToStr(unistr);\r
+       if (str_size == 0)\r
+       {\r
+               str_size = 1;\r
+               str = Malloc(1);\r
+               str[0] = 0;\r
+       }\r
+       else\r
+       {\r
+               str = Malloc(str_size);\r
+               UniToStr(str, str_size, unistr);\r
+       }\r
+\r
+       if (StrCmpi(name, "PREFIX") == 0)\r
+       {\r
+               // プレフィックスが指定された\r
+               StrCpy(prefix, prefix_size, str);\r
+               Trim(prefix);\r
+\r
+               if (StrCmpi(prefix, "$") == 0 || StrCmpi(prefix, "NULL") == 0)\r
+               {\r
+                       prefix[0] = 0;\r
+               }\r
+\r
+               Free(name);\r
+               Free(str);\r
+               Free(unistr);\r
+\r
+               return NULL;\r
+       }\r
+\r
+       name2_size = StrLen(name) + StrLen(prefix) + 2;\r
+       name2 = ZeroMalloc(name2_size);\r
+\r
+       if (prefix[0] != 0)\r
+       {\r
+               StrCat(name2, name2_size, prefix);\r
+               StrCat(name2, name2_size, "@");\r
+       }\r
+\r
+       StrCat(name2, name2_size, name);\r
+\r
+       Free(name);\r
+\r
+       // TABLE を作成する\r
+       t = Malloc(sizeof(TABLE));\r
+       StrUpper(name2);\r
+       t->name = name2;\r
+       t->str = str;\r
+       t->unistr = unistr;\r
+\r
+       return t;\r
+}\r
+\r
+// 文字列のアンエスケープ\r
+void UnescapeStr(char *src)\r
+{\r
+       UINT i, len, wp;\r
+       char *tmp;\r
+       // 引数チェック\r
+       if (src == NULL)\r
+       {\r
+               return;\r
+       }\r
+       \r
+       len = StrLen(src);\r
+       tmp = Malloc(len + 1);\r
+       wp = 0;\r
+       for (i = 0;i < len;i++)\r
+       {\r
+               if (src[i] == '\\')\r
+               {\r
+                       i++;\r
+                       switch (src[i])\r
+                       {\r
+                       case 0:\r
+                               goto FINISH;\r
+                       case '\\':\r
+                               tmp[wp++] = '\\';\r
+                               break;\r
+                       case ' ':\r
+                               tmp[wp++] = ' ';\r
+                               break;\r
+                       case 'n':\r
+                       case 'N':\r
+                               tmp[wp++] = '\n';\r
+                               break;\r
+                       case 'r':\r
+                       case 'R':\r
+                               tmp[wp++] = '\r';\r
+                               break;\r
+                       case 't':\r
+                       case 'T':\r
+                               tmp[wp++] = '\t';\r
+                               break;\r
+                       }\r
+               }\r
+               else\r
+               {\r
+                       tmp[wp++] = src[i];\r
+               }\r
+       }\r
+FINISH:\r
+       tmp[wp++] = 0;\r
+       StrCpy(src, 0, tmp);\r
+       Free(tmp);\r
+}\r
+\r
+// テーブルを解放する\r
+void FreeTable()\r
+{\r
+       UINT i, num;\r
+       TABLE **tables;\r
+       if (TableList == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       TrackingDisable();\r
+\r
+       num = LIST_NUM(TableList);\r
+       tables = ToArray(TableList);\r
+       for (i = 0;i < num;i++)\r
+       {\r
+               TABLE *t = tables[i];\r
+               Free(t->name);\r
+               Free(t->str);\r
+               Free(t->unistr);\r
+               Free(t);\r
+       }\r
+       ReleaseList(TableList);\r
+       TableList = NULL;\r
+       Free(tables);\r
+\r
+       Zero(old_table_name, sizeof(old_table_name));\r
+\r
+       TrackingEnable();\r
+}\r
+\r
+// バッファから文字列テーブルを読み込む\r
+bool LoadTableFromBuf(BUF *b)\r
+{\r
+       char *tmp;\r
+       char prefix[MAX_SIZE];\r
+       // 引数チェック\r
+       if (b == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // すでにテーブルがある場合は削除する\r
+       FreeTable();\r
+\r
+       // リストを作成する\r
+       TableList = NewList(CmpTableName);\r
+\r
+       Zero(prefix, sizeof(prefix));\r
+\r
+       // バッファの内容を 1 行ずつ読んでいく\r
+       while (true)\r
+       {\r
+               TABLE *t;\r
+               tmp = CfgReadNextLine(b);\r
+               if (tmp == NULL)\r
+               {\r
+                       break;\r
+               }\r
+               t = ParseTableLine(tmp, prefix, sizeof(prefix));\r
+               if (t != NULL)\r
+               {\r
+                       // 登録する\r
+                       Insert(TableList, t);\r
+               }\r
+               Free(tmp);\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// Unicode 文字列キャッシュファイル名の生成\r
+void GenerateUnicodeCacheFileName(wchar_t *name, UINT size, wchar_t *strfilename, UINT strfilesize, UCHAR *filehash)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       wchar_t hashstr[64];\r
+       wchar_t hashtemp[MAX_SIZE];\r
+       wchar_t exe[MAX_SIZE];\r
+       UCHAR hash[SHA1_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL || strfilename == NULL || filehash == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       GetExeDirW(exe, sizeof(exe));\r
+       UniStrCpy(hashtemp, sizeof(hashtemp), strfilename);\r
+       BinToStrW(tmp, sizeof(tmp), filehash, MD5_SIZE);\r
+       UniStrCat(hashtemp, sizeof(hashtemp), tmp);\r
+       UniStrCat(hashtemp, sizeof(hashtemp), exe);\r
+       UniStrLower(hashtemp);\r
+\r
+       Hash(hash, hashtemp, UniStrLen(hashtemp) * sizeof(wchar_t), true);\r
+       BinToStrW(hashstr, sizeof(hashstr), hash, 4);\r
+       UniFormat(tmp, sizeof(tmp), UNICODE_CACHE_FILE, hashstr);\r
+       UniStrLower(tmp);\r
+\r
+#ifndef        OS_WIN32\r
+       UniStrCpy(exe, sizeof(exe), L"/tmp");\r
+#else  // OS_WIN32\r
+       StrToUni(exe, sizeof(exe), MsGetTempDir());\r
+#endif // OS_WIN32\r
+\r
+       UniFormat(name, size, L"%s/%s", exe, tmp);\r
+       NormalizePathW(name, size, name);\r
+}\r
+\r
+// Unicode キャッシュの保存\r
+void SaveUnicodeCache(wchar_t *strfilename, UINT strfilesize, UCHAR *hash)\r
+{\r
+       UNICODE_CACHE c;\r
+       BUF *b;\r
+       UINT i;\r
+       IO *io;\r
+       wchar_t name[MAX_PATH];\r
+       UCHAR binhash[MD5_SIZE];\r
+       // 引数チェック\r
+       if (strfilename == NULL || hash == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Zero(&c, sizeof(c));\r
+       UniToStr(c.StrFileName, sizeof(c.StrFileName), strfilename);\r
+       c.StrFileSize = strfilesize;\r
+       GetMachineName(c.MachineName, sizeof(c.MachineName));\r
+       c.OsType = GetOsInfo()->OsType;\r
+       Copy(c.hash, hash, MD5_SIZE);\r
+\r
+#ifdef OS_UNIX\r
+       GetCurrentCharSet(c.CharSet, sizeof(c.CharSet));\r
+#else  // OS_UNIX\r
+       {\r
+               UINT id = MsGetThreadLocale();\r
+               Copy(c.CharSet, &id, sizeof(id));\r
+       }\r
+#endif // OS_UNIX\r
+\r
+       b = NewBuf();\r
+       WriteBuf(b, &c, sizeof(c));\r
+\r
+       WriteBufInt(b, LIST_NUM(TableList));\r
+       for (i = 0;i < LIST_NUM(TableList);i++)\r
+       {\r
+               TABLE *t = LIST_DATA(TableList, i);\r
+               WriteBufInt(b, StrLen(t->name));\r
+               WriteBuf(b, t->name, StrLen(t->name));\r
+               WriteBufInt(b, StrLen(t->str));\r
+               WriteBuf(b, t->str, StrLen(t->str));\r
+               WriteBufInt(b, UniStrLen(t->unistr));\r
+               WriteBuf(b, t->unistr, UniStrLen(t->unistr) * sizeof(wchar_t));\r
+       }\r
+\r
+       Hash(binhash, b->Buf, b->Size, false);\r
+       WriteBuf(b, binhash, MD5_SIZE);\r
+\r
+       GenerateUnicodeCacheFileName(name, sizeof(name), strfilename, strfilesize, hash);\r
+\r
+       io = FileCreateW(name);\r
+       if (io != NULL)\r
+       {\r
+               SeekBuf(b, 0, 0);\r
+               BufToFile(io, b);\r
+               FileClose(io);\r
+       }\r
+\r
+       FreeBuf(b);\r
+}\r
+\r
+// Unicode キャッシュの読み込み\r
+bool LoadUnicodeCache(wchar_t *strfilename, UINT strfilesize, UCHAR *hash)\r
+{\r
+       UNICODE_CACHE c, t;\r
+       BUF *b;\r
+       UINT i, num;\r
+       IO *io;\r
+       wchar_t name[MAX_PATH];\r
+       UCHAR binhash[MD5_SIZE];\r
+       UCHAR binhash_2[MD5_SIZE];\r
+       // 引数チェック\r
+       if (strfilename == NULL || hash == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       GenerateUnicodeCacheFileName(name, sizeof(name), strfilename, strfilesize, hash);\r
+\r
+       io = FileOpenW(name, false);\r
+       if (io == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       b = FileToBuf(io);\r
+       if (b == NULL)\r
+       {\r
+               FileClose(io);\r
+               return false;\r
+       }\r
+\r
+       SeekBuf(b, 0, 0);\r
+       FileClose(io);\r
+\r
+       Hash(binhash, b->Buf, b->Size >= MD5_SIZE ? (b->Size - MD5_SIZE) : 0, false);\r
+       Copy(binhash_2, ((UCHAR *)b->Buf) + (b->Size >= MD5_SIZE ? (b->Size - MD5_SIZE) : 0), MD5_SIZE);\r
+       if (Cmp(binhash, binhash_2, MD5_SIZE) != 0)\r
+       {\r
+               FreeBuf(b);\r
+               return false;\r
+       }\r
+\r
+       Zero(&c, sizeof(c));\r
+       UniToStr(c.StrFileName, sizeof(c.StrFileName), strfilename);\r
+       c.StrFileSize = strfilesize;\r
+       DisableNetworkNameCache();\r
+       GetMachineName(c.MachineName, sizeof(c.MachineName));\r
+       EnableNetworkNameCache();\r
+       c.OsType = GetOsInfo()->OsType;\r
+       Copy(c.hash, hash, MD5_SIZE);\r
+\r
+#ifdef OS_UNIX\r
+       GetCurrentCharSet(c.CharSet, sizeof(c.CharSet));\r
+#else  // OS_UNIX\r
+       {\r
+               UINT id = MsGetThreadLocale();\r
+               Copy(c.CharSet, &id, sizeof(id));\r
+       }\r
+#endif // OS_UNIX\r
+\r
+       Zero(&t, sizeof(t));\r
+       ReadBuf(b, &t, sizeof(t));\r
+\r
+       if (Cmp(&c, &t, sizeof(UNICODE_CACHE)) != 0)\r
+       {\r
+               FreeBuf(b);\r
+               return false;\r
+       }\r
+\r
+       num = ReadBufInt(b);\r
+\r
+       FreeTable();\r
+       TableList = NewList(CmpTableName);\r
+\r
+       for (i = 0;i < num;i++)\r
+       {\r
+               UINT len;\r
+               TABLE *t = ZeroMalloc(sizeof(TABLE));\r
+\r
+               len = ReadBufInt(b);\r
+               t->name = ZeroMalloc(len + 1);\r
+               ReadBuf(b, t->name, len);\r
+\r
+               len = ReadBufInt(b);\r
+               t->str = ZeroMalloc(len + 1);\r
+               ReadBuf(b, t->str, len);\r
+\r
+               len = ReadBufInt(b);\r
+               t->unistr = ZeroMalloc((len + 1) * sizeof(wchar_t));\r
+               ReadBuf(b, t->unistr, len * sizeof(wchar_t));\r
+\r
+               Add(TableList, t);\r
+       }\r
+\r
+       FreeBuf(b);\r
+\r
+       Sort(TableList);\r
+\r
+       return true;\r
+}\r
+\r
+// 文字列テーブルを読み込む\r
+bool LoadTableMain(wchar_t *filename)\r
+{\r
+       BUF *b;\r
+       UINT64 t1, t2;\r
+       UCHAR hash[MD5_SIZE];\r
+       // 引数チェック\r
+       if (filename == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MayaquaIsMinimalMode())\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (UniStrCmpi(old_table_name, filename) == 0)\r
+       {\r
+               // すでに読み込まれている\r
+               return true;\r
+       }\r
+\r
+       t1 = Tick64();\r
+\r
+       // ファイルを開く\r
+       b = ReadDumpW(filename);\r
+       if (b == NULL)\r
+       {\r
+               char tmp[MAX_SIZE];\r
+               StrCpy(tmp, sizeof(tmp), "Error: Can't read string tables (file not found).\r\nPlease check hamcore.utvpn.\r\n\r\n(First, reboot the computer. If this problem occurs again, please reinstall VPN software files.)");\r
+               Alert(tmp, NULL);\r
+               exit(0);\r
+               return false;\r
+       }\r
+\r
+       Hash(hash, b->Buf, b->Size, false);\r
+\r
+       if (LoadUnicodeCache(filename, b->Size, hash) == false)\r
+       {\r
+               if (LoadTableFromBuf(b) == false)\r
+               {\r
+                       FreeBuf(b);\r
+                       return false;\r
+               }\r
+\r
+               SaveUnicodeCache(filename, b->Size, hash);\r
+\r
+               Debug("Unicode Source: strtable.stb\n");\r
+       }\r
+       else\r
+       {\r
+               Debug("Unicode Source: unicode_cache\n");\r
+       }\r
+\r
+       FreeBuf(b);\r
+\r
+       SetLocale(_UU("DEFAULE_LOCALE"));\r
+\r
+       UniStrCpy(old_table_name, sizeof(old_table_name), filename);\r
+\r
+       t2 = Tick64();\r
+\r
+       if (StrCmpi(_SS("STRTABLE_ID"), STRTABLE_ID) != 0)\r
+       {\r
+               char tmp[MAX_SIZE];\r
+               StrCpy(tmp, sizeof(tmp), "Error: Can't read string tables (invalid version).\r\nPlease check hamcore.utvpn.\r\n\r\n(First, reboot the computer. If this problem occurs again, please reinstall VPN software files.)");\r
+               Alert(tmp, NULL);\r
+               exit(0);\r
+               return false;\r
+       }\r
+\r
+       Debug("Unicode File Read Cost: %u (%u Lines)\n", (UINT)(t2 - t1), LIST_NUM(TableList));\r
+\r
+       return true;\r
+}\r
+bool LoadTable(char *filename)\r
+{\r
+       wchar_t *filename_a = CopyStrToUni(filename);\r
+       bool ret = LoadTableW(filename_a);\r
+\r
+       Free(filename_a);\r
+\r
+       return ret;\r
+}\r
+bool LoadTableW(wchar_t *filename)\r
+{\r
+       bool ret;\r
+       BUF *b;\r
+       wchar_t replace_name[MAX_PATH];\r
+\r
+       Zero(replace_name, sizeof(replace_name));\r
+\r
+       TrackingDisable();\r
+\r
+       b = ReadDump("@table_name.txt");\r
+       if (b != NULL)\r
+       {\r
+               char *s = CfgReadNextLine(b);\r
+               if (s != NULL)\r
+               {\r
+                       if (IsEmptyStr(s) == false)\r
+                       {\r
+                               StrToUni(replace_name, sizeof(replace_name), s);\r
+                               filename = replace_name;\r
+                       }\r
+\r
+                       Free(s);\r
+               }\r
+               FreeBuf(b);\r
+       }\r
+\r
+       ret = LoadTableMain(filename);\r
+\r
+       TrackingEnable();\r
+\r
+       return ret;\r
+}\r
+\r
+\r