* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Mayaqua / Mayaqua.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Mayaqua.c b/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Mayaqua.c
new file mode 100644 (file)
index 0000000..eff8e54
--- /dev/null
@@ -0,0 +1,1174 @@
+// 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
+// Mayaqua.c\r
+// Mayaqua Kernel プログラム\r
+\r
+#include <stdio.h>\r
+#include <stdlib.h>\r
+#include <string.h>\r
+#include <wchar.h>\r
+#include <stdarg.h>\r
+#include <locale.h>\r
+#include <time.h>\r
+#include <errno.h>\r
+#include <Mayaqua/Mayaqua.h>\r
+\r
+// グローバル変数\r
+bool g_memcheck;                                                               // メモリチェックの有効化\r
+bool g_debug;                                                                  // デバッグモード\r
+UINT64 kernel_status[NUM_KERNEL_STATUS];               // カーネル状態\r
+UINT64 kernel_status_max[NUM_KERNEL_STATUS];   // カーネル状態 (最大値)\r
+LOCK *kernel_status_lock[NUM_KERNEL_STATUS];   // カーネル状態ロック\r
+BOOL kernel_status_inited = false;                             // カーネル状態初期化フラグ\r
+bool g_little_endian = true;\r
+char *cmdline = NULL;                                                  // コマンドライン\r
+wchar_t *uni_cmdline = NULL;                                   // Unicode コマンドライン\r
+\r
+// スタティック変数\r
+static char *exename = NULL;                                           // EXE ファイル名 (ANSI)\r
+static wchar_t *exename_w = NULL;                                      // EXE ファイル名 (Unicode)\r
+static TOKEN_LIST *cmdline_token = NULL;                       // コマンドライントークン\r
+static UNI_TOKEN_LIST *cmdline_uni_token = NULL;       // コマンドライントークン (Unicode)\r
+static OS_INFO *os_info = NULL;                                                // OS 情報\r
+static bool dot_net_mode = false;\r
+static bool minimal_mode = false;\r
+static UINT last_time_check = 0;\r
+static UINT first_time_check = 0;\r
+static bool is_nt = false;\r
+static bool is_ham_mode = false;\r
+static UINT init_mayaqua_counter = 0;\r
+static bool use_probe = false;\r
+static BUF *probe_buf = NULL;\r
+static LOCK *probe_lock = NULL;\r
+static UINT64 probe_start = 0;\r
+static UINT64 probe_last = 0;\r
+static bool probe_enabled = false;\r
+\r
+// チェックサムを計算する\r
+USHORT CalcChecksum16(void *buf, UINT size)\r
+{\r
+       int sum = 0;\r
+       USHORT *addr = (USHORT *)buf;\r
+       int len = (int)size;\r
+       USHORT *w = addr;\r
+       int nleft = len;\r
+       USHORT answer = 0;\r
+\r
+       while (nleft > 1)\r
+       {\r
+               sum += *w++;\r
+               nleft -= 2;\r
+       }\r
+\r
+       if (nleft == 1)\r
+       {\r
+               *(UCHAR *)(&answer) = *(UCHAR *)w;\r
+               sum += answer;\r
+       }\r
+\r
+       sum = (sum >> 16) + (sum & 0xffff);\r
+       sum += (sum >> 16);\r
+\r
+       answer = ~sum;\r
+\r
+       return answer;\r
+}\r
+\r
+// データ付き Probe の書き込み\r
+void WriteProbeData(char *filename, UINT line, char *str, void *data, UINT size)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       USHORT cs;\r
+\r
+       if (IsProbeEnabled() == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // データのチェックサムをとる\r
+       if (size != 0)\r
+       {\r
+               cs = CalcChecksum16(data, size);\r
+       }\r
+       else\r
+       {\r
+               cs = 0;\r
+       }\r
+\r
+       // 文字列の生成\r
+       snprintf(tmp, sizeof(tmp), "\"%s\" (Size=%5u, Crc=0x%04X)", str, size, cs);\r
+\r
+       WriteProbe(filename, line, tmp);\r
+}\r
+\r
+// Probe の書き込み\r
+void WriteProbe(char *filename, UINT line, char *str)\r
+{\r
+#ifdef OS_WIN32\r
+       char *s;\r
+       char tmp[MAX_SIZE];\r
+       char tmp2[MAX_SIZE];\r
+       UINT64 now = 0;\r
+       UINT64 time;\r
+\r
+       if (IsProbeEnabled() == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       now = MsGetHiResCounter();\r
+\r
+       Lock(probe_lock);\r
+       {\r
+               UINT64 diff;\r
+               \r
+               time = MsGetHiResTimeSpanUSec(now - probe_start);\r
+\r
+               diff = time - probe_last;\r
+\r
+               if (time < probe_last)\r
+               {\r
+                       diff = 0;\r
+               }\r
+\r
+               probe_last = time;\r
+\r
+               ToStr64(tmp, time);\r
+               MakeCharArray2(tmp2, ' ', (UINT)(MIN(12, (int)12 - (int)StrLen(tmp))));\r
+               WriteBuf(probe_buf, tmp2, StrLen(tmp2));\r
+               WriteBuf(probe_buf, tmp, StrLen(tmp));\r
+\r
+               s = " [+";\r
+               WriteBuf(probe_buf, s, StrLen(s));\r
+\r
+               ToStr64(tmp, diff);\r
+               MakeCharArray2(tmp2, ' ', (UINT)(MIN(12, (int)12 - (int)StrLen(tmp))));\r
+               WriteBuf(probe_buf, tmp2, StrLen(tmp2));\r
+               WriteBuf(probe_buf, tmp, StrLen(tmp));\r
+\r
+               s = "] - ";\r
+               WriteBuf(probe_buf, s, StrLen(s));\r
+\r
+               WriteBuf(probe_buf, filename, StrLen(filename));\r
+\r
+               s = "(";\r
+               WriteBuf(probe_buf, s, StrLen(s));\r
+\r
+               ToStr64(tmp, (UINT64)line);\r
+               WriteBuf(probe_buf, tmp, StrLen(tmp));\r
+\r
+               s = "): ";\r
+               WriteBuf(probe_buf, s, StrLen(s));\r
+\r
+               WriteBuf(probe_buf, str, StrLen(str));\r
+\r
+               s = "\r\n";\r
+               WriteBuf(probe_buf, s, StrLen(s));\r
+       }\r
+       Unlock(probe_lock);\r
+#endif // OS_WIN32\r
+}\r
+\r
+// Probe の初期化\r
+void InitProbe()\r
+{\r
+       probe_buf = NewBuf();\r
+       probe_lock = NewLock();\r
+       probe_enabled = false;\r
+\r
+       probe_start = 0;\r
+\r
+#ifdef OS_WIN32\r
+       probe_start = MsGetHiResCounter();\r
+#endif // OS_WIN32\r
+}\r
+\r
+// Probe の解放\r
+void FreeProbe()\r
+{\r
+       if (probe_buf->Size >= 1)\r
+       {\r
+               SYSTEMTIME st;\r
+               char filename[MAX_SIZE];\r
+\r
+               // ファイルにすべて書き出す\r
+               MakeDirEx("@probe_log");\r
+\r
+               LocalTime(&st);\r
+\r
+               snprintf(filename, sizeof(filename), "@probe_log/%04u%02u%02u_%02u%02u%02u.log",\r
+                       st.wYear, st.wMonth, st.wDay, st.wHour, st.wMinute, st.wSecond);\r
+\r
+               DumpBuf(probe_buf, filename);\r
+       }\r
+\r
+       FreeBuf(probe_buf);\r
+       DeleteLock(probe_lock);\r
+}\r
+\r
+// Probe を有効 / 無効にする\r
+void EnableProbe(bool enable)\r
+{\r
+       probe_enabled = enable;\r
+}\r
+\r
+// Probe が有効かどうか取得する\r
+bool IsProbeEnabled()\r
+{\r
+#ifndef        USE_PROBE\r
+       return false;\r
+#else  // USE_PROBE\r
+       return probe_enabled;\r
+#endif // USE_PROBE\r
+}\r
+\r
+// Ham モードに設定する\r
+void SetHamMode()\r
+{\r
+       is_ham_mode = true;\r
+}\r
+\r
+// Ham モードかどうか取得する\r
+bool IsHamMode()\r
+{\r
+       return is_ham_mode;\r
+}\r
+\r
+// 前回の呼び出しから現在までにかかった時間を表示\r
+void TimeCheck()\r
+{\r
+#ifdef OS_WIN32\r
+       UINT now, ret, total;\r
+       now = Win32GetTick();\r
+       if (last_time_check == 0)\r
+       {\r
+               ret = 0;\r
+       }\r
+       else\r
+       {\r
+               ret = now - last_time_check;\r
+       }\r
+       last_time_check = now;\r
+\r
+       if (first_time_check == 0)\r
+       {\r
+               first_time_check = now;\r
+       }\r
+\r
+       total = now - first_time_check;\r
+\r
+       printf(" -- %3.3f / %3.3f\n", (double)ret / 1000.0f, (double)total / 1000.0f);\r
+#endif // OS_WIN32\r
+}\r
+\r
+// IA64 かどうか\r
+bool IsIA64()\r
+{\r
+       if (Is64() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+#ifndef        MAYAQUA_IA_64\r
+       return false;\r
+#else  // MAYAQUA_IA_64\r
+       return true;\r
+#endif // MAYAQUA_IA_64\r
+}\r
+\r
+// x64 かどうか\r
+bool IsX64()\r
+{\r
+       if (Is64() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+#ifndef        MAYAQUA_IA_64\r
+       return true;\r
+#else  // MAYAQUA_IA_64\r
+       return false;\r
+#endif // MAYAQUA_IA_64\r
+}\r
+\r
+// 64bit かどうか\r
+bool Is64()\r
+{\r
+#ifdef CPU_64\r
+       return true;\r
+#else  // CPU_64\r
+       return false;\r
+#endif // CPU_64\r
+}\r
+\r
+// 32bit かどうか\r
+bool Is32()\r
+{\r
+       return Is64() ? false : true;\r
+}\r
+\r
+// .NET モード\r
+void MayaquaDotNetMode()\r
+{\r
+       dot_net_mode = true;\r
+}\r
+\r
+// .NET モードかどうか取得\r
+bool MayaquaIsDotNetMode()\r
+{\r
+       return dot_net_mode;\r
+}\r
+\r
+// エンディアンチェック\r
+void CheckEndian()\r
+{\r
+       unsigned short test;\r
+       UCHAR *buf;\r
+\r
+       test = 0x1234;\r
+       buf = (UCHAR *)&test;\r
+       if (buf[0] == 0x12)\r
+       {\r
+               g_little_endian = false;\r
+       }\r
+       else\r
+       {\r
+               g_little_endian = true;\r
+       }\r
+}\r
+\r
+// 最小モードにする\r
+void MayaquaMinimalMode()\r
+{\r
+       minimal_mode = true;\r
+}\r
+bool MayaquaIsMinimalMode()\r
+{\r
+       return minimal_mode;\r
+}\r
+\r
+// NT かどうか\r
+bool IsNt()\r
+{\r
+       return is_nt;\r
+}\r
+\r
+// Unicode をサポートしているかどうか\r
+bool IsUnicode()\r
+{\r
+#ifdef OS_WIN32\r
+       // Windows\r
+       return IsNt();\r
+#else  // OS_WIN32\r
+       // UNIX\r
+       return true;\r
+#endif // OS_WIN32\r
+}\r
+\r
+// Mayaqua ライブラリの初期化\r
+void InitMayaqua(bool memcheck, bool debug, int argc, char **argv)\r
+{\r
+       wchar_t tmp[MAX_PATH];\r
+\r
+       if ((init_mayaqua_counter++) > 0)\r
+       {\r
+               return;\r
+       }\r
+\r
+       g_memcheck = memcheck;\r
+       g_debug = debug;\r
+       cmdline = NULL;\r
+       if (dot_net_mode == false)\r
+       {\r
+               // .NET モードでこれを呼ぶとなぜか落ちる\r
+               setbuf(stdout, NULL);\r
+       }\r
+\r
+       // NT かどうか取得\r
+#ifdef OS_WIN32\r
+       is_nt = Win32IsNt();\r
+#endif // OS_WIN32\r
+\r
+       // エンディアンチェック\r
+       CheckEndian();\r
+\r
+#ifdef OS_WIN32\r
+       _configthreadlocale(_DISABLE_PER_THREAD_LOCALE);\r
+#endif // OS_WIN32\r
+\r
+       // CRT のロケール情報を日本語にする\r
+       setlocale(LC_ALL, "");\r
+\r
+       // OS の初期化\r
+       OSInit();\r
+\r
+       // 乱数の初期化\r
+       srand((UINT)SystemTime64());\r
+\r
+       tick_manual_lock = NewLock();\r
+\r
+       // FIFO システムの初期化\r
+       InitFifo();\r
+\r
+       // カーネルステータスの初期化\r
+       InitKernelStatus();\r
+\r
+       // トラッキングの初期化\r
+       InitTracking();\r
+\r
+       // スレッドプールの初期化\r
+       InitThreading();\r
+\r
+       // 文字列ライブラリの初期化\r
+       InitStringLibrary();\r
+\r
+       // ロケール情報の初期化\r
+       SetLocale(NULL);\r
+\r
+       // 暗号化ライブラリの初期化\r
+       InitCryptLibrary();\r
+\r
+       // リアルタイムクロックの初期化\r
+       InitTick64();\r
+\r
+       // ネットワーク通信モジュールの初期化\r
+       InitNetwork();\r
+\r
+       // EXE ファイル名の取得の初期化\r
+       InitGetExeName(argc >= 1 ? argv[0] : NULL);\r
+\r
+       // コマンドライン文字列の初期化\r
+       InitCommandLineStr(argc, argv);\r
+\r
+       // OS 情報の初期化\r
+       InitOsInfo();\r
+\r
+       // オペレーティングシステム固有の初期化\r
+#ifdef OS_WIN32\r
+       MsInit();       // Microsoft Win32\r
+#endif // OS_WIN32\r
+\r
+       // セキュリティトークンモジュールの初期化\r
+       InitSecure();\r
+\r
+       if (OSIsSupportedOs() == false)\r
+       {\r
+               // 強制終了\r
+               exit(0);\r
+       }\r
+\r
+       // RSA チェック\r
+       if (RsaCheckEx() == false)\r
+       {\r
+               // 強制終了\r
+               Alert("OpenSSL Library Init Failed. (too old?)\nPlease install the latest version of OpenSSL.\n\n", "RsaCheck()");\r
+               exit(0);\r
+       }\r
+\r
+       // HamCore ファイルシステムの初期化\r
+       InitHamcore();\r
+\r
+       // デフォルトで japanese.stb を読み込む\r
+       LoadTable(DEFAULT_TABLE_FILE_NAME);\r
+\r
+       if (exename == NULL)\r
+       {\r
+               // 実行可能ファイル名\r
+               exename = CopyStr("unknown");\r
+       }\r
+\r
+       // 自分自身の実行可能ファイル名が見つかるかどうか検査する\r
+       // (見つからない場合は変なパスで起動されているので終了する)\r
+       GetExeNameW(tmp, sizeof(tmp));\r
+       if (IsFileExistsW(tmp) == false)\r
+       {\r
+               wchar_t tmp2[MAX_SIZE];\r
+\r
+               UniFormat(tmp2, sizeof(tmp2),\r
+                       L"Error: Executable binary file \"%s\" not found.\r\n\r\n"\r
+                       L"Please execute program with full path.\r\n",\r
+                       tmp);\r
+\r
+               AlertW(tmp2, NULL);\r
+               _exit(0);\r
+       }\r
+\r
+       CheckUnixTempDir();\r
+\r
+       // Probe の初期化\r
+       InitProbe();\r
+}\r
+\r
+// Mayaqua ライブラリの解放\r
+void FreeMayaqua()\r
+{\r
+       if ((--init_mayaqua_counter) > 0)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // Probe の解放\r
+       FreeProbe();\r
+\r
+       // テーブルを削除\r
+       FreeTable();\r
+\r
+       // セキュリティトークンモジュールの解放\r
+       FreeSecure();\r
+\r
+       // オペレーティングシステム固有の解放\r
+#ifdef OS_WIN32\r
+       MsFree();\r
+#endif // OS_WIN32\r
+\r
+       // OS 情報の解放\r
+       FreeOsInfo();\r
+\r
+       // HamCore ファイルシステムの解放\r
+       FreeHamcore();\r
+\r
+       // コマンドライン文字列の解放\r
+       FreeCommandLineStr();\r
+\r
+       // コマンドライントークンの解放\r
+       FreeCommandLineTokens();\r
+\r
+       // ネットワーク通信モジュールの解放\r
+       FreeNetwork();\r
+\r
+       // リアルタイムクロックの解放\r
+       FreeTick64();\r
+\r
+       // 暗号化ライブラリの解放\r
+       FreeCryptLibrary();\r
+\r
+       // 文字列ライブラリの解放\r
+       FreeStringLibrary();\r
+\r
+       // スレッドプールの解放\r
+       FreeThreading();\r
+\r
+#ifndef        VPN_SPEED\r
+       // カーネル状態の表示\r
+       if (g_debug)\r
+       {\r
+               PrintKernelStatus();\r
+       }\r
+\r
+       // デバッグ情報の表示\r
+       if (g_memcheck)\r
+       {\r
+               PrintDebugInformation();\r
+       }\r
+#endif // VPN_SPEED\r
+\r
+       // トラッキングの解放\r
+       FreeTracking();\r
+\r
+       // カーネルステータスの解放\r
+       FreeKernelStatus();\r
+\r
+       DeleteLock(tick_manual_lock);\r
+       tick_manual_lock = NULL;\r
+\r
+       // OS の解放\r
+       OSFree();\r
+}\r
+\r
+// UNIX で /tmp が使用できるかどうか確認する\r
+void CheckUnixTempDir()\r
+{\r
+       if (OS_IS_UNIX(GetOsInfo()->OsType))\r
+       {\r
+               char tmp[128], tmp2[64];\r
+               UINT64 now = SystemTime64();\r
+               IO *o;\r
+\r
+               MakeDir("/tmp");\r
+\r
+               Format(tmp2, sizeof(tmp2), "%I64u", now);\r
+\r
+               Format(tmp, sizeof(tmp), "/tmp/.%s", tmp2);\r
+\r
+               o = FileCreate(tmp);\r
+               if (o == NULL)\r
+               {\r
+                       o = FileOpen(tmp, false);\r
+                       if (o == NULL)\r
+                       {\r
+                               Print("Unable to use /tmp.\n\n");\r
+                               exit(0);\r
+                               return;\r
+                       }\r
+               }\r
+\r
+               FileClose(o);\r
+\r
+               FileDelete(tmp);\r
+       }\r
+}\r
+\r
+// アラートの表示\r
+void Alert(char *msg, char *caption)\r
+{\r
+       OSAlert(msg, caption);\r
+}\r
+void AlertW(wchar_t *msg, wchar_t *caption)\r
+{\r
+       OSAlertW(msg, caption);\r
+}\r
+\r
+// OS 情報の表示\r
+void PrintOsInfo(OS_INFO *info)\r
+{\r
+       // 引数チェック\r
+       if (info == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Print(\r
+               "OS Type          : %u\n"\r
+               "OS Service Pack  : %u\n"\r
+               "os_is_windows    : %s\n"\r
+               "os_is_windows_nt : %s\n"\r
+               "OS System Name   : %s\n"\r
+               "OS Product Name  : %s\n"\r
+               "OS Vendor Name   : %s\n"\r
+               "OS Version       : %s\n"\r
+               "Kernel Name      : %s\n"\r
+               "Kernel Version   : %s\n",\r
+               info->OsType,\r
+               info->OsServicePack,\r
+               OS_IS_WINDOWS(info->OsType) ? "true" : "false",\r
+               OS_IS_WINDOWS_NT(info->OsType) ? "true" : "false",\r
+               info->OsSystemName,\r
+               info->OsProductName,\r
+               info->OsVendorName,\r
+               info->OsVersion,\r
+               info->KernelName,\r
+               info->KernelVersion);\r
+\r
+#ifdef OS_WIN32\r
+       {\r
+               char *exe, *dir;\r
+               exe = MsGetExeFileName();\r
+               dir = MsGetExeDirName();\r
+\r
+               Print(\r
+                       "EXE File Path    : %s\n"\r
+                       "EXE Dir Path     : %s\n"\r
+                       "Process Id       : %u\n"\r
+                       "Process Handle   : 0x%X\n",\r
+                       exe, dir, MsGetCurrentProcessId(), MsGetCurrentProcess());\r
+       }\r
+#endif // OS_WIN32\r
+}\r
+\r
+// OS 種類の取得\r
+UINT GetOsType()\r
+{\r
+       OS_INFO *i = GetOsInfo();\r
+\r
+       if (i == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return i->OsType;\r
+}\r
+\r
+// OS 情報の取得\r
+OS_INFO *GetOsInfo()\r
+{\r
+       return os_info;\r
+}\r
+\r
+// OS 情報の初期化\r
+void InitOsInfo()\r
+{\r
+       if (os_info != NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       os_info = ZeroMalloc(sizeof(OS_INFO));\r
+\r
+       OSGetOsInfo(os_info);\r
+}\r
+\r
+// OS 情報の解放\r
+void FreeOsInfo()\r
+{\r
+       if (os_info == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Free(os_info->OsSystemName);\r
+       Free(os_info->OsProductName);\r
+       Free(os_info->OsVendorName);\r
+       Free(os_info->OsVersion);\r
+       Free(os_info->KernelName);\r
+       Free(os_info->KernelVersion);\r
+       Free(os_info);\r
+\r
+       os_info = NULL;\r
+}\r
+\r
+// Unicode コマンドライントークンの取得\r
+UNI_TOKEN_LIST *GetCommandLineUniToken()\r
+{\r
+       if (cmdline_uni_token == NULL)\r
+       {\r
+               return UniNullToken();\r
+       }\r
+       else\r
+       {\r
+               return UniCopyToken(cmdline_uni_token);\r
+       }\r
+}\r
+\r
+// コマンドライントークンの取得\r
+TOKEN_LIST *GetCommandLineToken()\r
+{\r
+       if (cmdline_token == NULL)\r
+       {\r
+               return NullToken();\r
+       }\r
+       else\r
+       {\r
+               return CopyToken(cmdline_token);\r
+       }\r
+}\r
+\r
+// コマンドライン文字列をトークンに変換する\r
+void ParseCommandLineTokens()\r
+{\r
+       if (cmdline_token != NULL)\r
+       {\r
+               FreeToken(cmdline_token);\r
+       }\r
+       cmdline_token = ParseCmdLine(cmdline);\r
+\r
+       if (cmdline_uni_token != NULL)\r
+       {\r
+               UniFreeToken(cmdline_uni_token);\r
+       }\r
+       cmdline_uni_token = UniParseCmdLine(uni_cmdline);\r
+}\r
+\r
+// コマンドライントークンを解放する\r
+void FreeCommandLineTokens()\r
+{\r
+       if (cmdline_token != NULL)\r
+       {\r
+               FreeToken(cmdline_token);\r
+       }\r
+       cmdline_token = NULL;\r
+\r
+       if (cmdline_uni_token != NULL)\r
+       {\r
+               UniFreeToken(cmdline_uni_token);\r
+       }\r
+       cmdline_uni_token = NULL;\r
+}\r
+\r
+// コマンドライン文字列の初期化\r
+void InitCommandLineStr(int argc, char **argv)\r
+{\r
+       if (argc >= 1)\r
+       {\r
+#ifdef OS_UNIX\r
+               exename_w = CopyUtfToUni(argv[0]);\r
+               exename = CopyUniToStr(exename_w);\r
+#else  // OS_UNIX\r
+               exename = CopyStr(argv[0]);\r
+               exename_w = CopyStrToUni(exename);\r
+#endif // OS_UNIX\r
+       }\r
+       if (argc < 2 || argv == NULL)\r
+       {\r
+               // コマンドライン文字列無し\r
+               SetCommandLineStr(NULL);\r
+       }\r
+       else\r
+       {\r
+               // コマンドライン文字列有り\r
+               int i, total_len = 1;\r
+               char *tmp;\r
+\r
+               for (i = 1;i < argc;i++)\r
+               {\r
+                       total_len += StrLen(argv[i]) * 2 + 32;\r
+               }\r
+               tmp = ZeroMalloc(total_len);\r
+\r
+               for (i = 1;i < argc;i++)\r
+               {\r
+                       UINT s_size = StrLen(argv[i]) * 2;\r
+                       char *s = ZeroMalloc(s_size);\r
+                       bool dq = (SearchStrEx(argv[i], " ", 0, true) != INFINITE);\r
+                       ReplaceStrEx(s, s_size, argv[i], "\"", "\"\"", true);\r
+                       if (dq)\r
+                       {\r
+                               StrCat(tmp, total_len, "\"");\r
+                       }\r
+                       StrCat(tmp, total_len, s);\r
+                       if (dq)\r
+                       {\r
+                               StrCat(tmp, total_len, "\"");\r
+                       }\r
+                       StrCat(tmp, total_len, " ");\r
+                       Free(s);\r
+               }\r
+\r
+               Trim(tmp);\r
+               SetCommandLineStr(tmp);\r
+               Free(tmp);\r
+       }\r
+}\r
+\r
+// コマンドライン文字列の解放\r
+void FreeCommandLineStr()\r
+{\r
+       SetCommandLineStr(NULL);\r
+\r
+       if (exename != NULL)\r
+       {\r
+               Free(exename);\r
+               exename = NULL;\r
+       }\r
+\r
+       if (exename_w != NULL)\r
+       {\r
+               Free(exename_w);\r
+               exename_w = NULL;\r
+       }\r
+}\r
+\r
+// Unicode コマンドライン文字列の取得\r
+wchar_t *GetCommandLineUniStr()\r
+{\r
+       if (uni_cmdline == NULL)\r
+       {\r
+               return UniCopyStr(L"");\r
+       }\r
+       else\r
+       {\r
+               return UniCopyStr(uni_cmdline);\r
+       }\r
+}\r
+\r
+// コマンドライン文字列の取得\r
+char *GetCommandLineStr()\r
+{\r
+       if (cmdline == NULL)\r
+       {\r
+               return CopyStr("");\r
+       }\r
+       else\r
+       {\r
+               return CopyStr(cmdline);\r
+       }\r
+}\r
+\r
+// Unicode コマンドライン文字列の設定\r
+void SetCommandLineUniStr(wchar_t *str)\r
+{\r
+       if (uni_cmdline != NULL)\r
+       {\r
+               Free(uni_cmdline);\r
+       }\r
+       if (str == NULL)\r
+       {\r
+               uni_cmdline = NULL;\r
+       }\r
+       else\r
+       {\r
+               uni_cmdline = CopyUniStr(str);\r
+       }\r
+\r
+       ParseCommandLineTokens();\r
+}\r
+\r
+// コマンドライン文字列の設定\r
+void SetCommandLineStr(char *str)\r
+{\r
+       // 引数チェック\r
+       if (str == NULL)\r
+       {\r
+               if (cmdline != NULL)\r
+               {\r
+                       Free(cmdline);\r
+               }\r
+               cmdline = NULL;\r
+       }\r
+       else\r
+       {\r
+               if (cmdline != NULL)\r
+               {\r
+                       Free(cmdline);\r
+               }\r
+               cmdline = CopyStr(str);\r
+       }\r
+\r
+       if (cmdline == NULL)\r
+       {\r
+               if (uni_cmdline != NULL)\r
+               {\r
+                       Free(uni_cmdline);\r
+                       uni_cmdline = NULL;\r
+               }\r
+       }\r
+       else\r
+       {\r
+               if (uni_cmdline != NULL)\r
+               {\r
+                       Free(uni_cmdline);\r
+               }\r
+               uni_cmdline = CopyStrToUni(cmdline);\r
+       }\r
+\r
+       ParseCommandLineTokens();\r
+}\r
+\r
+// カーネルステータスの表示\r
+void PrintKernelStatus()\r
+{\r
+       bool leaked = false;\r
+\r
+       Print("\n");\r
+       Print(\r
+               "     --------- Mayaqua Kernel Status ---------\n"\r
+               "        Malloc Count ............... %u\n"\r
+               "        ReAlloc Count .............. %u\n"\r
+               "        Free Count ................. %u\n"\r
+               "        Total Memory Size .......... %I64u bytes\n"\r
+               "      * Current Memory Blocks ...... %u Blocks (Peek: %u)\n"\r
+               "        Total Memory Blocks ........ %u Blocks\n"\r
+               "      * Current MemPool Blocks ..... %u Blocks (Peek: %u)\n"\r
+               "        Total MemPool Mallocs ...... %u Mallocs\n"\r
+               "        Total MemPool ReAllocs ..... %u ReAllocs\n"\r
+               "        NewLock Count .............. %u\n"\r
+               "        DeleteLock Count ........... %u\n"\r
+               "      * Current Lock Objects ....... %u Objects\n"\r
+               "      * Current Locked Objects ..... %u Objects\n"\r
+               "        NewRef Count ............... %u\n"\r
+               "        FreeRef Count .............. %u\n"\r
+               "      * Current Ref Objects ........ %u Objects\n"\r
+               "      * Current Ref Count .......... %u Refs\n"\r
+               "        GetTime Count .............. %u\n"\r
+               "        GetTick Count .............. %u\n"\r
+               "        NewThread Count ............ %u\n"\r
+               "        FreeThread Count ........... %u\n"\r
+               "      * Current Threads ............ %u Threads\n"\r
+               "        Wait For Event Count ....... %u\n\n",\r
+               KS_GET(KS_MALLOC_COUNT),\r
+               KS_GET(KS_REALLOC_COUNT),\r
+               KS_GET(KS_FREE_COUNT),\r
+               KS_GET64(KS_TOTAL_MEM_SIZE),\r
+               KS_GET(KS_CURRENT_MEM_COUNT),\r
+               KS_GETMAX(KS_CURRENT_MEM_COUNT),\r
+               KS_GET(KS_TOTAL_MEM_COUNT),\r
+               KS_GET(KS_MEMPOOL_CURRENT_NUM),\r
+               KS_GETMAX(KS_MEMPOOL_CURRENT_NUM),\r
+               KS_GET(KS_MEMPOOL_MALLOC_COUNT),\r
+               KS_GET(KS_MEMPOOL_REALLOC_COUNT),\r
+               KS_GET(KS_NEWLOCK_COUNT),\r
+               KS_GET(KS_DELETELOCK_COUNT),\r
+               KS_GET(KS_CURRENT_LOCK_COUNT),\r
+               KS_GET(KS_CURRENT_LOCKED_COUNT),\r
+               KS_GET(KS_NEWREF_COUNT),\r
+               KS_GET(KS_FREEREF_COUNT),\r
+               KS_GET(KS_CURRENT_REF_COUNT),\r
+               KS_GET(KS_CURRENT_REFED_COUNT),\r
+               KS_GET(KS_GETTIME_COUNT),\r
+               KS_GET(KS_GETTICK_COUNT),\r
+               KS_GET(KS_NEWTHREAD_COUNT),\r
+               KS_GET(KS_FREETHREAD_COUNT),\r
+               KS_GET(KS_NEWTHREAD_COUNT) - KS_GET(KS_FREETHREAD_COUNT),\r
+               KS_GET(KS_WAIT_COUNT)\r
+               );\r
+\r
+       if (KS_GET(KS_CURRENT_MEM_COUNT) != 0 || KS_GET(KS_CURRENT_LOCK_COUNT) != 0 ||\r
+               KS_GET(KS_MEMPOOL_CURRENT_NUM) != 0 ||\r
+               KS_GET(KS_CURRENT_LOCKED_COUNT) != 0 || KS_GET(KS_CURRENT_REF_COUNT) != 0)\r
+       {\r
+               leaked = true;\r
+       }\r
+\r
+       if (leaked)\r
+       {\r
+               Print("      !!! MEMORY LEAKS DETECTED !!!\n\n");\r
+               if (g_memcheck == false)\r
+               {\r
+                       GetLine(NULL, 0);\r
+               }\r
+       }\r
+       else\r
+       {\r
+               Print("        @@@ NO MEMORY LEAKS @@@\n\n");\r
+       }\r
+}\r
+\r
+// カーネルステータスの初期化\r
+void InitKernelStatus()\r
+{\r
+       UINT i;\r
+\r
+       // メモリ初期化\r
+       Zero(kernel_status, sizeof(kernel_status));\r
+       Zero(kernel_status_max, sizeof(kernel_status_max));\r
+\r
+       // ロック初期化\r
+       for (i = 0;i < NUM_KERNEL_STATUS;i++)\r
+       {\r
+               kernel_status_lock[i] = OSNewLock();\r
+       }\r
+\r
+       kernel_status_inited = true;\r
+}\r
+\r
+// カーネルステータスの解放\r
+void FreeKernelStatus()\r
+{\r
+       UINT i;\r
+\r
+       kernel_status_inited = false;\r
+\r
+       // ロック解放\r
+       for (i = 0;i < NUM_KERNEL_STATUS;i++)\r
+       {\r
+               OSDeleteLock(kernel_status_lock[i]);\r
+       }\r
+}\r
+\r
+// カーネルステータスのロック\r
+void LockKernelStatus(UINT id)\r
+{\r
+       // 引数チェック\r
+       if (id >= NUM_KERNEL_STATUS)\r
+       {\r
+               return;\r
+       }\r
+\r
+       OSLock(kernel_status_lock[id]);\r
+}\r
+\r
+// カーネルステータスのロック解除\r
+void UnlockKernelStatus(UINT id)\r
+{\r
+       // 引数チェック\r
+       if (id >= NUM_KERNEL_STATUS)\r
+       {\r
+               return;\r
+       }\r
+\r
+       OSUnlock(kernel_status_lock[id]);\r
+}\r
+\r
+// デバッグ情報の表示\r
+void PrintDebugInformation()\r
+{\r
+       MEMORY_STATUS memory_status;\r
+       GetMemoryStatus(&memory_status);\r
+\r
+       // ヘッダ\r
+       Print("====== SoftEther UT-VPN System Debug Information ======\n");\r
+\r
+       // メモリ情報\r
+       Print(" <Memory Status>\n"\r
+               "       Number of Allocated Memory Blocks: %u\n"\r
+               "   Total Size of Allocated Memory Blocks: %u bytes\n",\r
+               memory_status.MemoryBlocksNum, memory_status.MemorySize);\r
+\r
+       // フッタ\r
+       Print("====================================================\n");\r
+\r
+       if (KS_GET(KS_CURRENT_MEM_COUNT) != 0 || KS_GET(KS_CURRENT_LOCK_COUNT) != 0 ||\r
+               KS_GET(KS_CURRENT_LOCKED_COUNT) != 0 || KS_GET(KS_CURRENT_REF_COUNT) != 0)\r
+       {\r
+               // メモリリークしている可能性があるのでデバッグメニューを出す\r
+               MemoryDebugMenu();\r
+       }\r
+}\r
+\r
+\r
+\r
+\r