* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Mayaqua / FileIO.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/FileIO.c b/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/FileIO.c
new file mode 100644 (file)
index 0000000..b667620
--- /dev/null
@@ -0,0 +1,2234 @@
+// 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
+// FileIO.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
+static char exe_file_name[MAX_SIZE] = "/tmp/a.out";\r
+static wchar_t exe_file_name_w[MAX_SIZE] = L"/tmp/a.out";\r
+static LIST *hamcore = NULL;\r
+static IO *hamcore_io = NULL;\r
+\r
+\r
+// ファイルを保存する\r
+bool SaveFileW(wchar_t *name, void *data, UINT size)\r
+{\r
+       IO *io;\r
+       // 引数チェック\r
+       if (name == NULL || (data == NULL && size != 0))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       io = FileCreateW(name);\r
+       if (io == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (FileWrite(io, data, size) == false)\r
+       {\r
+               FileClose(io);\r
+               return false;\r
+       }\r
+\r
+       FileClose(io);\r
+\r
+       return true;\r
+}\r
+bool SaveFile(char *name, void *data, UINT size)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = SaveFileW(name_w, data, size);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+\r
+// ファイルが存在するかどうか確認する\r
+bool IsFile(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = IsFileW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool IsFileW(wchar_t *name)\r
+{\r
+       IO *io;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       io = FileOpenExW(name, false, false);\r
+       if (io == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       FileClose(io);\r
+\r
+       return true;\r
+}\r
+\r
+// ファイルを置換してリネームする\r
+bool FileReplaceRename(char *old_name, char *new_name)\r
+{\r
+       wchar_t *old_name_w = CopyStrToUni(old_name);\r
+       wchar_t *new_name_w = CopyStrToUni(new_name);\r
+       bool ret = FileReplaceRenameW(old_name_w, new_name_w);\r
+\r
+       Free(old_name_w);\r
+       Free(new_name_w);\r
+\r
+       return ret;\r
+}\r
+bool FileReplaceRenameW(wchar_t *old_name, wchar_t *new_name)\r
+{\r
+       // 引数チェック\r
+       if (old_name == NULL || new_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (FileCopyW(old_name, new_name) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       FileDeleteW(old_name);\r
+\r
+       return true;\r
+}\r
+\r
+// ファイル名を安全な名前にする\r
+void ConvertSafeFileName(char *dst, UINT size, char *src)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (dst == NULL || src == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       StrCpy(dst, size, src);\r
+       for (i = 0;i < StrLen(dst);i++)\r
+       {\r
+               if (IsSafeChar(dst[i]) == false)\r
+               {\r
+                       dst[i] = '_';\r
+               }\r
+       }\r
+}\r
+void ConvertSafeFileNameW(wchar_t *dst, UINT size, wchar_t *src)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (dst == NULL || src == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       UniStrCpy(dst, size, src);\r
+       for (i = 0;i < UniStrLen(dst);i++)\r
+       {\r
+               if (UniIsSafeChar(dst[i]) == false)\r
+               {\r
+                       dst[i] = L'_';\r
+               }\r
+       }\r
+}\r
+\r
+// ディスクの空き容量を取得する\r
+bool GetDiskFree(char *path, UINT64 *free_size, UINT64 *used_size, UINT64 *total_size)\r
+{\r
+       bool ret;\r
+       // 引数チェック\r
+       if (path == NULL)\r
+       {\r
+               path = "./";\r
+       }\r
+\r
+#ifdef OS_WIN32\r
+       ret = Win32GetDiskFree(path, free_size, used_size, total_size);\r
+#else  // OS_WIN32\r
+       ret = UnixGetDiskFree(path, free_size, used_size, total_size);\r
+#endif // OS_WIN32\r
+\r
+       return ret;\r
+}\r
+bool GetDiskFreeW(wchar_t *path, UINT64 *free_size, UINT64 *used_size, UINT64 *total_size)\r
+{\r
+       bool ret;\r
+       // 引数チェック\r
+       if (path == NULL)\r
+       {\r
+               path = L"./";\r
+       }\r
+\r
+#ifdef OS_WIN32\r
+       ret = Win32GetDiskFreeW(path, free_size, used_size, total_size);\r
+#else  // OS_WIN32\r
+       ret = UnixGetDiskFreeW(path, free_size, used_size, total_size);\r
+#endif // OS_WIN32\r
+\r
+       return ret;\r
+}\r
+\r
+// ディレクトリの列挙\r
+DIRLIST *EnumDirEx(char *dirname, COMPARE *compare)\r
+{\r
+       wchar_t *dirname_w = CopyStrToUni(dirname);\r
+       DIRLIST *ret = EnumDirExW(dirname_w, compare);\r
+\r
+       Free(dirname_w);\r
+\r
+       return ret;\r
+}\r
+DIRLIST *EnumDirExW(wchar_t *dirname, COMPARE *compare)\r
+{\r
+       DIRLIST *d = NULL;\r
+       // 引数チェック\r
+       if (dirname == NULL)\r
+       {\r
+               dirname = L"./";\r
+       }\r
+\r
+       if (compare == NULL)\r
+       {\r
+               compare = CompareDirListByName;\r
+       }\r
+\r
+#ifdef OS_WIN32\r
+       d = Win32EnumDirExW(dirname, compare);\r
+#else  // OS_WIN32\r
+       d = UnixEnumDirExW(dirname, compare);\r
+#endif // OS_WIN32\r
+\r
+       return d;\r
+}\r
+DIRLIST *EnumDir(char *dirname)\r
+{\r
+       return EnumDirEx(dirname, NULL);\r
+}\r
+DIRLIST *EnumDirW(wchar_t *dirname)\r
+{\r
+       return EnumDirExW(dirname, NULL);\r
+}\r
+\r
+// DIRLIST リストの比較\r
+int CompareDirListByName(void *p1, void *p2)\r
+{\r
+       DIRENT *d1, *d2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       d1 = *(DIRENT **)p1;\r
+       d2 = *(DIRENT **)p2;\r
+       if (d1 == NULL || d2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       return UniStrCmpi(d1->FileNameW, d2->FileNameW);\r
+}\r
+\r
+// ディレクトリ列挙の解放\r
+void FreeDir(DIRLIST *d)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (d == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < d->NumFiles;i++)\r
+       {\r
+               DIRENT *f = d->File[i];\r
+               Free(f->FileName);\r
+               Free(f->FileNameW);\r
+               Free(f);\r
+       }\r
+       Free(d->File);\r
+       Free(d);\r
+}\r
+\r
+\r
+// ファイル名を安全にする\r
+void UniSafeFileName(wchar_t *name)\r
+{\r
+       UINT i, len, dlen;\r
+       static wchar_t *danger_str = L"\\/:*?\"<>|";\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       dlen = UniStrLen(danger_str);\r
+       len = UniStrLen(name);\r
+\r
+       for (i = 0;i < len;i++)\r
+       {\r
+               wchar_t c = name[i];\r
+               UINT j;\r
+               for (j = 0;j < dlen;j++)\r
+               {\r
+                       if (c == danger_str[j])\r
+                       {\r
+                               c = L'_';\r
+                       }\r
+               }\r
+               name[i] = c;\r
+       }\r
+}\r
+void SafeFileNameW(wchar_t *name)\r
+{\r
+       UniSafeFileName(name);\r
+}\r
+\r
+// HamCore の読み込み\r
+BUF *ReadHamcoreW(wchar_t *filename)\r
+{\r
+       char *filename_a = CopyUniToStr(filename);\r
+       BUF *ret;\r
+\r
+       ret = ReadHamcore(filename_a);\r
+\r
+       Free(filename_a);\r
+\r
+       return ret;\r
+}\r
+BUF *ReadHamcore(char *name)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       wchar_t exe_dir[MAX_SIZE];\r
+       BUF *b;\r
+       char filename[MAX_PATH];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (name[0] == '|')\r
+       {\r
+               name++;\r
+       }\r
+\r
+       if (name[0] == '/' || name[0] == '\\')\r
+       {\r
+               name++;\r
+       }\r
+\r
+       StrCpy(filename, sizeof(filename), name);\r
+\r
+       ReplaceStrEx(filename, sizeof(filename), filename, "/", "\\", true);\r
+\r
+       if (MayaquaIsMinimalMode())\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // ローカルディスクの hamcore/ ディレクトリにファイルがあればそれを読み込む\r
+       GetExeDirW(exe_dir, sizeof(exe_dir));\r
+\r
+       UniFormat(tmp, sizeof(tmp), L"%s/%S/%S", exe_dir, HAMCORE_DIR_NAME, filename);\r
+\r
+       b = ReadDumpW(tmp);\r
+       if (b != NULL)\r
+       {\r
+               return b;\r
+       }\r
+\r
+       // 無い場合は HamCore ファイルシステムから探す\r
+       LockList(hamcore);\r
+       {\r
+               HC t, *c;\r
+               UINT i;\r
+\r
+               Zero(&t, sizeof(t));\r
+               t.FileName = filename;\r
+               c = Search(hamcore, &t);\r
+\r
+               if (c == NULL)\r
+               {\r
+                       // ファイルが存在しない\r
+                       b = NULL;\r
+               }\r
+               else\r
+               {\r
+                       // ファイルが存在する\r
+                       if (c->Buffer != NULL)\r
+                       {\r
+                               // 既に読み込まれている\r
+                               b = NewBuf();\r
+                               WriteBuf(b, c->Buffer, c->Size);\r
+                               SeekBuf(b, 0, 0);\r
+                               c->LastAccess = Tick64();\r
+                       }\r
+                       else\r
+                       {\r
+                               // 読み込まれていないのでファイルから読み込む\r
+                               if (FileSeek(hamcore_io, 0, c->Offset) == false)\r
+                               {\r
+                                       // シークに失敗した\r
+                                       b = NULL;\r
+                               }\r
+                               else\r
+                               {\r
+                                       // 圧縮データを読み込む\r
+                                       void *data = Malloc(c->SizeCompressed);\r
+                                       if (FileRead(hamcore_io, data, c->SizeCompressed) == false)\r
+                                       {\r
+                                               // 読み込みに失敗した\r
+                                               Free(data);\r
+                                               b = NULL;\r
+                                       }\r
+                                       else\r
+                                       {\r
+                                               // 展開する\r
+                                               c->Buffer = ZeroMalloc(c->Size);\r
+                                               if (Uncompress(c->Buffer, c->Size, data, c->SizeCompressed) != c->Size)\r
+                                               {\r
+                                                       // 展開に失敗した\r
+                                                       Free(data);\r
+                                                       Free(c->Buffer);\r
+                                                       b = NULL;\r
+                                               }\r
+                                               else\r
+                                               {\r
+                                                       // 成功した\r
+                                                       Free(data);\r
+                                                       b = NewBuf();\r
+                                                       WriteBuf(b, c->Buffer, c->Size);\r
+                                                       SeekBuf(b, 0, 0);\r
+                                                       c->LastAccess = Tick64();\r
+                                               }\r
+                                       }\r
+                               }\r
+                       }\r
+               }\r
+\r
+               // 有効期限の切れたキャッシュを削除する\r
+               for (i = 0;i < LIST_NUM(hamcore);i++)\r
+               {\r
+                       HC *c = LIST_DATA(hamcore, i);\r
+\r
+                       if (c->Buffer != NULL)\r
+                       {\r
+                               if (((c->LastAccess + HAMCORE_CACHE_EXPIRES) <= Tick64()) ||\r
+                                       (StartWith(c->FileName, "Li")))\r
+                               {\r
+                                       Free(c->Buffer);\r
+                                       c->Buffer = NULL;\r
+                               }\r
+                       }\r
+               }\r
+       }\r
+       UnlockList(hamcore);\r
+\r
+       return b;\r
+}\r
+\r
+// HamCore ファイルシステムの初期化\r
+void InitHamcore()\r
+{\r
+       wchar_t tmp[MAX_PATH];\r
+       wchar_t tmp2[MAX_PATH];\r
+       wchar_t exe_dir[MAX_PATH];\r
+       UINT i, num;\r
+       char header[HAMCORE_HEADER_SIZE];\r
+\r
+       hamcore = NewList(CompareHamcore);\r
+\r
+       if (MayaquaIsMinimalMode())\r
+       {\r
+               return;\r
+       }\r
+\r
+       GetExeDirW(exe_dir, sizeof(exe_dir));\r
+       UniFormat(tmp, sizeof(tmp), L"%s/%S", exe_dir, HAMCORE_FILE_NAME);\r
+\r
+       UniFormat(tmp2, sizeof(tmp2), L"%s/%S", exe_dir, HAMCORE_FILE_NAME_2);\r
+\r
+       // _hamcore.utvpn がある場合は hamcore.utvpn に上書きする\r
+       FileReplaceRenameW(tmp2, tmp);\r
+\r
+       // hamcore.utvpn ファイルがあれば読み込む\r
+       hamcore_io = FileOpenW(tmp, false);\r
+       if (hamcore_io == NULL)\r
+       {\r
+               // 無い場合は別の場所を探す\r
+#ifdef OS_WIN32\r
+               UniFormat(tmp, sizeof(tmp), L"%S/%S", MsGetSystem32Dir(), HAMCORE_FILE_NAME);\r
+#else  // OS_WIN32\r
+               UniFormat(tmp, sizeof(tmp), L"/bin/%S", HAMCORE_FILE_NAME);\r
+#endif // OS_WIN32\r
+\r
+               hamcore_io = FileOpenW(tmp, false);\r
+               if (hamcore_io == NULL)\r
+               {\r
+                       return;\r
+               }\r
+       }\r
+\r
+       // ファイルヘッダを読み込む\r
+       Zero(header, sizeof(header));\r
+       FileRead(hamcore_io, header, HAMCORE_HEADER_SIZE);\r
+\r
+       if (Cmp(header, HAMCORE_HEADER_DATA, HAMCORE_HEADER_SIZE) != 0)\r
+       {\r
+               // ヘッダ不正\r
+               FileClose(hamcore_io);\r
+               hamcore_io = NULL;\r
+               return;\r
+       }\r
+\r
+       // ファイル個数\r
+       num = 0;\r
+       FileRead(hamcore_io, &num, sizeof(num));\r
+       num = Endian32(num);\r
+       for (i = 0;i < num;i++)\r
+       {\r
+               // ファイル名\r
+               char tmp[MAX_SIZE];\r
+               UINT str_size = 0;\r
+               HC *c;\r
+\r
+               FileRead(hamcore_io, &str_size, sizeof(str_size));\r
+               str_size = Endian32(str_size);\r
+               if (str_size >= 1)\r
+               {\r
+                       str_size--;\r
+               }\r
+\r
+               Zero(tmp, sizeof(tmp));\r
+               FileRead(hamcore_io, tmp, str_size);\r
+\r
+               c = ZeroMalloc(sizeof(HC));\r
+               c->FileName = CopyStr(tmp);\r
+\r
+               FileRead(hamcore_io, &c->Size, sizeof(UINT));\r
+               c->Size = Endian32(c->Size);\r
+\r
+               FileRead(hamcore_io, &c->SizeCompressed, sizeof(UINT));\r
+               c->SizeCompressed = Endian32(c->SizeCompressed);\r
+\r
+               FileRead(hamcore_io, &c->Offset, sizeof(UINT));\r
+               c->Offset = Endian32(c->Offset);\r
+\r
+               Insert(hamcore, c);\r
+       }\r
+}\r
+\r
+// HamCore ファイルシステムの解放\r
+void FreeHamcore()\r
+{\r
+       UINT i;\r
+       for (i = 0;i < LIST_NUM(hamcore);i++)\r
+       {\r
+               HC *c = LIST_DATA(hamcore, i);\r
+               Free(c->FileName);\r
+               if (c->Buffer != NULL)\r
+               {\r
+                       Free(c->Buffer);\r
+               }\r
+               Free(c);\r
+       }\r
+       ReleaseList(hamcore);\r
+\r
+       FileClose(hamcore_io);\r
+       hamcore_io = NULL;\r
+       hamcore = NULL;\r
+}\r
+\r
+// Hamcore のビルド\r
+void BuildHamcore()\r
+{\r
+       BUF *b;\r
+       char tmp[MAX_SIZE];\r
+       char exe_dir[MAX_SIZE];\r
+       char *s;\r
+       bool ok = true;\r
+       LIST *o;\r
+       UINT i;\r
+\r
+       GetExeDir(exe_dir, sizeof(exe_dir));\r
+       Format(tmp, sizeof(tmp), "%s/%s", exe_dir, HAMCORE_TEXT_NAME);\r
+\r
+       b = ReadDump(tmp);\r
+       if (b == NULL)\r
+       {\r
+               Print("Failed to open %s.\n", tmp);\r
+               return;\r
+       }\r
+\r
+       o = NewListFast(CompareHamcore);\r
+\r
+       while ((s = CfgReadNextLine(b)) != NULL)\r
+       {\r
+               char tmp[MAX_SIZE];\r
+               BUF *b;\r
+               Trim(s);\r
+\r
+               Format(tmp, sizeof(tmp), "%s/%s/%s", exe_dir, HAMCORE_DIR_NAME, s);\r
+\r
+               b = ReadDump(tmp);\r
+               if (b == NULL)\r
+               {\r
+                       Print("Failed to open %s.\n", s);\r
+                       ok = false;\r
+               }\r
+               else\r
+               {\r
+                       HC *c = ZeroMalloc(sizeof(HC));\r
+                       UINT tmp_size;\r
+                       void *tmp;\r
+                       c->FileName = CopyStr(s);\r
+                       c->Size = b->Size;\r
+                       tmp_size = CalcCompress(c->Size);\r
+                       tmp = Malloc(tmp_size);\r
+                       c->SizeCompressed = Compress(tmp, tmp_size, b->Buf, b->Size);\r
+                       c->Buffer = tmp;\r
+                       Insert(o, c);\r
+                       Print("%s: %u -> %u\n", s, c->Size, c->SizeCompressed);\r
+                       FreeBuf(b);\r
+               }\r
+\r
+               Free(s);\r
+       }\r
+\r
+       if (ok)\r
+       {\r
+               // 各ファイルのバッファのオフセットを計算する\r
+               UINT i, z;\r
+               char tmp[MAX_SIZE];\r
+               BUF *b;\r
+               z = 0;\r
+               z += HAMCORE_HEADER_SIZE;\r
+               // ファイルの個数\r
+               z += sizeof(UINT);\r
+               // まずファイルテーブル\r
+               for (i = 0;i < LIST_NUM(o);i++)\r
+               {\r
+                       HC *c = LIST_DATA(o, i);\r
+                       // ファイル名\r
+                       z += StrLen(c->FileName) + sizeof(UINT);\r
+                       // ファイルサイズ\r
+                       z += sizeof(UINT);\r
+                       z += sizeof(UINT);\r
+                       // オフセットデータ\r
+                       z += sizeof(UINT);\r
+               }\r
+               // ファイル本体\r
+               for (i = 0;i < LIST_NUM(o);i++)\r
+               {\r
+                       HC *c = LIST_DATA(o, i);\r
+                       // バッファ本体\r
+                       c->Offset = z;\r
+                       printf("%s: offset: %u\n", c->FileName, c->Offset);\r
+                       z += c->SizeCompressed;\r
+               }\r
+               // 書き込み\r
+               b = NewBuf();\r
+               // ヘッダ\r
+               WriteBuf(b, HAMCORE_HEADER_DATA, HAMCORE_HEADER_SIZE);\r
+               WriteBufInt(b, LIST_NUM(o));\r
+               for (i = 0;i < LIST_NUM(o);i++)\r
+               {\r
+                       HC *c = LIST_DATA(o, i);\r
+                       // ファイル名\r
+                       WriteBufStr(b, c->FileName);\r
+                       // ファイルサイズ\r
+                       WriteBufInt(b, c->Size);\r
+                       WriteBufInt(b, c->SizeCompressed);\r
+                       // オフセット\r
+                       WriteBufInt(b, c->Offset);\r
+               }\r
+               // 本体\r
+               for (i = 0;i < LIST_NUM(o);i++)\r
+               {\r
+                       HC *c = LIST_DATA(o, i);\r
+                       WriteBuf(b, c->Buffer, c->SizeCompressed);\r
+               }\r
+               // 書き込み\r
+               Format(tmp, sizeof(tmp), "%s/%s", exe_dir, HAMCORE_FILE_NAME "__");\r
+               Print("Writing %s...\n", tmp);\r
+               FileDelete(tmp);\r
+               DumpBuf(b, tmp);\r
+               FreeBuf(b);\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               HC *c = LIST_DATA(o, i);\r
+               Free(c->Buffer);\r
+               Free(c->FileName);\r
+               Free(c);\r
+       }\r
+\r
+       ReleaseList(o);\r
+\r
+       FreeBuf(b);\r
+}\r
+\r
+// HC の比較\r
+int CompareHamcore(void *p1, void *p2)\r
+{\r
+       HC *c1, *c2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       c1 = *(HC **)p1;\r
+       c2 = *(HC **)p2;\r
+       if (c1 == NULL || c2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       return StrCmpi(c1->FileName, c2->FileName);\r
+}\r
+\r
+// EXE ファイルがあるディレクトリ名の取得\r
+void GetExeDir(char *name, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       GetDirNameFromFilePath(name, size, exe_file_name);\r
+}\r
+void GetExeDirW(wchar_t *name, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       GetDirNameFromFilePathW(name, size, exe_file_name_w);\r
+}\r
+\r
+// EXE ファイル名の取得\r
+void GetExeName(char *name, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       StrCpy(name, size, exe_file_name);\r
+}\r
+void GetExeNameW(wchar_t *name, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       UniStrCpy(name, size, exe_file_name_w);\r
+}\r
+\r
+// EXE ファイル名の取得初期化\r
+void InitGetExeName(char *arg)\r
+{\r
+       wchar_t *arg_w = NULL;\r
+       // 引数チェック\r
+       if (arg == NULL)\r
+       {\r
+               arg = "./a.out";\r
+       }\r
+\r
+       arg_w = CopyUtfToUni(arg);\r
+\r
+#ifdef OS_WIN32\r
+       Win32GetExeNameW(exe_file_name_w, sizeof(exe_file_name_w));\r
+#else  // OS_WIN32\r
+       UnixGetExeNameW(exe_file_name_w, sizeof(exe_file_name_w), arg_w);\r
+#endif // OS_WIN32\r
+\r
+       UniToStr(exe_file_name, sizeof(exe_file_name), exe_file_name_w);\r
+\r
+       Free(arg_w);\r
+}\r
+\r
+// Unix における実行可能バイナリファイルのフルパスの取得\r
+void UnixGetExeNameW(wchar_t *name, UINT size, wchar_t *arg)\r
+{\r
+       UNI_TOKEN_LIST *t;\r
+       char *path_str;\r
+       wchar_t *path_str_w;\r
+       bool ok = false;\r
+       // 引数チェック\r
+       if (name == NULL || arg == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       path_str = GetCurrentPathEnvStr();\r
+       path_str_w = CopyUtfToUni(path_str);\r
+\r
+       t = ParseSplitedPathW(path_str_w);\r
+\r
+       if (t != NULL)\r
+       {\r
+               UINT i;\r
+               for (i = 0;i < t->NumTokens;i++)\r
+               {\r
+                       wchar_t *s = t->Token[i];\r
+                       wchar_t tmp[MAX_SIZE];\r
+\r
+                       ConbinePathW(tmp, sizeof(tmp), s, arg);\r
+\r
+                       if (IsFileExistsInnerW(tmp))\r
+                       {\r
+#ifdef OS_UNIX\r
+                               if (UnixCheckExecAccessW(tmp) == false)\r
+                               {\r
+                                       continue;\r
+                               }\r
+#endif // OS_UNIX\r
+                               ok = true;\r
+                               UniStrCpy(name, size, tmp);\r
+                               break;\r
+                       }\r
+               }\r
+\r
+               UniFreeToken(t);\r
+       }\r
+\r
+       Free(path_str);\r
+       Free(path_str_w);\r
+\r
+       if (ok == false)\r
+       {\r
+               // パスの検索に失敗した場合\r
+#ifdef OS_UNIX\r
+               UnixGetCurrentDirW(name, size);\r
+#else  // OS_UNIX\r
+               Win32GetCurrentDirW(name, size);\r
+#endif // OS_UNIX\r
+               ConbinePathW(name, size, name, arg);\r
+       }\r
+}\r
+\r
+// 安全なファイル名を生成する\r
+void MakeSafeFileName(char *dst, UINT size, char *src)\r
+{\r
+       char tmp[MAX_PATH];\r
+       // 引数チェック\r
+       if (dst == NULL || src == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       StrCpy(tmp, sizeof(tmp), src);\r
+       ReplaceStrEx(tmp, sizeof(tmp), tmp, "..", "__", false);\r
+       ReplaceStrEx(tmp, sizeof(tmp), tmp, "/", "_", false);\r
+       ReplaceStrEx(tmp, sizeof(tmp), tmp, "\\", "_", false);\r
+       ReplaceStrEx(tmp, sizeof(tmp), tmp, "@", "_", false);\r
+       ReplaceStrEx(tmp, sizeof(tmp), tmp, "|", "_", false);\r
+\r
+       StrCpy(dst, size, tmp);\r
+}\r
+void MakeSafeFileNameW(wchar_t *dst, UINT size, wchar_t *src)\r
+{\r
+       wchar_t tmp[MAX_PATH];\r
+       // 引数チェック\r
+       if (dst == NULL || src == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       UniStrCpy(tmp, sizeof(tmp), src);\r
+       UniReplaceStrEx(tmp, sizeof(tmp), tmp, L"..", L"__", false);\r
+       UniReplaceStrEx(tmp, sizeof(tmp), tmp, L"/", L"_", false);\r
+       UniReplaceStrEx(tmp, sizeof(tmp), tmp, L"\\", L"_", false);\r
+       UniReplaceStrEx(tmp, sizeof(tmp), tmp, L"@", L"_", false);\r
+       UniReplaceStrEx(tmp, sizeof(tmp), tmp, L"|", L"_", false);\r
+\r
+       UniStrCpy(dst, size, tmp);\r
+}\r
+\r
+// ファイルパスからファイル名を取得する\r
+void GetFileNameFromFilePathW(wchar_t *dst, UINT size, wchar_t *filepath)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       UINT i, len, wp;\r
+       // 引数チェック\r
+       if (dst == NULL || filepath == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       len = MIN(UniStrLen(filepath), (MAX_SIZE - 2));\r
+       wp = 0;\r
+\r
+       for (i = 0;i < (len + 1);i++)\r
+       {\r
+               wchar_t c = filepath[i];\r
+\r
+               switch (c)\r
+               {\r
+               case L'\\':\r
+               case L'/':\r
+               case 0:\r
+                       tmp[wp] = 0;\r
+                       wp = 0;\r
+                       break;\r
+\r
+               default:\r
+                       tmp[wp] = c;\r
+                       wp++;\r
+                       break;\r
+               }\r
+       }\r
+\r
+       UniStrCpy(dst, size, tmp);\r
+}\r
+void GetFileNameFromFilePath(char *dst, UINT size, char *filepath)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       UINT i, len, wp;\r
+       // 引数チェック\r
+       if (dst == NULL || filepath == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       len = MIN(StrLen(filepath), (MAX_SIZE - 2));\r
+       wp = 0;\r
+\r
+       for (i = 0;i < (len + 1);i++)\r
+       {\r
+               char c = filepath[i];\r
+\r
+               switch (c)\r
+               {\r
+               case '\\':\r
+               case '/':\r
+               case 0:\r
+                       tmp[wp] = 0;\r
+                       wp = 0;\r
+                       break;\r
+\r
+               default:\r
+                       tmp[wp] = c;\r
+                       wp++;\r
+                       break;\r
+               }\r
+       }\r
+\r
+       StrCpy(dst, size, tmp);\r
+}\r
+void GetDirNameFromFilePathW(wchar_t *dst, UINT size, wchar_t *filepath)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       UINT wp;\r
+       UINT i;\r
+       UINT len;\r
+       // 引数チェック\r
+       if (dst == NULL || filepath == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       UniStrCpy(tmp, sizeof(tmp), filepath);\r
+       if (UniEndWith(tmp, L"\\") || UniEndWith(tmp, L"/"))\r
+       {\r
+               tmp[UniStrLen(tmp) - 1] = 0;\r
+       }\r
+\r
+       len = UniStrLen(tmp);\r
+\r
+       UniStrCpy(dst, size, L"");\r
+\r
+       wp = 0;\r
+\r
+       for (i = 0;i < len;i++)\r
+       {\r
+               wchar_t c = tmp[i];\r
+               if (c == L'/' || c == L'\\')\r
+               {\r
+                       tmp[wp++] = 0;\r
+                       wp = 0;\r
+                       UniStrCat(dst, size, tmp);\r
+                       tmp[wp++] = c;\r
+               }\r
+               else\r
+               {\r
+                       tmp[wp++] = c;\r
+               }\r
+       }\r
+\r
+       if (UniStrLen(dst) == 0)\r
+       {\r
+               UniStrCpy(dst, size, L"/");\r
+       }\r
+\r
+       NormalizePathW(dst, size, dst);\r
+}\r
+\r
+// ファイルパスからディレクトリ名を取得する\r
+void GetDirNameFromFilePath(char *dst, UINT size, char *filepath)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       UINT wp;\r
+       UINT i;\r
+       UINT len;\r
+       // 引数チェック\r
+       if (dst == NULL || filepath == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       StrCpy(tmp, sizeof(tmp), filepath);\r
+       if (EndWith(tmp, "\\") || EndWith(tmp, "/"))\r
+       {\r
+               tmp[StrLen(tmp) - 1] = 0;\r
+       }\r
+\r
+       len = StrLen(tmp);\r
+\r
+       StrCpy(dst, size, "");\r
+\r
+       wp = 0;\r
+\r
+       for (i = 0;i < len;i++)\r
+       {\r
+               char c = tmp[i];\r
+               if (c == '/' || c == '\\')\r
+               {\r
+                       tmp[wp++] = 0;\r
+                       wp = 0;\r
+                       StrCat(dst, size, tmp);\r
+                       tmp[wp++] = c;\r
+               }\r
+               else\r
+               {\r
+                       tmp[wp++] = c;\r
+               }\r
+       }\r
+\r
+       if (StrLen(dst) == 0)\r
+       {\r
+               StrCpy(dst, size, "/");\r
+       }\r
+\r
+       NormalizePath(dst, size, dst);\r
+}\r
+\r
+// 2 つのパスを結合する\r
+void ConbinePath(char *dst, UINT size, char *dirname, char *filename)\r
+{\r
+       wchar_t dst_w[MAX_PATH];\r
+       wchar_t *dirname_w = CopyStrToUni(dirname);\r
+       wchar_t *filename_w = CopyStrToUni(filename);\r
+\r
+       ConbinePathW(dst_w, sizeof(dst_w), dirname_w, filename_w);\r
+\r
+       Free(dirname_w);\r
+       Free(filename_w);\r
+\r
+       UniToStr(dst, size, dst_w);\r
+}\r
+void ConbinePathW(wchar_t *dst, UINT size, wchar_t *dirname, wchar_t *filename)\r
+{\r
+       bool is_full_path;\r
+       wchar_t tmp[MAX_SIZE];\r
+       wchar_t filename_ident[MAX_SIZE];\r
+       // 引数チェック\r
+       if (dst == NULL || dirname == NULL || filename == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       NormalizePathW(filename_ident, sizeof(filename_ident), filename);\r
+\r
+       is_full_path = false;\r
+\r
+       if (UniStartWith(filename_ident, L"\\") || UniStartWith(filename_ident, L"/"))\r
+       {\r
+               is_full_path = true;\r
+       }\r
+\r
+       filename = &filename_ident[0];\r
+\r
+#ifdef OS_WIN32\r
+       if (UniStrLen(filename) >= 2)\r
+       {\r
+               if ((L'a' <= filename[0] && filename[0] <= L'z') || (L'A' <= filename[0] && filename[0] <= L'Z'))\r
+               {\r
+                       if (filename[1] == L':')\r
+                       {\r
+                               is_full_path = true;\r
+                       }\r
+               }\r
+       }\r
+#endif // OS_WIN32\r
+\r
+       if (is_full_path == false)\r
+       {\r
+               UniStrCpy(tmp, sizeof(tmp), dirname);\r
+               if (UniEndWith(tmp, L"/") == false && UniEndWith(tmp, L"\\") == false)\r
+               {\r
+                       UniStrCat(tmp, sizeof(tmp), L"/");\r
+               }\r
+               UniStrCat(tmp, sizeof(tmp), filename);\r
+       }\r
+       else\r
+       {\r
+               UniStrCpy(tmp, sizeof(tmp), filename);\r
+       }\r
+\r
+       NormalizePathW(dst, size, tmp);\r
+}\r
+void CombinePath(char *dst, UINT size, char *dirname, char *filename)\r
+{\r
+       ConbinePath(dst, size, dirname, filename);\r
+}\r
+void CombinePathW(wchar_t *dst, UINT size, wchar_t *dirname, wchar_t *filename)\r
+{\r
+       ConbinePathW(dst, size, dirname, filename);\r
+}\r
+\r
+// ファイルが存在するかどうか確認する\r
+bool IsFileExists(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = IsFileExistsW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool IsFileExistsW(wchar_t *name)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       InnerFilePathW(tmp, sizeof(tmp), name);\r
+\r
+       return IsFileExistsInnerW(tmp);\r
+}\r
+bool IsFileExistsInner(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = IsFileExistsInnerW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool IsFileExistsInnerW(wchar_t *name)\r
+{\r
+       IO *o;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       o = FileOpenInnerW(name, false, false);\r
+       if (o == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       FileClose(o);\r
+\r
+       return true;\r
+}\r
+\r
+// 現在の環境変数 PATH の内容を取得\r
+char *GetCurrentPathEnvStr()\r
+{\r
+       char tmp[1024];\r
+       char *tag_name;\r
+\r
+#ifdef OS_WIN32\r
+       tag_name = "Path";\r
+#else  // OS_WIN32\r
+       tag_name = "PATH";\r
+#endif // OS_WIN32\r
+\r
+       if (GetEnv(tag_name, tmp, sizeof(tmp)) == false)\r
+       {\r
+#ifdef OS_WIN32\r
+               Win32GetCurrentDir(tmp, sizeof(tmp));\r
+#else  // OS_WIN32\r
+               UnixGetCurrentDir(tmp, sizeof(tmp));\r
+#endif // OS_WIN32\r
+       }\r
+\r
+       return CopyStr(tmp);\r
+}\r
+\r
+// コロン文字列で区切られた複数のパスを取得する\r
+UNI_TOKEN_LIST *ParseSplitedPathW(wchar_t *path)\r
+{\r
+       UNI_TOKEN_LIST *ret;\r
+       wchar_t *tmp = UniCopyStr(path);\r
+       wchar_t *split_str;\r
+       UINT i;\r
+\r
+       UniTrim(tmp);\r
+       UniTrimCrlf(tmp);\r
+       UniTrim(tmp);\r
+       UniTrimCrlf(tmp);\r
+\r
+#ifdef OS_WIN32\r
+       split_str = L";";\r
+#else  // OS_WIN32\r
+       split_str = L":";\r
+#endif // OS_WIN32\r
+\r
+       ret = UniParseToken(tmp, split_str);\r
+\r
+       if (ret != NULL)\r
+       {\r
+               for (i = 0;i < ret->NumTokens;i++)\r
+               {\r
+                       UniTrim(ret->Token[i]);\r
+                       UniTrimCrlf(ret->Token[i]);\r
+                       UniTrim(ret->Token[i]);\r
+                       UniTrimCrlf(ret->Token[i]);\r
+               }\r
+       }\r
+\r
+       Free(tmp);\r
+\r
+       return ret;\r
+}\r
+TOKEN_LIST *ParseSplitedPath(char *path)\r
+{\r
+       TOKEN_LIST *ret;\r
+       char *tmp = CopyStr(path);\r
+       char *split_str;\r
+       UINT i;\r
+\r
+       Trim(tmp);\r
+       TrimCrlf(tmp);\r
+       Trim(tmp);\r
+       TrimCrlf(tmp);\r
+\r
+#ifdef OS_WIN32\r
+       split_str = ";";\r
+#else  // OS_WIN32\r
+       split_str = ":";\r
+#endif // OS_WIN32\r
+\r
+       ret = ParseToken(tmp, split_str);\r
+\r
+       if (ret != NULL)\r
+       {\r
+               for (i = 0;i < ret->NumTokens;i++)\r
+               {\r
+                       Trim(ret->Token[i]);\r
+                       TrimCrlf(ret->Token[i]);\r
+                       Trim(ret->Token[i]);\r
+                       TrimCrlf(ret->Token[i]);\r
+               }\r
+       }\r
+\r
+       Free(tmp);\r
+\r
+       return ret;\r
+}\r
+\r
+// 現在のディレクトリを取得する\r
+void GetCurrentDirW(wchar_t *name, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+#ifdef OS_WIN32\r
+       Win32GetCurrentDirW(name, size);\r
+#else  // OS_WIN32\r
+       UnixGetCurrentDirW(name, size);\r
+#endif // OS_WIN32\r
+}\r
+void GetCurrentDir(char *name, UINT size)\r
+{\r
+       wchar_t name_w[MAX_PATH];\r
+\r
+       GetCurrentDirW(name_w, sizeof(name_w));\r
+\r
+       UniToStr(name, size, name_w);\r
+}\r
+\r
+// ファイルパスを正規化する\r
+void NormalizePathW(wchar_t *dst, UINT size, wchar_t *src)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       UNI_TOKEN_LIST *t;\r
+       bool first_double_slash = false;\r
+       bool first_single_slash = false;\r
+       wchar_t win32_drive_char = 0;\r
+       bool is_full_path = false;\r
+       UINT i;\r
+       SK *sk;\r
+       // 引数チェック\r
+       if (dst == NULL || src == 0)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // パスを変換する (Win32, UNIX 変換)\r
+       UniStrCpy(tmp, sizeof(tmp), src);\r
+       ConvertPathW(tmp);\r
+       UniTrim(tmp);\r
+\r
+       // 先頭が "./" や "../" で始まっている場合はカレントディレクトリに置換する\r
+       if (UniStartWith(tmp, L"./") || UniStartWith(tmp, L".\\") ||\r
+               UniStartWith(tmp, L"../") || UniStartWith(tmp, L"..\\") ||\r
+               UniStrCmpi(tmp, L".") == 0 || UniStrCmpi(tmp, L"..") == 0)\r
+       {\r
+               wchar_t cd[MAX_SIZE];\r
+               Zero(cd, sizeof(cd));\r
+\r
+#ifdef OS_WIN32\r
+               Win32GetCurrentDirW(cd, sizeof(cd));\r
+#else  // OS_WIN32\r
+               UnixGetCurrentDirW(cd, sizeof(cd));\r
+#endif // OS_WIN32\r
+\r
+               if (UniStartWith(tmp, L".."))\r
+               {\r
+                       UniStrCat(cd, sizeof(cd), L"/../");\r
+                       UniStrCat(cd, sizeof(cd), tmp + 2);\r
+               }\r
+               else\r
+               {\r
+                       UniStrCat(cd, sizeof(cd), L"/");\r
+                       UniStrCat(cd, sizeof(cd), tmp);\r
+               }\r
+\r
+               UniStrCpy(tmp, sizeof(tmp), cd);\r
+       }\r
+\r
+       // 先頭が "~/" で始まっている場合はホームディレクトリに置換する\r
+       if (UniStartWith(tmp, L"~/") || UniStartWith(tmp, L"~\\"))\r
+       {\r
+               wchar_t tmp2[MAX_SIZE];\r
+               GetHomeDirW(tmp2, sizeof(tmp2));\r
+               UniStrCat(tmp2, sizeof(tmp2), L"/");\r
+               UniStrCat(tmp2, sizeof(tmp2), tmp + 2);\r
+               UniStrCpy(tmp, sizeof(tmp), tmp2);\r
+       }\r
+\r
+       if (UniStartWith(tmp, L"//") || UniStartWith(tmp, L"\\\\"))\r
+       {\r
+               // 最初が "//" または "\\" で始まる\r
+               first_double_slash = true;\r
+               is_full_path = true;\r
+       }\r
+       else if (UniStartWith(tmp, L"/") || UniStartWith(tmp, L"\\"))\r
+       {\r
+               // 最初が "\" で始まる\r
+               first_single_slash = true;\r
+               is_full_path = true;\r
+       }\r
+\r
+       if (UniStrLen(tmp) >= 2)\r
+       {\r
+               if (tmp[1] == L':')\r
+               {\r
+                       if (OS_IS_WINDOWS(GetOsInfo()->OsType))\r
+                       {\r
+                               // Win32 のドライブ文字列表記\r
+                               wchar_t tmp2[MAX_SIZE];\r
+                               is_full_path = true;\r
+                               win32_drive_char = tmp[0];\r
+                               UniStrCpy(tmp2, sizeof(tmp2), tmp + 2);\r
+                               UniStrCpy(tmp, sizeof(tmp), tmp2);\r
+                       }\r
+               }\r
+       }\r
+\r
+       if (UniStrLen(tmp) == 1 && (tmp[0] == L'/' || tmp[0] == L'\\'))\r
+       {\r
+               tmp[0] = 0;\r
+       }\r
+\r
+       // トークン分割\r
+       t = UniParseToken(tmp, L"/\\");\r
+\r
+       sk = NewSk();\r
+\r
+       for (i = 0;i < t->NumTokens;i++)\r
+       {\r
+               wchar_t *s = t->Token[i];\r
+\r
+               if (UniStrCmpi(s, L".") == 0)\r
+               {\r
+                       continue;\r
+               }\r
+               else if (UniStrCmpi(s, L"..") == 0)\r
+               {\r
+                       if (sk->num_item >= 1 && (first_double_slash == false || sk->num_item >= 2))\r
+                       {\r
+                               Pop(sk);\r
+                       }\r
+               }\r
+               else\r
+               {\r
+                       Push(sk, s);\r
+               }\r
+       }\r
+\r
+       // トークン結合\r
+       UniStrCpy(tmp, sizeof(tmp), L"");\r
+\r
+       if (first_double_slash)\r
+       {\r
+               UniStrCat(tmp, sizeof(tmp), L"//");\r
+       }\r
+       else if (first_single_slash)\r
+       {\r
+               UniStrCat(tmp, sizeof(tmp), L"/");\r
+       }\r
+\r
+       if (win32_drive_char != 0)\r
+       {\r
+               wchar_t d[2];\r
+               d[0] = win32_drive_char;\r
+               d[1] = 0;\r
+               UniStrCat(tmp, sizeof(tmp), d);\r
+               UniStrCat(tmp, sizeof(tmp), L":/");\r
+       }\r
+\r
+       for (i = 0;i < sk->num_item;i++)\r
+       {\r
+               UniStrCat(tmp, sizeof(tmp), (wchar_t *)sk->p[i]);\r
+               if (i != (sk->num_item - 1))\r
+               {\r
+                       UniStrCat(tmp, sizeof(tmp), L"/");\r
+               }\r
+       }\r
+\r
+       ReleaseSk(sk);\r
+\r
+       UniFreeToken(t);\r
+\r
+       ConvertPathW(tmp);\r
+\r
+       UniStrCpy(dst, size, tmp);\r
+}\r
+void NormalizePath(char *dst, UINT size, char *src)\r
+{\r
+       wchar_t dst_w[MAX_SIZE];\r
+       wchar_t *src_w = CopyStrToUni(src);\r
+\r
+       NormalizePathW(dst_w, sizeof(dst_w), src_w);\r
+\r
+       Free(src_w);\r
+\r
+       UniToStr(dst, size, dst_w);\r
+}\r
+\r
+// ファイルを閉じて削除する\r
+void FileCloseAndDelete(IO *o)\r
+{\r
+       wchar_t *name;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       name = CopyUniStr(o->NameW);\r
+       FileClose(o);\r
+\r
+       FileDeleteW(name);\r
+\r
+       Free(name);\r
+}\r
+\r
+// ファイル名の変更\r
+bool FileRename(char *old_name, char *new_name)\r
+{\r
+       wchar_t *old_name_w = CopyStrToUni(old_name);\r
+       wchar_t *new_name_w = CopyStrToUni(new_name);\r
+       bool ret = FileRenameW(old_name_w, new_name_w);\r
+\r
+       Free(old_name_w);\r
+       Free(new_name_w);\r
+\r
+       return ret;\r
+}\r
+bool FileRenameW(wchar_t *old_name, wchar_t *new_name)\r
+{\r
+       wchar_t tmp1[MAX_SIZE];\r
+       wchar_t tmp2[MAX_SIZE];\r
+       // 引数チェック\r
+       if (old_name == NULL || new_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       InnerFilePathW(tmp1, sizeof(tmp1), old_name);\r
+       InnerFilePathW(tmp2, sizeof(tmp2), new_name);\r
+\r
+       return FileRenameInnerW(tmp1, tmp2);\r
+}\r
+bool FileRenameInner(char *old_name, char *new_name)\r
+{\r
+       wchar_t *old_name_w = CopyStrToUni(old_name);\r
+       wchar_t *new_name_w = CopyStrToUni(new_name);\r
+       bool ret = FileRenameInnerW(old_name_w, new_name_w);\r
+\r
+       Free(old_name_w);\r
+       Free(new_name_w);\r
+\r
+       return ret;\r
+}\r
+bool FileRenameInnerW(wchar_t *old_name, wchar_t *new_name)\r
+{\r
+       // 引数チェック\r
+       if (old_name == NULL || new_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return OSFileRenameW(old_name, new_name);\r
+}\r
+\r
+// パスの変換\r
+void ConvertPath(char *path)\r
+{\r
+       UINT i, len;\r
+#ifdef PATH_BACKSLASH\r
+       char new_char = '\\';\r
+#else\r
+       char new_char = '/';\r
+#endif\r
+\r
+       len = StrLen(path);\r
+       for (i = 0;i < len;i++)\r
+       {\r
+               if (path[i] == '\\' || path[i] == '/')\r
+               {\r
+                       path[i] = new_char;\r
+               }\r
+       }\r
+}\r
+void ConvertPathW(wchar_t *path)\r
+{\r
+       UINT i, len;\r
+#ifdef PATH_BACKSLASH\r
+       wchar_t new_char = L'\\';\r
+#else\r
+       wchar_t new_char = L'/';\r
+#endif\r
+\r
+       len = UniStrLen(path);\r
+       for (i = 0;i < len;i++)\r
+       {\r
+               if (path[i] == L'\\' || path[i] == L'/')\r
+               {\r
+                       path[i] = new_char;\r
+               }\r
+       }\r
+}\r
+\r
+// ディレクトリの削除\r
+bool DeleteDir(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = DeleteDirW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool DeleteDirW(wchar_t *name)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       InnerFilePathW(tmp, sizeof(tmp), name);\r
+\r
+       return DeleteDirInnerW(tmp);\r
+}\r
+bool DeleteDirInner(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = DeleteDirInnerW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool DeleteDirInnerW(wchar_t *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return OSDeleteDirW(name);\r
+}\r
+\r
+// 内部ファイルパスの生成\r
+void InnerFilePathW(wchar_t *dst, UINT size, wchar_t *src)\r
+{\r
+       // 引数チェック\r
+       if (dst == NULL || src == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (src[0] != L'@')\r
+       {\r
+               NormalizePathW(dst, size, src);\r
+       }\r
+       else\r
+       {\r
+               wchar_t dir[MAX_SIZE];\r
+               GetExeDirW(dir, sizeof(dir));\r
+               ConbinePathW(dst, size, dir, &src[1]);\r
+       }\r
+}\r
+void InnerFilePath(char *dst, UINT size, char *src)\r
+{\r
+       wchar_t dst_w[MAX_PATH];\r
+       wchar_t *src_w = CopyStrToUni(src);\r
+\r
+       InnerFilePathW(dst_w, sizeof(dst_w), src_w);\r
+\r
+       Free(src_w);\r
+\r
+       UniToStr(dst, size, dst_w);\r
+}\r
+\r
+// ディレクトリの再帰作成\r
+void MakeDirEx(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+\r
+       MakeDirExW(name_w);\r
+\r
+       Free(name_w);\r
+}\r
+void MakeDirExW(wchar_t *name)\r
+{\r
+       LIST *o;\r
+       wchar_t tmp[MAX_PATH];\r
+       wchar_t tmp2[MAX_PATH];\r
+       UINT i;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       o = NewListFast(NULL);\r
+\r
+       UniStrCpy(tmp, sizeof(tmp), name);\r
+       while (true)\r
+       {\r
+               wchar_t *s = CopyUniStr(tmp);\r
+\r
+               Add(o, s);\r
+\r
+               GetDirNameFromFilePathW(tmp2, sizeof(tmp2), tmp);\r
+\r
+               if (UniStrCmpi(tmp2, tmp) == 0)\r
+               {\r
+                       break;\r
+               }\r
+\r
+               UniStrCpy(tmp, sizeof(tmp), tmp2);\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               UINT j = LIST_NUM(o) - i - 1;\r
+               wchar_t *s = LIST_DATA(o, j);\r
+\r
+               if (UniStrCmpi(s, L"\\") != 0 && UniStrCmpi(s, L"/") != 0)\r
+               {\r
+                       MakeDirW(s);\r
+               }\r
+       }\r
+\r
+       UniFreeStrList(o);\r
+}\r
+\r
+// ディレクトリの作成\r
+bool MakeDir(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = MakeDirW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool MakeDirW(wchar_t *name)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       InnerFilePathW(tmp, sizeof(tmp), name);\r
+\r
+       return MakeDirInnerW(tmp);\r
+}\r
+bool MakeDirInner(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = MakeDirInnerW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool MakeDirInnerW(wchar_t *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return OSMakeDirW(name);\r
+}\r
+\r
+// ファイルの削除\r
+bool FileDelete(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = FileDeleteW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool FileDeleteW(wchar_t *name)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       InnerFilePathW(tmp, sizeof(tmp), name);\r
+\r
+       return FileDeleteInnerW(tmp);\r
+}\r
+bool FileDeleteInner(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       bool ret = FileDeleteInnerW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+bool FileDeleteInnerW(wchar_t *name)\r
+{\r
+       wchar_t name2[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       UniStrCpy(name2, sizeof(name2), name);\r
+       ConvertPathW(name2);\r
+\r
+       return OSFileDeleteW(name2);\r
+}\r
+\r
+// ファイルをシークする\r
+bool FileSeek(IO *o, UINT mode, int offset)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (o->HamMode == false)\r
+       {\r
+               return OSFileSeek(o->pData, mode, offset);\r
+       }\r
+       else\r
+       {\r
+               return false;\r
+       }\r
+}\r
+\r
+// ファイル名を指定してファイルサイズを取得する\r
+UINT FileSizeEx(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       UINT ret = FileSizeExW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+UINT FileSizeExW(wchar_t *name)\r
+{\r
+       IO *io;\r
+       UINT size;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       io = FileOpenW(name, false);\r
+       if (io == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       size = FileSize(io);\r
+\r
+       FileClose(io);\r
+\r
+       return size;\r
+}\r
+\r
+// ファイルサイズを取得する\r
+UINT64 FileSize64(IO *o)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (o->HamMode == false)\r
+       {\r
+               return OSFileSize(o->pData);\r
+       }\r
+       else\r
+       {\r
+               return (UINT64)o->HamBuf->Size;\r
+       }\r
+}\r
+UINT FileSize(IO *o)\r
+{\r
+       UINT64 size = (UINT)(FileSize64(o));\r
+\r
+       if (size >= 4294967296ULL)\r
+       {\r
+               size = 4294967295ULL;\r
+       }\r
+\r
+       return (UINT)size;\r
+}\r
+\r
+// ファイルから読み込む\r
+bool FileRead(IO *o, void *buf, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL || buf == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_IO_READ_COUNT);\r
+       KS_ADD(KS_IO_TOTAL_READ_SIZE, size);\r
+\r
+       if (size == 0)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (o->HamMode == false)\r
+       {\r
+               return OSFileRead(o->pData, buf, size);\r
+       }\r
+       else\r
+       {\r
+               return ReadBuf(o->HamBuf, buf, size) == size ? true : false;\r
+       }\r
+}\r
+\r
+// ファイルに書き込む\r
+bool FileWrite(IO *o, void *buf, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL || buf == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (o->WriteMode == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // KS\r
+       KS_INC(KS_IO_WRITE_COUNT);\r
+       KS_ADD(KS_IO_TOTAL_WRITE_SIZE, size);\r
+\r
+       if (size == 0)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return OSFileWrite(o->pData, buf, size);\r
+}\r
+\r
+// ファイルをフラッシュする\r
+void FileFlush(IO *o)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (o->HamMode)\r
+       {\r
+               return;\r
+       }\r
+\r
+       OSFileFlush(o->pData);\r
+}\r
+\r
+// ファイルを閉じる\r
+void FileClose(IO *o)\r
+{\r
+       FileCloseEx(o, false);\r
+}\r
+void FileCloseEx(IO *o, bool no_flush)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (o->HamMode == false)\r
+       {\r
+               OSFileClose(o->pData, no_flush);\r
+       }\r
+       else\r
+       {\r
+               FreeBuf(o->HamBuf);\r
+       }\r
+       Free(o);\r
+\r
+       // KS\r
+       KS_INC(KS_IO_CLOSE_COUNT);\r
+}\r
+\r
+// ファイルを作成する\r
+IO *FileCreateInner(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       IO *ret = FileCreateInnerW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+IO *FileCreateInnerW(wchar_t *name)\r
+{\r
+       IO *o;\r
+       void *p;\r
+       wchar_t name2[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       UniStrCpy(name2, sizeof(name2), name);\r
+       ConvertPathW(name2);\r
+\r
+       p = OSFileCreateW(name2);\r
+       if (p == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       o = ZeroMalloc(sizeof(IO));\r
+       o->pData = p;\r
+       UniStrCpy(o->NameW, sizeof(o->NameW), name2);\r
+       UniToStr(o->Name, sizeof(o->Name), o->NameW);\r
+       o->WriteMode = true;\r
+\r
+       // KS\r
+       KS_INC(KS_IO_CREATE_COUNT);\r
+\r
+       return o;\r
+}\r
+IO *FileCreate(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       IO *ret = FileCreateW(name_w);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+IO *FileCreateW(wchar_t *name)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       InnerFilePathW(tmp, sizeof(tmp), name);\r
+\r
+       return FileCreateInnerW(tmp);\r
+}\r
+\r
+// ファイルにすべてのデータを書き込む\r
+bool FileWriteAll(char *name, void *data, UINT size)\r
+{\r
+       IO *io;\r
+       // 引数チェック\r
+       if (name == NULL || (data == NULL && size != 0))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       io = FileCreate(name);\r
+\r
+       if (io == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       FileWrite(io, data, size);\r
+\r
+       FileClose(io);\r
+\r
+       return true;\r
+}\r
+bool FileWriteAllW(wchar_t *name, void *data, UINT size)\r
+{\r
+       IO *io;\r
+       // 引数チェック\r
+       if (name == NULL || (data == NULL && size != 0))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       io = FileCreateW(name);\r
+\r
+       if (io == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       FileWrite(io, data, size);\r
+\r
+       FileClose(io);\r
+\r
+       return true;\r
+}\r
+\r
+// ファイルを開く\r
+IO *FileOpenInner(char *name, bool write_mode, bool read_lock)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       IO *ret = FileOpenInnerW(name_w, write_mode, read_lock);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+IO *FileOpenInnerW(wchar_t *name, bool write_mode, bool read_lock)\r
+{\r
+       IO *o;\r
+       void *p;\r
+       wchar_t name2[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       UniStrCpy(name2, sizeof(name2), name);\r
+       ConvertPathW(name2);\r
+\r
+       p = OSFileOpenW(name2, write_mode, read_lock);\r
+       if (p == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       o = ZeroMalloc(sizeof(IO));\r
+       o->pData = p;\r
+       UniStrCpy(o->NameW, sizeof(o->NameW), name2);\r
+       UniToStr(o->Name, sizeof(o->Name), o->NameW);\r
+       o->WriteMode = write_mode;\r
+\r
+       // KS\r
+       KS_INC(KS_IO_OPEN_COUNT);\r
+\r
+       return o;\r
+}\r
+IO *FileOpen(char *name, bool write_mode)\r
+{\r
+       return FileOpenEx(name, write_mode, true);\r
+}\r
+IO *FileOpenW(wchar_t *name, bool write_mode)\r
+{\r
+       return FileOpenExW(name, write_mode, true);\r
+}\r
+IO *FileOpenEx(char *name, bool write_mode, bool read_lock)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+       IO *ret = FileOpenExW(name_w, write_mode, read_lock);\r
+\r
+       Free(name_w);\r
+\r
+       return ret;\r
+}\r
+IO *FileOpenExW(wchar_t *name, bool write_mode, bool read_lock)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       InnerFilePathW(tmp, sizeof(tmp), name);\r
+\r
+       if (name[0] == L'|')\r
+       {\r
+               IO *o = ZeroMalloc(sizeof(IO));\r
+               name++;\r
+               UniStrCpy(o->NameW, sizeof(o->NameW), name);\r
+               UniToStr(o->Name, sizeof(o->Name), o->NameW);\r
+               o->HamMode = true;\r
+               o->HamBuf = ReadHamcoreW(name);\r
+               if (o->HamBuf == NULL)\r
+               {\r
+                       Free(o);\r
+                       return NULL;\r
+               }\r
+               return o;\r
+       }\r
+       else\r
+       {\r
+               return FileOpenInnerW(tmp, write_mode, read_lock);\r
+       }\r
+}\r
+\r
+\r