* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Mayaqua / Microsoft.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Microsoft.c b/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Microsoft.c
new file mode 100644 (file)
index 0000000..f9d1ec8
--- /dev/null
@@ -0,0 +1,12112 @@
+// 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
+// Microsoft.c\r
+// Microsoft Windows 用コード\r
+// (Windows 以外の環境ではコンパイルされない)\r
+\r
+#ifdef WIN32\r
+\r
+#define        MICROSOFT_C\r
+\r
+typedef enum    _PNP_VETO_TYPE {\r
+    PNP_VetoTypeUnknown,            // Name is unspecified\r
+    PNP_VetoLegacyDevice,           // Name is an Instance Path\r
+    PNP_VetoPendingClose,           // Name is an Instance Path\r
+    PNP_VetoWindowsApp,             // Name is a Module\r
+    PNP_VetoWindowsService,         // Name is a Service\r
+    PNP_VetoOutstandingOpen,        // Name is an Instance Path\r
+    PNP_VetoDevice,                 // Name is an Instance Path\r
+    PNP_VetoDriver,                 // Name is a Driver Service Name\r
+    PNP_VetoIllegalDeviceRequest,   // Name is an Instance Path\r
+    PNP_VetoInsufficientPower,      // Name is unspecified\r
+    PNP_VetoNonDisableable,         // Name is an Instance Path\r
+    PNP_VetoLegacyDriver,           // Name is a Service\r
+    PNP_VetoInsufficientRights      // Name is unspecified\r
+}   PNP_VETO_TYPE, *PPNP_VETO_TYPE;\r
+\r
+#define        _WIN32_IE                       0x0600\r
+#define        _WIN32_WINNT            0x0502\r
+#define        WINVER                          0x0502\r
+#define   SECURITY_WIN32\r
+#include <winsock2.h>\r
+#include <windows.h>\r
+#include <Wintrust.h>\r
+#include <Softpub.h>\r
+#include <Iphlpapi.h>\r
+#include <tlhelp32.h>\r
+#include <wincon.h>\r
+#include <Nb30.h>\r
+#include <shlobj.h>\r
+#include <commctrl.h>\r
+#include <Dbghelp.h>\r
+#include <setupapi.h>\r
+#include <regstr.h>\r
+#include <process.h>\r
+#include <psapi.h>\r
+#include <wtsapi32.h>\r
+#include <security.h>\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
+#include <cfgmgr32.h>\r
+#include <sddl.h>\r
+#include <Aclapi.h>\r
+\r
+\r
+static MS *ms = NULL;\r
+\r
+// 関数プロトタイプ\r
+UINT MsgBox(HWND hWnd, UINT flag, wchar_t *msg);\r
+UINT MsgBoxEx(HWND hWnd, UINT flag, wchar_t *msg, ...);\r
+void ShowTcpIpConfigUtil(HWND hWnd, bool util_mode);\r
+void CmTraffic(HWND hWnd);\r
+void CnStart();\r
+void InitCedar();\r
+void FreeCedar();\r
+void InitWinUi(wchar_t *software_name, char *font, UINT fontsize);\r
+void FreeWinUi();\r
+\r
+// グローバル変数\r
+void *ms_critical_section = NULL;\r
+UINT64 ms_uint64_1 = 0;\r
+\r
+// アダプタリスト関係\r
+static LOCK *lock_adapter_list = NULL;\r
+static MS_ADAPTER_LIST *last_adapter_list = NULL;\r
+\r
+// サービス関係\r
+static SERVICE_STATUS_HANDLE ssh = NULL;\r
+static SERVICE_STATUS status;\r
+static char g_service_name[MAX_SIZE];\r
+static SERVICE_FUNCTION *g_start, *g_stop;\r
+static bool exiting = false;\r
+static bool wnd_end;\r
+static bool is_usermode = false;\r
+static HICON tray_icon;\r
+static NOTIFYICONDATA nid;\r
+static NOTIFYICONDATAW nid_nt;\r
+static bool service_for_9x_mode = false;\r
+static THREAD *starter_thread = NULL;\r
+static EVENT *server_stopped_event = NULL;\r
+static THREAD *service_stopper_thread = NULL;\r
+static bool tray_inited = false;\r
+static HWND hWndUsermode = NULL;\r
+\r
+// [ネットワーク接続] を開くためのショートカット (最新版では未使用)\r
+static UCHAR network_connection_link[] =\r
+{\r
+       0x4C, 0x00, 0x00, 0x00, 0x01, 0x14, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0xC0, 0x00, 0x00, 0x00, \r
+       0x00, 0x00, 0x00, 0x46, 0x81, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \r
+       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \r
+       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, \r
+       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3E, 0x00, 0x14, 0x00, \r
+       0x1F, 0x50, 0xE0, 0x4F, 0xD0, 0x20, 0xEA, 0x3A, 0x69, 0x10, 0xA2, 0xD8, 0x08, 0x00, 0x2B, 0x30, \r
+       0x30, 0x9D, 0x14, 0x00, 0x2E, 0x00, 0x20, 0x20, 0xEC, 0x21, 0xEA, 0x3A, 0x69, 0x10, 0xA2, 0xDD, \r
+       0x08, 0x00, 0x2B, 0x30, 0x30, 0x9D, 0x14, 0x00, 0x70, 0x00, 0xC7, 0xAC, 0x07, 0x70, 0x02, 0x32, \r
+       0xD1, 0x11, 0xAA, 0xD2, 0x00, 0x80, 0x5F, 0xC1, 0x27, 0x0E, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, \r
+};\r
+\r
+// Windows Vista 関係の新しいセキュリティ構造体等\r
+#if    0\r
+typedef struct _TOKEN_MANDATORY_LABEL\r
+{\r
+       SID_AND_ATTRIBUTES Label;\r
+} TOKEN_MANDATORY_LABEL, *PTOKEN_MANDATORY_LABEL;\r
+#endif\r
+\r
+#define SE_GROUP_INTEGRITY                 (0x00000020L)\r
+\r
+typedef enum _TOKEN_INFORMATION_CLASS_VISTA\r
+{\r
+       VistaTokenUser = 1,\r
+       VistaTokenGroups,\r
+       VistaTokenPrivileges,\r
+       VistaTokenOwner,\r
+       VistaTokenPrimaryGroup,\r
+       VistaTokenDefaultDacl,\r
+       VistaTokenSource,\r
+       VistaTokenType,\r
+       VistaTokenImpersonationLevel,\r
+       VistaTokenStatistics,\r
+       VistaTokenRestrictedSids,\r
+       VistaTokenSessionId,\r
+       VistaTokenGroupsAndPrivileges,\r
+       VistaTokenSessionReference,\r
+       VistaTokenSandBoxInert,\r
+       VistaTokenAuditPolicy,\r
+       VistaTokenOrigin,\r
+       VistaTokenElevationType,\r
+       VistaTokenLinkedToken,\r
+       VistaTokenElevation,\r
+       VistaTokenHasRestrictions,\r
+       VistaTokenAccessInformation,\r
+       VistaTokenVirtualizationAllowed,\r
+       VistaTokenVirtualizationEnabled,\r
+       VistaTokenIntegrityLevel,\r
+       VistaTokenUIAccess,\r
+       VistaTokenMandatoryPolicy,\r
+       VistaTokenLogonSid,\r
+       VistaMaxTokenInfoClass\r
+} TOKEN_INFORMATION_CLASS_VISTA, *PTOKEN_INFORMATION_CLASS_VISTA;\r
+\r
+// エラーを表示しないモードにする\r
+void MsSetErrorModeToSilent()\r
+{\r
+       SetErrorMode(SEM_FAILCRITICALERRORS | SEM_NOGPFAULTERRORBOX | SEM_NOOPENFILEERRORBOX);\r
+}\r
+\r
+// ファイル情報の取得\r
+bool MsGetFileInformation(void *h, void *info)\r
+{\r
+       // 引数チェック\r
+       if (h == INVALID_HANDLE_VALUE || info == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (ms->nt->GetFileInformationByHandle == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return ms->nt->GetFileInformationByHandle(h, info);\r
+}\r
+\r
+// プロセスのシャットダウンパラメータの設定\r
+void MsSetShutdownParameters(UINT level, UINT flag)\r
+{\r
+       if (MsIsNt() == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (ms->nt == false || ms->nt->SetProcessShutdownParameters == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       ms->nt->SetProcessShutdownParameters(level, flag);\r
+}\r
+\r
+// OS のバージョンが Windows XP または Windows Vista 以降かどうか取得する\r
+bool MsIsWinXPOrWinVista()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+       if (info == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (OS_IS_WINDOWS_NT(info->OsType) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) >= 3)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// イベントログに書き込む\r
+bool MsWriteEventLog(void *p, UINT type, wchar_t *str)\r
+{\r
+       MS_EVENTLOG *g = (MS_EVENTLOG *)p;\r
+       wchar_t *strings[2];\r
+       UINT id = 0;\r
+       UINT typeapi = 0;\r
+       // 引数チェック\r
+       if (g == NULL || type >= 5 || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       strings[0] = str;\r
+\r
+       switch (type)\r
+       {\r
+       case MS_EVENTLOG_TYPE_INFORMATION:\r
+               id = MS_RC_EVENTLOG_TYPE_INFORMATION;\r
+               typeapi = EVENTLOG_INFORMATION_TYPE;\r
+               break;\r
+\r
+       case MS_EVENTLOG_TYPE_WARNING:\r
+               id = MS_RC_EVENTLOG_TYPE_WARNING;\r
+               typeapi = EVENTLOG_WARNING_TYPE;\r
+               break;\r
+\r
+       case MS_EVENTLOG_TYPE_ERROR:\r
+               id = MS_RC_EVENTLOG_TYPE_ERROR;\r
+               typeapi = EVENTLOG_ERROR_TYPE;\r
+               break;\r
+       }\r
+\r
+       return ms->nt->ReportEventW(g->hEventLog, typeapi, 0, id, NULL, 1, 0, strings, NULL);\r
+}\r
+\r
+// イベントログの解放\r
+void MsFreeEventLog(void *p)\r
+{\r
+       MS_EVENTLOG *g = (MS_EVENTLOG *)p;\r
+       // 引数チェック\r
+       if (g == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       ms->nt->DeregisterEventSource(g->hEventLog);\r
+\r
+       Free(g);\r
+}\r
+\r
+// イベントログの初期化\r
+void *MsInitEventLog(wchar_t *src_name)\r
+{\r
+       MS_EVENTLOG *g;\r
+       HANDLE h;\r
+       wchar_t keyname[MAX_PATH];\r
+       char keyname_a[MAX_PATH];\r
+       wchar_t *exename;\r
+       // 引数チェック\r
+       if (src_name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // レジストリにキーを書き込む\r
+       exename = MsGetExeFileNameW();\r
+       UniFormat(keyname, sizeof(keyname),\r
+               L"SYSTEM\\CurrentControlSet\\Services\\Eventlog\\Application\\%s",\r
+               src_name);\r
+       UniToStr(keyname_a, sizeof(keyname_a), keyname);\r
+\r
+       MsRegWriteStrExpandExW(REG_LOCAL_MACHINE, keyname_a, "EventMessageFile",\r
+               exename, false);\r
+\r
+       MsRegWriteIntEx(REG_LOCAL_MACHINE, keyname_a, "TypesSupported", 7, false);\r
+\r
+       h = ms->nt->RegisterEventSourceW(NULL, src_name);\r
+       if (h == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       g = ZeroMalloc(sizeof(MS_EVENTLOG));\r
+\r
+       g->hEventLog = h;\r
+\r
+       return (void *)g;\r
+}\r
+\r
+// クリップボードを空にする\r
+void MsDeleteClipboard()\r
+{\r
+       OpenClipboard(NULL);\r
+\r
+       EmptyClipboard();\r
+\r
+       CloseClipboard();\r
+}\r
+\r
+// クリップボード所有者のプロセス ID を取得する\r
+UINT MsGetClipboardOwnerProcessId()\r
+{\r
+       HWND hWnd = GetClipboardOwner();\r
+       DWORD pid = 0;\r
+\r
+       if (hWnd == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       GetWindowThreadProcessId(hWnd, &pid);\r
+\r
+       return pid;\r
+}\r
+\r
+// MMCSS の再起動\r
+// 注意: この実装は完璧ではない。\r
+void MsRestartMMCSS()\r
+{\r
+       MsStopService("CTAudSvcService");\r
+       MsStopService("audiosrv");\r
+       MsStopService("MMCSS");\r
+       MsStartService("MMCSS");\r
+       MsStartService("audiosrv");\r
+       MsStartService("CTAudSvcService");\r
+}\r
+\r
+// MMCSS によるネットワークスロットリングを有効 / 無効にする\r
+void MsSetMMCSSNetworkThrottlingEnable(bool enable)\r
+{\r
+       UINT value;\r
+       if (MsIsVista() == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (enable)\r
+       {\r
+               value = 0x0000000a;\r
+       }\r
+       else\r
+       {\r
+               value = 0xffffffff;\r
+       }\r
+\r
+       MsRegWriteIntEx2(REG_LOCAL_MACHINE, MMCSS_PROFILE_KEYNAME, "NetworkThrottlingIndex",\r
+               value,\r
+               false, true);\r
+\r
+       MsRestartMMCSS();\r
+}\r
+\r
+// MMCSS によるネットワークスロットリングが有効になっているかどうか調査\r
+bool MsIsMMCSSNetworkThrottlingEnabled()\r
+{\r
+       UINT value;\r
+       if (MsIsVista() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsRegIsKeyEx2(REG_LOCAL_MACHINE, MMCSS_PROFILE_KEYNAME, false, true) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       value = MsRegReadIntEx2(REG_LOCAL_MACHINE, MMCSS_PROFILE_KEYNAME,\r
+               "NetworkThrottlingIndex", false, true);\r
+\r
+       if (value == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (value == 0x0000000a)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// サブキーをすべて削除する\r
+void MsRegDeleteSubkeys(UINT root, char *keyname, bool force32bit, bool force64bit)\r
+{\r
+       TOKEN_LIST *t;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       t = MsRegEnumKeyEx2(root, keyname, force32bit, force64bit);\r
+       if (t == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < t->NumTokens;i++)\r
+       {\r
+               char tmp[MAX_PATH];\r
+\r
+               Format(tmp, sizeof(tmp), "%s\\%s", keyname, t->Token[i]);\r
+\r
+               MsRegDeleteKeyEx2(root, tmp, force32bit, force64bit);\r
+       }\r
+\r
+       FreeToken(t);\r
+}\r
+\r
+// バッファのデータをレジストリのサブキーに変換する\r
+void MsBufToRegSubkeys(UINT root, char *keyname, BUF *b, bool overwrite, bool force32bit, bool force64bit)\r
+{\r
+       UINT i;\r
+       UINT a;\r
+       UINT num_keys;\r
+       // 引数チェック\r
+       if (keyname == NULL || b == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       SeekBuf(b, 0, 0);\r
+\r
+       num_keys = ReadBufInt(b);\r
+\r
+       for (i = 0;i < num_keys;i++)\r
+       {\r
+               char subkeyname[MAX_PATH];\r
+               char fullkeyname[MAX_PATH];\r
+               UINT j;\r
+               UINT num_values;\r
+\r
+               Zero(subkeyname, sizeof(subkeyname));\r
+               ReadBufStr(b, subkeyname, sizeof(subkeyname));\r
+\r
+               Format(fullkeyname, sizeof(fullkeyname), "%s\\%s", keyname, subkeyname);\r
+\r
+               num_values = ReadBufInt(b);\r
+\r
+               for (j = 0;j < num_values;j++)\r
+               {\r
+                       char valuename[MAX_PATH];\r
+                       char data[MAX_SIZE];\r
+\r
+                       Zero(valuename, sizeof(valuename));\r
+                       ReadBufStr(b, valuename, sizeof(valuename));\r
+\r
+                       a = ReadBufInt(b);\r
+\r
+                       if (a == 0)\r
+                       {\r
+                               Zero(data, sizeof(data));\r
+                               ReadBufStr(b, data, sizeof(data));\r
+\r
+                               if (overwrite || MsRegIsValueEx2(root, fullkeyname, valuename, force32bit, force64bit) == false)\r
+                               {\r
+                                       MsRegWriteStrEx2(root, fullkeyname, valuename, data, force32bit, force64bit);\r
+                               }\r
+                       }\r
+                       else\r
+                       {\r
+                               if (overwrite || MsRegIsValueEx2(root, fullkeyname, valuename, force32bit, force64bit) == false)\r
+                               {\r
+                                       MsRegWriteIntEx2(root, fullkeyname, valuename, ReadBufInt(b), force32bit, force64bit);\r
+                               }\r
+                       }\r
+               }\r
+       }\r
+}\r
+\r
+// レジストリのサブキーのデータをバッファに変換する\r
+BUF *MsRegSubkeysToBuf(UINT root, char *keyname, bool force32bit, bool force64bit)\r
+{\r
+       TOKEN_LIST *t;\r
+       UINT i;\r
+       BUF *b;\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       t = MsRegEnumKeyEx2(root, keyname, force32bit, force64bit);\r
+\r
+       if (t == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       b = NewBuf();\r
+\r
+       WriteBufInt(b, t->NumTokens);\r
+\r
+       for (i = 0;i < t->NumTokens;i++)\r
+       {\r
+               char *name = t->Token[i];\r
+               char tmp[MAX_PATH];\r
+               TOKEN_LIST *v;\r
+\r
+               Format(tmp, sizeof(tmp), "%s\\%s", keyname, name);\r
+\r
+               WriteBufStr(b, name);\r
+\r
+               v = MsRegEnumValueEx2(root, tmp, force32bit, force64bit);\r
+               if (v == NULL)\r
+               {\r
+                       WriteBufInt(b, 0);\r
+               }\r
+               else\r
+               {\r
+                       UINT j;\r
+\r
+                       WriteBufInt(b, v->NumTokens);\r
+\r
+                       for (j = 0;j < v->NumTokens;j++)\r
+                       {\r
+                               char *valuename = v->Token[j];\r
+                               char *str;\r
+\r
+                               WriteBufStr(b, valuename);\r
+\r
+                               str = MsRegReadStrEx2(root, tmp, valuename, force32bit, force64bit);\r
+                               if (str != NULL)\r
+                               {\r
+                                       WriteBufInt(b, 0);\r
+                                       WriteBufStr(b, str);\r
+                                       Free(str);\r
+                               }\r
+                               else\r
+                               {\r
+                                       WriteBufInt(b, 1);\r
+                                       WriteBufInt(b, MsRegReadIntEx2(root, tmp, valuename, force32bit, force64bit));\r
+                               }\r
+                       }\r
+\r
+                       FreeToken(v);\r
+               }\r
+       }\r
+\r
+       FreeToken(t);\r
+\r
+       return b;\r
+}\r
+\r
+// 指定した EXE ファイル名のプロセスが存在しているかどうかチェック\r
+bool MsIsProcessExists(char *exename)\r
+{\r
+       LIST *o;\r
+       bool ret = false;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (exename == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       o = MsGetProcessList();\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               MS_PROCESS *proc = LIST_DATA(o, i);\r
+               char exe[MAX_PATH];\r
+\r
+               GetFileNameFromFilePath(exe, sizeof(exe), proc->ExeFilename);\r
+\r
+               if (StrCmpi(exename, exe) == 0)\r
+               {\r
+                       ret = true;\r
+                       break;\r
+               }\r
+       }\r
+\r
+       MsFreeProcessList(o);\r
+\r
+       return ret;\r
+}\r
+bool MsIsProcessExistsW(wchar_t *exename)\r
+{\r
+       LIST *o;\r
+       bool ret = false;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (exename == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       o = MsGetProcessList();\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               MS_PROCESS *proc = LIST_DATA(o, i);\r
+               wchar_t exe[MAX_PATH];\r
+\r
+               GetFileNameFromFilePathW(exe, sizeof(exe), proc->ExeFilenameW);\r
+\r
+               if (UniStrCmpi(exename, exe) == 0)\r
+               {\r
+                       ret = true;\r
+                       break;\r
+               }\r
+       }\r
+\r
+       MsFreeProcessList(o);\r
+\r
+       return ret;\r
+}\r
+\r
+typedef struct _ASTAT_\r
+{\r
+       ADAPTER_STATUS adapt;\r
+       NAME_BUFFER    NameBuff[30];\r
+} ASTAT, *PASTAT;\r
+\r
+// 高精度カウンタの値から精密な時間を取得する\r
+double MsGetHiResTimeSpan(UINT64 diff)\r
+{\r
+       LARGE_INTEGER t;\r
+       UINT64 freq;\r
+\r
+       if (QueryPerformanceFrequency(&t) == false)\r
+       {\r
+               freq = 1000ULL;\r
+       }\r
+       else\r
+       {\r
+               Copy(&freq, &t, sizeof(UINT64));\r
+       }\r
+\r
+       return (double)diff / (double)freq;\r
+}\r
+UINT64 MsGetHiResTimeSpanUSec(UINT64 diff)\r
+{\r
+       LARGE_INTEGER t;\r
+       UINT64 freq;\r
+\r
+       if (QueryPerformanceFrequency(&t) == false)\r
+       {\r
+               freq = 1000ULL;\r
+       }\r
+       else\r
+       {\r
+               Copy(&freq, &t, sizeof(UINT64));\r
+       }\r
+\r
+       return (UINT64)(diff) * 1000ULL * 1000ULL / (UINT64)freq;\r
+}\r
+\r
+// 高精度カウンタを取得する\r
+UINT64 MsGetHiResCounter()\r
+{\r
+       LARGE_INTEGER t;\r
+       UINT64 ret;\r
+\r
+       if (QueryPerformanceCounter(&t) == false)\r
+       {\r
+               return Tick64();\r
+       }\r
+\r
+       Copy(&ret, &t, sizeof(UINT64));\r
+\r
+       return ret;\r
+}\r
+\r
+// ようこそ画面を使用しているかどうか\r
+bool MsIsUseWelcomeLogin()\r
+{\r
+       UINT os_type;\r
+       if (MsIsNt() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       os_type = GetOsInfo()->OsType;\r
+\r
+       if (OS_IS_WINDOWS_NT(os_type))\r
+       {\r
+               if (GET_KETA(os_type, 100) == 3)\r
+               {\r
+                       if (MsRegReadIntEx2(REG_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon",\r
+                               "LogonType", false, true) == 0)\r
+                       {\r
+                               return false;\r
+                       }\r
+                       else\r
+                       {\r
+                               return true;\r
+                       }\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// コンピュータの物理的な MAC アドレスを 1 つ取得する\r
+bool MsGetPhysicalMacAddress(void *address)\r
+{\r
+       // 引数チェック\r
+       if (address == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsGetPhysicalMacAddressFromApi(address))\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (MsGetPhysicalMacAddressFromNetbios(address))\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// 物理的な MAC アドレスを取得する (API から)\r
+bool MsGetPhysicalMacAddressFromApi(void *address)\r
+{\r
+       MS_ADAPTER_LIST *o;\r
+       UINT i;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (address == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(address, 6);\r
+\r
+       o = MsCreateAdapterList();\r
+\r
+       for (i = 0;i < o->Num;i++)\r
+       {\r
+               MS_ADAPTER *a = o->Adapters[i];\r
+\r
+               if (a->AddressSize == 6 && a->Mtu == 1500)\r
+               {\r
+                       bool b = false;\r
+                       switch (a->Type)\r
+                       {\r
+                       case MIB_IF_TYPE_OTHER:\r
+                       case MIB_IF_TYPE_ETHERNET:\r
+                               b = true;\r
+                               break;\r
+\r
+                       case MIB_IF_TYPE_TOKENRING:\r
+                       case MIB_IF_TYPE_FDDI:\r
+                       case MIB_IF_TYPE_PPP:\r
+                       case MIB_IF_TYPE_LOOPBACK:\r
+                       case MIB_IF_TYPE_SLIP:\r
+                               b = false;\r
+                               break;\r
+\r
+                       default:\r
+                               b = true;\r
+                               break;\r
+                       }\r
+\r
+                       if (b)\r
+                       {\r
+                               if (SearchStrEx(a->Title, "WAN", 0, false) == INFINITE)\r
+                               {\r
+                                       if (a->Status == MIB_IF_OPER_STATUS_CONNECTED || a->Status == MIB_IF_OPER_STATUS_OPERATIONAL)\r
+                                       {\r
+                                               if (a->AddressSize == 6)\r
+                                               {\r
+                                                       if (IsZero(a->Address, 6) == false)\r
+                                                       {\r
+                                                               if (Cmp(address, a->Address, 6) <= 0)\r
+                                                               {\r
+                                                                       Copy(address, a->Address, 6);\r
+                                                                       ret = true;\r
+                                                               }\r
+                                                       }\r
+                                               }\r
+                                       }\r
+                               }\r
+                       }\r
+               }\r
+       }\r
+\r
+       MsFreeAdapterList(o);\r
+\r
+       return ret;\r
+}\r
+\r
+// 物理的な MAC アドレスを取得する (NetBIOS から)\r
+bool MsGetPhysicalMacAddressFromNetbios(void *address)\r
+{\r
+       NCB ncb;\r
+       UCHAR ret;\r
+       LANA_ENUM lenum;\r
+       UINT i;\r
+       ASTAT adapter;\r
+       bool b = false;\r
+       // 引数チェック\r
+       if (address == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&ncb, sizeof(ncb));\r
+       Zero(&lenum, sizeof(lenum));\r
+\r
+       ncb.ncb_command = NCBENUM;\r
+       ncb.ncb_buffer = (UCHAR *)&lenum;\r
+       ncb.ncb_length = sizeof(lenum);\r
+       ret = Netbios(&ncb);\r
+\r
+       Zero(address, 6);\r
+\r
+       for (i = 0;i < lenum.length;i++)\r
+       {\r
+               Zero(&ncb, sizeof(ncb));\r
+               ncb.ncb_command = NCBRESET;\r
+               ncb.ncb_lana_num = lenum.lana[i];\r
+\r
+               ret = Netbios(&ncb);\r
+\r
+               Zero(&ncb, sizeof(ncb));\r
+               ncb.ncb_command = NCBASTAT;\r
+               ncb.ncb_lana_num = lenum.lana[i];\r
+\r
+               StrCpy(ncb.ncb_callname, sizeof(ncb.ncb_callname), "*               ");\r
+               Zero(&adapter, sizeof(adapter));\r
+               ncb.ncb_buffer = (char *)&adapter;\r
+               ncb.ncb_length = sizeof(adapter);\r
+\r
+               ret = Netbios(&ncb);\r
+\r
+               if (ret == 0)\r
+               {\r
+                       if (Cmp(address, adapter.adapt.adapter_address, 6) <= 0)\r
+                       {\r
+                               Copy(address, adapter.adapt.adapter_address, 6);\r
+                               b = true;\r
+                       }\r
+               }\r
+       }\r
+\r
+       return b;\r
+}\r
+\r
+// システム全体のアップデート通知\r
+void MsUpdateSystem()\r
+{\r
+       static DWORD dw = 0;\r
+\r
+       SendMessageTimeoutA(HWND_BROADCAST, WM_WININICHANGE, 0, 0, SMTO_NORMAL, 1, (PDWORD_PTR)&dw);\r
+}\r
+\r
+// 指定されたパスがローカルドライブかどうか取得する\r
+bool MsIsLocalDrive(char *name)\r
+{\r
+       char tmp[MAX_PATH];\r
+       UINT ret;\r
+\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(tmp, sizeof(tmp));\r
+       InnerFilePath(tmp, sizeof(tmp), name);\r
+\r
+       if (StartWith(tmp, "\\\\"))\r
+       {\r
+               // ネットワークディレクトリ\r
+               return false;\r
+       }\r
+\r
+       if (tmp[1] != ':' || tmp[2] != '\\')\r
+       {\r
+               // ドライブ名でない\r
+               return false;\r
+       }\r
+\r
+       tmp[3] = 0;\r
+\r
+       ret = GetDriveType(tmp);\r
+\r
+       if (ret == DRIVE_REMOTE || ret == DRIVE_CDROM || ret == DRIVE_RAMDISK)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+bool MsIsLocalDriveW(wchar_t *name)\r
+{\r
+       char name_a[MAX_PATH];\r
+\r
+       UniToStr(name_a, sizeof(name_a), name);\r
+\r
+       return MsIsLocalDrive(name_a);\r
+}\r
+\r
+// 指定されたファイルがロックされているかどうか取得する\r
+bool MsIsFileLocked(char *name)\r
+{\r
+       HANDLE h;\r
+       char tmp[MAX_PATH];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       InnerFilePath(tmp, sizeof(tmp), name);\r
+\r
+       h = CreateFile(tmp, GENERIC_READ | GENERIC_WRITE, 0, 0, OPEN_EXISTING, 0, NULL);\r
+       if (h == INVALID_HANDLE_VALUE)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       CloseHandle(h);\r
+\r
+       return false;\r
+}\r
+bool MsIsFileLockedW(wchar_t *name)\r
+{\r
+       HANDLE h;\r
+       wchar_t tmp[MAX_PATH];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char name_a[MAX_SIZE];\r
+\r
+               UniToStr(name_a, sizeof(name_a), name);\r
+\r
+               return MsIsFileLocked(name_a);\r
+       }\r
+\r
+       InnerFilePathW(tmp, sizeof(tmp), name);\r
+\r
+       h = CreateFileW(tmp, GENERIC_READ | GENERIC_WRITE, 0, 0, OPEN_EXISTING, 0, NULL);\r
+       if (h == INVALID_HANDLE_VALUE)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       CloseHandle(h);\r
+\r
+       return false;\r
+}\r
+\r
+// プロセスの終了を待機\r
+UINT MsWaitProcessExit(void *process_handle)\r
+{\r
+       HANDLE h = (HANDLE)process_handle;\r
+       UINT ret = 1;\r
+\r
+       if (h == NULL)\r
+       {\r
+               return 1;\r
+       }\r
+\r
+       while (true)\r
+       {\r
+               WaitForSingleObject(h, INFINITE);\r
+\r
+               ret = 1;\r
+               if (GetExitCodeProcess(h, &ret) == false)\r
+               {\r
+                       break;\r
+               }\r
+\r
+               if (ret != STILL_ACTIVE)\r
+               {\r
+                       break;\r
+               }\r
+       }\r
+\r
+       CloseHandle(h);\r
+\r
+       return ret;\r
+}\r
+\r
+// ファイルの実行 (プロセスハンドル取得)\r
+bool MsExecuteEx(char *exe, char *arg, void **process_handle)\r
+{\r
+       SHELLEXECUTEINFO info;\r
+       HANDLE h;\r
+       // 引数チェック\r
+       if (exe == NULL || process_handle == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&info, sizeof(info));\r
+       info.cbSize = sizeof(info);\r
+       info.lpVerb = "open";\r
+       info.lpFile = exe;\r
+       info.fMask = SEE_MASK_NOCLOSEPROCESS;\r
+       info.lpParameters = arg;\r
+       info.nShow = SW_SHOWNORMAL;\r
+       if (ShellExecuteEx(&info) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       h = info.hProcess;\r
+\r
+       *process_handle = (void *)h;\r
+\r
+       return true;\r
+}\r
+bool MsExecuteExW(wchar_t *exe, wchar_t *arg, void **process_handle)\r
+{\r
+       SHELLEXECUTEINFOW info;\r
+       HANDLE h;\r
+       // 引数チェック\r
+       if (exe == NULL || process_handle == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char exe_a[MAX_SIZE];\r
+               char arg_a[MAX_SIZE];\r
+\r
+               UniToStr(exe_a, sizeof(exe_a), exe);\r
+               UniToStr(arg_a, sizeof(arg_a), arg);\r
+\r
+               return MsExecuteEx(exe_a, arg_a, process_handle);\r
+       }\r
+\r
+       Zero(&info, sizeof(info));\r
+       info.cbSize = sizeof(info);\r
+       info.lpVerb = L"open";\r
+       info.lpFile = exe;\r
+       info.fMask = SEE_MASK_NOCLOSEPROCESS;\r
+       info.lpParameters = arg;\r
+       info.nShow = SW_SHOWNORMAL;\r
+       if (ShellExecuteExW(&info) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       h = info.hProcess;\r
+\r
+       *process_handle = (void *)h;\r
+\r
+       return true;\r
+}\r
+\r
+// ファイルの実行\r
+bool MsExecute(char *exe, char *arg)\r
+{\r
+       DWORD d;\r
+       // 引数チェック\r
+       if (exe == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       d = (DWORD)ShellExecuteA(NULL, "open", exe, arg, MsGetExeDirName(), SW_SHOWNORMAL);\r
+\r
+       if (d > 32)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+bool MsExecuteW(wchar_t *exe, wchar_t *arg)\r
+{\r
+       DWORD d;\r
+       // 引数チェック\r
+       if (exe == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char exe_a[MAX_SIZE];\r
+               char arg_a[MAX_SIZE];\r
+\r
+               UniToStr(exe_a, sizeof(exe_a), exe);\r
+               UniToStr(arg_a, sizeof(arg_a), arg);\r
+\r
+               return MsExecute(exe_a, arg_a);\r
+       }\r
+\r
+       d = (DWORD)ShellExecuteW(NULL, L"open", exe, arg, MsGetExeDirNameW(), SW_SHOWNORMAL);\r
+\r
+       if (d > 32)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// ディレクトリの再帰作成\r
+void MsUniMakeDirEx(wchar_t *name)\r
+{\r
+       UINT wp;\r
+       wchar_t *tmp;\r
+       UINT i, len;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       tmp = ZeroMalloc(UniStrSize(name) * 2);\r
+       wp = 0;\r
+       len = UniStrLen(name);\r
+       for (i = 0;i < len;i++)\r
+       {\r
+               wchar_t c = name[i];\r
+\r
+               if (c == '\\')\r
+               {\r
+                       if (UniStrCmpi(tmp, L"\\\\") != 0 && UniStrCmpi(tmp, L"\\") != 0)\r
+                       {\r
+                               MsUniMakeDir(tmp);\r
+                       }\r
+               }\r
+\r
+               tmp[wp++] = c;\r
+       }\r
+\r
+       Free(tmp);\r
+\r
+       MsUniMakeDir(name);\r
+}\r
+void MsMakeDirEx(char *name)\r
+{\r
+       wchar_t *name_w = CopyStrToUni(name);\r
+\r
+       MsUniMakeDirEx(name_w);\r
+\r
+       Free(name_w);\r
+}\r
+\r
+// ディレクトリの作成\r
+bool MsUniMakeDir(wchar_t *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               char *s = CopyUniToStr(name);\r
+               bool ret = MsMakeDir(s);\r
+               Free(s);\r
+               return ret;\r
+       }\r
+\r
+       return CreateDirectoryW(name, NULL);\r
+}\r
+bool MsMakeDir(char *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return CreateDirectoryA(name, NULL);\r
+}\r
+\r
+// ディレクトリの削除\r
+bool MsUniDirectoryDelete(wchar_t *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               char *s = CopyUniToStr(name);\r
+               bool ret = MsDirectoryDelete(s);\r
+               Free(s);\r
+               return ret;\r
+       }\r
+\r
+       return RemoveDirectoryW(name);\r
+}\r
+bool MsDirectoryDelete(char *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return RemoveDirectoryA(name);\r
+}\r
+\r
+// ファイルの削除\r
+bool MsUniFileDelete(wchar_t *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               bool ret;\r
+               char *s = CopyUniToStr(name);\r
+               ret = MsFileDelete(s);\r
+               Free(s);\r
+               return ret;\r
+       }\r
+\r
+       return DeleteFileW(name);\r
+}\r
+bool MsFileDelete(char *name)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return DeleteFileA(name);\r
+}\r
+\r
+// 指定したファイル名がディレクトリかどうか取得する\r
+bool MsUniIsDirectory(wchar_t *name)\r
+{\r
+       DWORD ret;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               char *s = CopyUniToStr(name);\r
+               ret = MsIsDirectory(s);\r
+               Free(s);\r
+\r
+               return ret;\r
+       }\r
+\r
+       ret = GetFileAttributesW(name);\r
+       if (ret == 0xffffffff)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (ret & FILE_ATTRIBUTE_DIRECTORY)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+bool MsIsDirectoryW(wchar_t *name)\r
+{\r
+       return MsUniIsDirectory(name);\r
+}\r
+bool MsIsDirectory(char *name)\r
+{\r
+       DWORD ret;\r
+       char tmp[MAX_PATH];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       InnerFilePath(tmp, sizeof(tmp), name);\r
+\r
+       ret = GetFileAttributesA(tmp);\r
+       if (ret == 0xffffffff)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (ret & FILE_ATTRIBUTE_DIRECTORY)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// MSI ファイルから Cabinet を取り出す\r
+bool MsExtractCabFromMsi(char *msi, char *cab)\r
+{\r
+       wchar_t msi_w[MAX_PATH];\r
+       wchar_t cab_w[MAX_PATH];\r
+\r
+       StrToUni(msi_w, sizeof(msi_w), msi);\r
+       StrToUni(cab_w, sizeof(cab_w), cab);\r
+\r
+       return MsExtractCabFromMsiW(msi_w, cab_w);\r
+}\r
+bool MsExtractCabFromMsiW(wchar_t *msi, wchar_t *cab)\r
+{\r
+       BUF *b;\r
+       bool ret = false;\r
+       UINT i;\r
+       char sign[] = {'M', 'S', 'C', 'F', 0, 0, 0, 0,};\r
+       void *pointer = NULL;\r
+       UINT current_pos = 0;\r
+       UINT sign_size;\r
+       // 引数チェック\r
+       if (msi == NULL || cab == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // MSI を読み込む\r
+       b = ReadDumpW(msi);\r
+       if (b == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (b->Size < 128)\r
+       {\r
+               FreeBuf(b);\r
+               return false;\r
+       }\r
+\r
+       sign_size = sizeof(sign);\r
+\r
+       // "MSCF" を検索する\r
+       for (i = 0;i < (b->Size - sign_size);i++)\r
+       {\r
+               char *p = ((UCHAR *)b->Buf) + i;\r
+\r
+               if (Cmp(p, sign, sign_size) == 0)\r
+               {\r
+                       pointer = p;\r
+                       current_pos = i;\r
+               }\r
+       }\r
+\r
+       if (pointer != NULL)\r
+       {\r
+               UINT size = b->Size - current_pos;\r
+               BUF *b2 = NewBuf();\r
+\r
+               WriteBuf(b2, pointer, size);\r
+\r
+               ret = DumpBufW(b2, cab);\r
+\r
+               FreeBuf(b2);\r
+\r
+       }\r
+\r
+       FreeBuf(b);\r
+\r
+       return ret;\r
+}\r
+\r
+// Cabinet ファイルからファイルを取り出す\r
+bool MsExtractCab(char *cab_name, char *dest_dir_name)\r
+{\r
+       wchar_t cab_name_w[MAX_SIZE];\r
+       wchar_t dest_dir_name_w[MAX_SIZE];\r
+\r
+       StrToUni(cab_name_w, sizeof(cab_name_w), cab_name);\r
+       StrToUni(dest_dir_name_w, sizeof(dest_dir_name_w), dest_dir_name);\r
+\r
+       return MsExtractCabW(cab_name_w, dest_dir_name_w);\r
+}\r
+bool MsExtractCabW(wchar_t *cab_name, wchar_t *dest_dir_name)\r
+{\r
+       wchar_t cabarc[MAX_PATH];\r
+       wchar_t arg[MAX_PATH * 2];\r
+       wchar_t tmp[MAX_PATH];\r
+\r
+       // 引数チェック\r
+       if (cab_name == NULL || dest_dir_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsGetCabarcExeFilenameW(cabarc, sizeof(cabarc)) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       UniStrCpy(tmp, sizeof(tmp), dest_dir_name);\r
+       if (UniEndWith(tmp, L"\\"))\r
+       {\r
+               tmp[UniStrLen(tmp) - 1] = 0;\r
+       }\r
+\r
+       UniFormat(arg, sizeof(arg),\r
+               L"-o X \"%s\" * \"%s\"\\",\r
+               cab_name,\r
+               tmp);\r
+\r
+       MakeDirW(dest_dir_name);\r
+\r
+       if (RunW(cabarc, arg, true, true) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// cabarc.exe の展開\r
+bool MsGetCabarcExeFilename(char *name, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       ConbinePath(name, size, MsGetMyTempDir(), "cabarc.exe");\r
+\r
+       if (IsFileExists(name))\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (FileCopy("|cabarc.exe", name) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+bool MsGetCabarcExeFilenameW(wchar_t *name, UINT size)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       ConbinePathW(name, size, MsGetMyTempDirW(), L"cabarc.exe");\r
+\r
+       if (IsFileExistsW(name))\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (FileCopyW(L"|cabarc.exe", name) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// EXE ファイルから Cabinet ファイルを取り出す\r
+bool MsExtractCabinetFileFromExe(char *exe, char *cab)\r
+{\r
+       BUF *b;\r
+       // 引数チェック\r
+       if (exe == NULL || cab == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       b = MsExtractResourceFromExe(exe, RT_RCDATA, "CABINET");\r
+       if (b == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (DumpBuf(b, cab) == false)\r
+       {\r
+               FreeBuf(b);\r
+\r
+               return false;\r
+       }\r
+\r
+       FreeBuf(b);\r
+\r
+       return true;\r
+}\r
+bool MsExtractCabinetFileFromExeW(wchar_t *exe, wchar_t *cab)\r
+{\r
+       BUF *b;\r
+       // 引数チェック\r
+       if (exe == NULL || cab == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       b = MsExtractResourceFromExeW(exe, RT_RCDATA, "CABINET");\r
+       if (b == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (DumpBufW(b, cab) == false)\r
+       {\r
+               FreeBuf(b);\r
+\r
+               return false;\r
+       }\r
+\r
+       FreeBuf(b);\r
+\r
+       return true;\r
+}\r
+\r
+// EXE ファイルからリソースを取り出す\r
+BUF *MsExtractResourceFromExe(char *exe, char *type, char *name)\r
+{\r
+       HINSTANCE h;\r
+       HRSRC hr;\r
+       HGLOBAL hg;\r
+       UINT size;\r
+       void *data;\r
+       BUF *buf;\r
+       // 引数チェック\r
+       if (exe == NULL || type == NULL || name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       h = LoadLibraryExA(exe, NULL, LOAD_LIBRARY_AS_DATAFILE);\r
+       if (h == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       hr = FindResourceA(h, name, type);\r
+       if (hr == NULL)\r
+       {\r
+               FreeLibrary(h);\r
+               return NULL;\r
+       }\r
+\r
+       hg = LoadResource(h, hr);\r
+       if (hg == NULL)\r
+       {\r
+               FreeLibrary(h);\r
+               return NULL;\r
+       }\r
+\r
+       size = SizeofResource(h, hr);\r
+       data = (void *)LockResource(hg);\r
+\r
+       buf = NewBuf();\r
+       WriteBuf(buf, data, size);\r
+\r
+       FreeResource(hg);\r
+       FreeLibrary(h);\r
+\r
+       SeekBuf(buf, 0, 0);\r
+\r
+       return buf;\r
+}\r
+BUF *MsExtractResourceFromExeW(wchar_t *exe, char *type, char *name)\r
+{\r
+       HINSTANCE h;\r
+       HRSRC hr;\r
+       HGLOBAL hg;\r
+       UINT size;\r
+       void *data;\r
+       BUF *buf;\r
+       // 引数チェック\r
+       if (exe == NULL || type == NULL || name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char exe_a[MAX_PATH];\r
+\r
+               UniToStr(exe_a, sizeof(exe_a), exe);\r
+\r
+               return MsExtractResourceFromExe(exe_a, type, name);\r
+       }\r
+\r
+       h = LoadLibraryExW(exe, NULL, LOAD_LIBRARY_AS_DATAFILE);\r
+       if (h == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       hr = FindResource(h, name, type);\r
+       if (hr == NULL)\r
+       {\r
+               FreeLibrary(h);\r
+               return NULL;\r
+       }\r
+\r
+       hg = LoadResource(h, hr);\r
+       if (hg == NULL)\r
+       {\r
+               FreeLibrary(h);\r
+               return NULL;\r
+       }\r
+\r
+       size = SizeofResource(h, hr);\r
+       data = (void *)LockResource(hg);\r
+\r
+       buf = NewBuf();\r
+       WriteBuf(buf, data, size);\r
+\r
+       FreeResource(hg);\r
+       FreeLibrary(h);\r
+\r
+       SeekBuf(buf, 0, 0);\r
+\r
+       return buf;\r
+}\r
+\r
+// ファイルのバージョン情報を取得する\r
+bool MsGetFileVersion(char *name, UINT *v1, UINT *v2, UINT *v3, UINT *v4)\r
+{\r
+       void *data;\r
+       UINT size;\r
+       DWORD h;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       h = 0;\r
+       size = GetFileVersionInfoSize(name, &h);\r
+       if (size == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       data = ZeroMalloc(size);\r
+\r
+       if (GetFileVersionInfoA(name, 0, size, data))\r
+       {\r
+               VS_FIXEDFILEINFO *info = NULL;\r
+               UINT info_size = 0;\r
+               if (VerQueryValueA(data, "\\", &info, &info_size))\r
+               {\r
+                       if (v1 != NULL)\r
+                       {\r
+                               *v1 = HIWORD(info->dwFileVersionMS);\r
+                       }\r
+\r
+                       if (v2 != NULL)\r
+                       {\r
+                               *v2 = LOWORD(info->dwFileVersionMS);\r
+                       }\r
+\r
+                       if (v3 != NULL)\r
+                       {\r
+                               *v3 = HIWORD(info->dwFileVersionLS);\r
+                       }\r
+\r
+                       if (v4 != NULL)\r
+                       {\r
+                               *v4 = LOWORD(info->dwFileVersionLS);\r
+                       }\r
+\r
+                       ret = true;\r
+               }\r
+       }\r
+\r
+       Free(data);\r
+\r
+       return ret;\r
+}\r
+bool MsGetFileVersionW(wchar_t *name, UINT *v1, UINT *v2, UINT *v3, UINT *v4)\r
+{\r
+       void *data;\r
+       UINT size;\r
+       DWORD h;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char name_a[MAX_PATH];\r
+\r
+               UniToStr(name_a, sizeof(name_a), name);\r
+\r
+               return MsGetFileVersion(name_a, v1, v2, v3, v4);\r
+       }\r
+\r
+       h = 0;\r
+       size = GetFileVersionInfoSizeW(name, &h);\r
+       if (size == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       data = ZeroMalloc(size);\r
+\r
+       if (GetFileVersionInfoW(name, 0, size, data))\r
+       {\r
+               VS_FIXEDFILEINFO *info = NULL;\r
+               UINT info_size = 0;\r
+               if (VerQueryValue(data, "\\", &info, &info_size))\r
+               {\r
+                       if (v1 != NULL)\r
+                       {\r
+                               *v1 = HIWORD(info->dwFileVersionMS);\r
+                       }\r
+\r
+                       if (v2 != NULL)\r
+                       {\r
+                               *v2 = LOWORD(info->dwFileVersionMS);\r
+                       }\r
+\r
+                       if (v3 != NULL)\r
+                       {\r
+                               *v3 = HIWORD(info->dwFileVersionLS);\r
+                       }\r
+\r
+                       if (v4 != NULL)\r
+                       {\r
+                               *v4 = LOWORD(info->dwFileVersionLS);\r
+                       }\r
+\r
+                       ret = true;\r
+               }\r
+       }\r
+\r
+       Free(data);\r
+\r
+       return ret;\r
+}\r
+\r
+// ファイルを隠しファイルにする\r
+void MsSetFileToHidden(char *name)\r
+{\r
+       char tmp[MAX_PATH];\r
+       DWORD d;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       NormalizePath(tmp, sizeof(tmp), name);\r
+\r
+       d = GetFileAttributesA(tmp);\r
+       if (d != INVALID_FILE_ATTRIBUTES)\r
+       {\r
+               d |= FILE_ATTRIBUTE_HIDDEN;\r
+\r
+               SetFileAttributesA(tmp, d);\r
+       }\r
+}\r
+void MsSetFileToHiddenW(wchar_t *name)\r
+{\r
+       wchar_t tmp[MAX_PATH];\r
+       DWORD d;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char name_a[MAX_SIZE];\r
+\r
+               UniToStr(name_a, sizeof(name_a), name);\r
+\r
+               MsSetFileToHidden(name_a);\r
+\r
+               return;\r
+       }\r
+\r
+       NormalizePathW(tmp, sizeof(tmp), name);\r
+\r
+       d = GetFileAttributesW(tmp);\r
+       if (d != INVALID_FILE_ATTRIBUTES)\r
+       {\r
+               d |= FILE_ATTRIBUTE_HIDDEN;\r
+\r
+               SetFileAttributesW(tmp, d);\r
+       }\r
+}\r
+\r
+// スリープ防止用スレッド\r
+void MsNoSleepThread(THREAD *thread, void *param)\r
+{\r
+       MS_NOSLEEP *e;\r
+       EXECUTION_STATE (WINAPI *_SetThreadExecutionState)(EXECUTION_STATE);\r
+       HINSTANCE hKernel32;\r
+       // 引数チェック\r
+       if (thread == NULL || param == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       hKernel32 = LoadLibrary("kernel32.dll");\r
+\r
+       _SetThreadExecutionState =\r
+               (EXECUTION_STATE (__stdcall *)(EXECUTION_STATE))\r
+               GetProcAddress(hKernel32, "SetThreadExecutionState");\r
+\r
+       e = (MS_NOSLEEP *)param;\r
+\r
+       while (e->Halt == false)\r
+       {\r
+               DWORD flag = ES_SYSTEM_REQUIRED;\r
+\r
+               if (e->NoScreenSaver)\r
+               {\r
+                       flag |= ES_DISPLAY_REQUIRED;\r
+               }\r
+\r
+               if (_SetThreadExecutionState != NULL)\r
+               {\r
+                       _SetThreadExecutionState(flag);\r
+               }\r
+\r
+               Wait(e->HaltEvent, 30 * 1000);\r
+       }\r
+\r
+       FreeLibrary(hKernel32);\r
+}\r
+\r
+// スリープ防止用スレッド (Windows Vista 用)\r
+void MsNoSleepThreadVista(THREAD *thread, void *param)\r
+{\r
+       MS_NOSLEEP *e;\r
+       char *key = "Control Panel\\Desktop";\r
+       UINT64 last_set_flag = 0;\r
+       UINT last_c_x = INFINITE, last_c_y = INFINITE;\r
+       UINT64 last_mouse_move_time = 0;\r
+       EXECUTION_STATE (WINAPI *_SetThreadExecutionState)(EXECUTION_STATE);\r
+       HINSTANCE hKernel32;\r
+       // 引数チェック\r
+       if (thread == NULL || param == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       hKernel32 = LoadLibrary("kernel32.dll");\r
+\r
+       _SetThreadExecutionState =\r
+               (EXECUTION_STATE (__stdcall *)(EXECUTION_STATE))\r
+               GetProcAddress(hKernel32, "SetThreadExecutionState");\r
+\r
+       e = (MS_NOSLEEP *)param;\r
+\r
+       while (e->Halt == false)\r
+       {\r
+               DWORD flag = ES_SYSTEM_REQUIRED;\r
+               UINT64 now = Tick64();\r
+               POINT p;\r
+               bool mouse_move = false;\r
+\r
+               Zero(&p, sizeof(p));\r
+               GetCursorPos(&p);\r
+\r
+               if (p.x != last_c_x || p.y != last_c_y)\r
+               {\r
+                       if (last_c_x != INFINITE && last_c_y != INFINITE)\r
+                       {\r
+                               mouse_move = true;\r
+                       }\r
+\r
+                       last_c_x = p.x;\r
+                       last_c_y = p.y;\r
+               }\r
+\r
+               if (mouse_move)\r
+               {\r
+                       last_mouse_move_time = now;\r
+               }\r
+\r
+               if (last_mouse_move_time == 0 || (now > (last_mouse_move_time + 50000ULL)))\r
+               {\r
+                       wchar_t *active;\r
+                       wchar_t *exe;\r
+                       // マウスが 50 秒以上動かない場合はスクリーンセーバーの設定を削除する\r
+\r
+                       active = MsRegReadStrW(REG_CURRENT_USER, key, "ScreenSaveActive");\r
+                       exe = MsRegReadStrW(REG_CURRENT_USER, key, "SCRNSAVE.EXE");\r
+\r
+                       if (UniToInt(active) != 0 && UniIsEmptyStr(exe) == false)\r
+                       {\r
+                               // スクリーンセーバーが設定されている\r
+                               UniStrCpy(e->ScreenSaveActive, sizeof(e->ScreenSaveActive), active);\r
+                               UniStrCpy(e->SCRNSAVE_EXE, sizeof(e->SCRNSAVE_EXE), exe);\r
+\r
+                               MsRegWriteStrW(REG_CURRENT_USER, key, "ScreenSaveActive", L"0");\r
+                               MsRegDeleteValue(REG_CURRENT_USER, key, "SCRNSAVE.EXE");\r
+\r
+                               Debug("Push SS Settings.\n");\r
+                       }\r
+\r
+                       Free(active);\r
+                       Free(exe);\r
+\r
+                       last_mouse_move_time = now;\r
+               }\r
+               else\r
+               {\r
+                       if (mouse_move)\r
+                       {\r
+                               if (UniIsEmptyStr(e->ScreenSaveActive) == false && UniIsEmptyStr(e->SCRNSAVE_EXE) == false)\r
+                               {\r
+                                       // マウスが動いた場合でスクリーンセーバーが設定されていない場合は\r
+                                       // スクリーンセーバーの設定を復元する\r
+                                       wchar_t *active;\r
+                                       wchar_t *exe;\r
+\r
+                                       active = MsRegReadStrW(REG_CURRENT_USER, key, "ScreenSaveActive");\r
+                                       exe = MsRegReadStrW(REG_CURRENT_USER, key, "SCRNSAVE.EXE");\r
+\r
+                                       if (UniToInt(active) != 0 && UniIsEmptyStr(exe) == false)\r
+                                       {\r
+                                       }\r
+                                       else\r
+                                       {\r
+                                               MsRegWriteStrW(REG_CURRENT_USER, key, "ScreenSaveActive", e->ScreenSaveActive);\r
+                                               MsRegWriteStrW(REG_CURRENT_USER, key, "SCRNSAVE.EXE", e->SCRNSAVE_EXE);\r
+\r
+                                               Zero(e->ScreenSaveActive, sizeof(e->ScreenSaveActive));\r
+                                               Zero(e->SCRNSAVE_EXE, sizeof(e->SCRNSAVE_EXE));\r
+\r
+                                               Debug("Pop SS Settings.\n");\r
+                                       }\r
+\r
+                                       Free(active);\r
+                                       Free(exe);\r
+                               }\r
+                       }\r
+               }\r
+\r
+               if (last_set_flag == 0 || (now > (last_set_flag + 50000ULL)))\r
+               {\r
+                       // フラグセット (50 秒間隔)\r
+                       last_set_flag = now;\r
+\r
+                       if (_SetThreadExecutionState != NULL)\r
+                       {\r
+                               _SetThreadExecutionState(flag);\r
+                       }\r
+               }\r
+\r
+               Wait(e->HaltEvent, 512);\r
+       }\r
+\r
+       if (true)\r
+       {\r
+               // スクリーンセーバーの設定を復元する\r
+               wchar_t *active;\r
+               wchar_t *exe;\r
+\r
+               if (UniIsEmptyStr(e->ScreenSaveActive) == false && UniIsEmptyStr(e->SCRNSAVE_EXE) == false)\r
+               {\r
+                       active = MsRegReadStrW(REG_CURRENT_USER, key, "ScreenSaveActive");\r
+                       exe = MsRegReadStrW(REG_CURRENT_USER, key, "SCRNSAVE.EXE");\r
+\r
+                       if (UniToInt(active) != 0 && UniIsEmptyStr(exe) != 0)\r
+                       {\r
+                       }\r
+                       else\r
+                       {\r
+                               MsRegWriteStrW(REG_CURRENT_USER, key, "ScreenSaveActive", e->ScreenSaveActive);\r
+                               MsRegWriteStrW(REG_CURRENT_USER, key, "SCRNSAVE.EXE", e->SCRNSAVE_EXE);\r
+\r
+                               Zero(e->ScreenSaveActive, sizeof(e->ScreenSaveActive));\r
+                               Zero(e->SCRNSAVE_EXE, sizeof(e->SCRNSAVE_EXE));\r
+\r
+                               Debug("Pop SS Settings.\n");\r
+                       }\r
+\r
+                       Free(active);\r
+                       Free(exe);\r
+               }\r
+       }\r
+\r
+       FreeLibrary(hKernel32);\r
+}\r
+\r
+// スリープ防止の開始\r
+void *MsNoSleepStart(bool no_screensaver)\r
+{\r
+       MS_NOSLEEP *e;\r
+       bool is_vista = MsIsVista();\r
+       bool is_nt_4 = false;\r
+       UINT os_type = GetOsInfo()->OsType;\r
+\r
+       if (OS_IS_WINDOWS_NT(os_type))\r
+       {\r
+               if (GET_KETA(os_type, 100) == 1)\r
+               {\r
+                       is_nt_4 = true;\r
+               }\r
+       }\r
+\r
+       e = ZeroMalloc(sizeof(MS_NOSLEEP));\r
+\r
+       e->HaltEvent = NewEvent();\r
+       e->NoScreenSaver = no_screensaver;\r
+\r
+       if (e->NoScreenSaver == false || (is_vista == false && is_nt_4 == false))\r
+       {\r
+               e->Thread = NewThread(MsNoSleepThread, e);\r
+       }\r
+       else\r
+       {\r
+               e->Thread = NewThread(MsNoSleepThreadVista, e);\r
+       }\r
+\r
+       return (void *)e;\r
+}\r
+\r
+// スリープ防止の停止\r
+void MsNoSleepEnd(void *p)\r
+{\r
+       MS_NOSLEEP *e;\r
+       // 引数チェック\r
+       if (p == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       e = (MS_NOSLEEP *)p;\r
+\r
+       e->Halt = true;\r
+       Set(e->HaltEvent);\r
+\r
+       WaitThread(e->Thread, INFINITE);\r
+       ReleaseThread(e->Thread);\r
+       ReleaseEvent(e->HaltEvent);\r
+\r
+       Free(e);\r
+}\r
+\r
+// コンピュータ名の取得\r
+void MsGetComputerName(char *name, UINT size)\r
+{\r
+       DWORD sz;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       sz = size;\r
+       GetComputerName(name, &sz);\r
+}\r
+\r
+// マウスカーソルの位置のハッシュ値を取得\r
+UINT MsGetCursorPosHash()\r
+{\r
+       POINT p;\r
+\r
+       Zero(&p, sizeof(p));\r
+\r
+       if (GetCursorPos(&p) == false)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return MAKELONG((USHORT)p.x, (USHORT)p.y);\r
+}\r
+\r
+// 一般ユーザー権限としてのプロセスの起動\r
+void *MsRunAsUserEx(char *filename, char *arg, bool hide)\r
+{\r
+       void *ret = MsRunAsUserExInner(filename, arg, hide);\r
+\r
+       if (ret == NULL)\r
+       {\r
+               Debug("MsRunAsUserExInner Failed.\n");\r
+               ret = Win32RunEx(filename, arg, hide);\r
+       }\r
+\r
+       return ret;\r
+}\r
+void *MsRunAsUserExW(wchar_t *filename, wchar_t *arg, bool hide)\r
+{\r
+       void *ret = MsRunAsUserExInnerW(filename, arg, hide);\r
+\r
+       if (ret == NULL)\r
+       {\r
+               Debug("MsRunAsUserExInner Failed.\n");\r
+               ret = Win32RunExW(filename, arg, hide);\r
+       }\r
+\r
+       return ret;\r
+}\r
+void *MsRunAsUserExInner(char *filename, char *arg, bool hide)\r
+{\r
+       void *ret;\r
+       wchar_t *filename_w;\r
+       wchar_t *arg_w;\r
+\r
+       filename_w = CopyStrToUni(filename);\r
+       arg_w = CopyStrToUni(arg);\r
+\r
+       ret = MsRunAsUserExInnerW(filename_w, arg_w, hide);\r
+\r
+       Free(filename_w);\r
+       Free(arg_w);\r
+\r
+       return ret;\r
+}\r
+void *MsRunAsUserExInnerW(wchar_t *filename, wchar_t *arg, bool hide)\r
+{\r
+       STARTUPINFOW info;\r
+       PROCESS_INFORMATION ret;\r
+       wchar_t cmdline[MAX_SIZE];\r
+       wchar_t name[MAX_PATH];\r
+       HANDLE hToken;\r
+       // 引数チェック\r
+       if (filename == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (MsIsVista() == false)\r
+       {\r
+               // Windows Vista 以外では使用できない\r
+               return NULL;\r
+       }\r
+\r
+       UniStrCpy(name, sizeof(name), filename);\r
+       UniTrim(name);\r
+\r
+       if (UniSearchStr(name, L"\"", 0) == INFINITE)\r
+       {\r
+               if (arg == NULL)\r
+               {\r
+                       UniFormat(cmdline, sizeof(cmdline), L"%s", name);\r
+               }\r
+               else\r
+               {\r
+                       UniFormat(cmdline, sizeof(cmdline), L"%s %s", name, arg);\r
+               }\r
+       }\r
+       else\r
+       {\r
+               if (arg == NULL)\r
+               {\r
+                       UniFormat(cmdline, sizeof(cmdline), L"\"%s\"", name);\r
+               }\r
+               else\r
+               {\r
+                       UniFormat(cmdline, sizeof(cmdline), L"\"%s\" %s", name, arg);\r
+               }\r
+       }\r
+\r
+       Zero(&info, sizeof(info));\r
+       Zero(&ret, sizeof(ret));\r
+       info.cb = sizeof(info);\r
+       info.dwFlags = STARTF_USESHOWWINDOW;\r
+       info.wShowWindow = (hide == false ? SW_SHOWDEFAULT : SW_HIDE);\r
+\r
+       UniTrim(cmdline);\r
+\r
+       hToken = MsCreateUserToken();\r
+\r
+       if (hToken == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (ms->nt->CreateProcessAsUserW(hToken, NULL, cmdline, NULL, NULL, FALSE,\r
+               (hide == false ? CREATE_NEW_CONSOLE : CREATE_NO_WINDOW | CREATE_NEW_CONSOLE) | NORMAL_PRIORITY_CLASS,\r
+               NULL, NULL, &info, &ret) == FALSE)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       CloseHandle(hToken);\r
+\r
+       CloseHandle(ret.hThread);\r
+       return ret.hProcess;\r
+}\r
+\r
+// アカウント名から SID を取得する\r
+SID *MsGetSidFromAccountName(char *name)\r
+{\r
+       SID *sid;\r
+       UINT sid_size = 4096;\r
+       char *domain_name;\r
+       UINT domain_name_size = 4096;\r
+       SID_NAME_USE use = SidTypeUser;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       sid = ZeroMalloc(sid_size);\r
+       domain_name = ZeroMalloc(domain_name_size);\r
+\r
+       if (ms->nt->LookupAccountNameA(NULL, name, sid, &sid_size, domain_name, &domain_name_size, &use) == false)\r
+       {\r
+               Free(sid);\r
+               Free(domain_name);\r
+               return NULL;\r
+       }\r
+\r
+       Free(domain_name);\r
+\r
+       return sid;\r
+}\r
+\r
+// SID を解放する\r
+void MsFreeSid(SID *sid)\r
+{\r
+       // 引数チェック\r
+       if (sid == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Free(sid);\r
+}\r
+\r
+// 一般ユーザーのトークンを作成する\r
+HANDLE MsCreateUserToken()\r
+{\r
+       char *medium_sid = "S-1-16-8192";\r
+       char *administrators_sid = "S-1-5-32-544";\r
+       SID *sid = NULL;\r
+       TOKEN_MANDATORY_LABEL til;\r
+       HANDLE hCurrentToken, hNewToken;\r
+       if (MsIsNt() == false)\r
+       {\r
+               return NULL;\r
+       }\r
+       if (ms->nt->ConvertStringSidToSidA == NULL ||\r
+               ms->nt->OpenProcessToken == NULL ||\r
+               ms->nt->DuplicateTokenEx == NULL ||\r
+               ms->nt->GetTokenInformation == NULL ||\r
+               ms->nt->SetTokenInformation == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Zero(&til, sizeof(til));\r
+\r
+       if (ms->nt->ConvertStringSidToSidA(medium_sid, &sid) == false)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       til.Label.Attributes = SE_GROUP_INTEGRITY;\r
+       til.Label.Sid = sid;\r
+\r
+       if (ms->nt->OpenProcessToken(GetCurrentProcess(), MAXIMUM_ALLOWED, &hCurrentToken) == false)\r
+       {\r
+               LocalFree(sid);\r
+               return NULL;\r
+       }\r
+\r
+       if (ms->nt->DuplicateTokenEx(hCurrentToken, MAXIMUM_ALLOWED, NULL,\r
+               SecurityImpersonation, TokenPrimary, &hNewToken) == false)\r
+       {\r
+               CloseHandle(hCurrentToken);\r
+               LocalFree(sid);\r
+               return NULL;\r
+       }\r
+\r
+       if (ms->nt->SetTokenInformation(hNewToken, VistaTokenIntegrityLevel, &til,\r
+               sizeof(TOKEN_MANDATORY_LABEL) + GetLengthSid(sid)) == false)\r
+       {\r
+               CloseHandle(hNewToken);\r
+               CloseHandle(hCurrentToken);\r
+               LocalFree(sid);\r
+               return NULL;\r
+       }\r
+\r
+       CloseHandle(hCurrentToken);\r
+       LocalFree(sid);\r
+\r
+       return hNewToken;\r
+}\r
+\r
+// ファイルのデジタル署名をチェック\r
+bool MsCheckFileDigitalSignature(HWND hWnd, char *name, bool *danger)\r
+{\r
+       wchar_t tmp[MAX_PATH];\r
+\r
+       swprintf(tmp, sizeof(tmp), L"%S", name);\r
+\r
+       return MsCheckFileDigitalSignatureW(hWnd, tmp, danger);\r
+}\r
+bool MsCheckFileDigitalSignatureW(HWND hWnd, wchar_t *name, bool *danger)\r
+{\r
+       HRESULT ret = S_OK;\r
+       wchar_t *tmp;\r
+       LONG (WINAPI *_WinVerifyTrust)(HWND, GUID *, LPVOID) = NULL;\r
+       HINSTANCE hDll;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (danger != NULL)\r
+       {\r
+               *danger = false;\r
+       }\r
+\r
+       tmp = name;\r
+\r
+       hDll = LoadLibrary("Wintrust.dll");\r
+       if (hDll == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       _WinVerifyTrust =\r
+               (LONG (__stdcall *)(HWND,GUID *,LPVOID))\r
+               GetProcAddress(hDll, "WinVerifyTrust");\r
+       if (_WinVerifyTrust == NULL)\r
+       {\r
+               FreeLibrary(hDll);\r
+               return false;\r
+       }\r
+       else\r
+       {\r
+               GUID action_id = WINTRUST_ACTION_GENERIC_VERIFY_V2;\r
+               WINTRUST_FILE_INFO file;\r
+               WINTRUST_DATA data;\r
+\r
+               Zero(&file, sizeof(file));\r
+               file.cbStruct = sizeof(file);\r
+               file.pcwszFilePath = tmp;\r
+\r
+               Zero(&data, sizeof(data));\r
+               data.cbStruct = sizeof(data);\r
+               data.fdwRevocationChecks = WTD_REVOKE_WHOLECHAIN;\r
+               data.dwUIChoice = (hWnd != NULL ? WTD_UI_NOGOOD : WTD_UI_NONE);\r
+               data.dwProvFlags = WTD_REVOCATION_CHECK_CHAIN;\r
+               data.dwUnionChoice = WTD_CHOICE_FILE;\r
+               data.pFile = &file;\r
+\r
+               ret = _WinVerifyTrust(hWnd, &action_id, &data);\r
+\r
+               if (ret == ERROR_SUCCESS && danger != NULL)\r
+               {\r
+                       if (hWnd != NULL)\r
+                       {\r
+                               if (MsCheckFileDigitalSignatureW(NULL, name, NULL) == false)\r
+                               {\r
+                                       // 危険なファイルだがユーザーが [OK] を選択してしまった\r
+                                       *danger = true;\r
+                               }\r
+                       }\r
+               }\r
+       }\r
+\r
+       FreeLibrary(hDll);\r
+\r
+       if (ret != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// WoW64 リダイレクションを有効または無効にする\r
+void MsSetWow64FileSystemRedirectionEnable(bool enable)\r
+{\r
+       if (MsIs64BitWindows() == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (ms->nt->Wow64EnableWow64FsRedirection == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       ms->nt->Wow64EnableWow64FsRedirection(enable ? 1 : 0);\r
+}\r
+\r
+// WoW64 リダイレクションを無効にする\r
+void *MsDisableWow64FileSystemRedirection()\r
+{\r
+       void *p = NULL;\r
+       if (MsIs64BitWindows() == false)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (ms->nt->Wow64DisableWow64FsRedirection == NULL ||\r
+               ms->nt->Wow64RevertWow64FsRedirection == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (ms->nt->Wow64DisableWow64FsRedirection(&p) == false)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (p == NULL)\r
+       {\r
+               p = (void *)0x12345678;\r
+       }\r
+\r
+       return p;\r
+}\r
+\r
+// WoW64 リダイレクションを元に戻す\r
+void MsRestoreWow64FileSystemRedirection(void *p)\r
+{\r
+       // 引数チェック\r
+       if (p == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (p == (void *)0x12345678)\r
+       {\r
+               p = NULL;\r
+       }\r
+       if (MsIs64BitWindows() == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (ms->nt->Wow64DisableWow64FsRedirection == NULL ||\r
+               ms->nt->Wow64RevertWow64FsRedirection == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       ms->nt->Wow64RevertWow64FsRedirection(p);\r
+}\r
+\r
+// 現在 x64 版 Windows が動作しているかどうか取得\r
+bool MsIsX64()\r
+{\r
+       SYSTEM_INFO info;\r
+\r
+       if (MsIs64BitWindows() == false)\r
+       {\r
+               return false;\r
+       }\r
+       if (ms->nt->GetNativeSystemInfo == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&info, sizeof(info));\r
+       ms->nt->GetNativeSystemInfo(&info);\r
+\r
+       if (info.wProcessorArchitecture == PROCESSOR_ARCHITECTURE_AMD64)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// 現在 IA64 版 Windows が動作しているかどうか取得\r
+bool MsIsIA64()\r
+{\r
+       if (MsIs64BitWindows() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsX64())\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 64bit Windows かどうか取得\r
+bool MsIs64BitWindows()\r
+{\r
+       if (Is64())\r
+       {\r
+               return true;\r
+       }\r
+       else\r
+       {\r
+               if (MsIsNt() == false)\r
+               {\r
+                       return false;\r
+               }\r
+               else\r
+               {\r
+                       if (ms == NULL || ms->nt == NULL)\r
+                       {\r
+                               return false;\r
+                       }\r
+\r
+                       if (ms->nt->IsWow64Process == NULL)\r
+                       {\r
+                               return false;\r
+                       }\r
+                       else\r
+                       {\r
+                               bool b = false;\r
+                               if (ms->nt->IsWow64Process(GetCurrentProcess(), &b) == false)\r
+                               {\r
+                                       return false;\r
+                               }\r
+                               return b;\r
+                       }\r
+               }\r
+       }\r
+}\r
+\r
+// Windows ファイアウォール登録\r
+void MsRegistWindowsFirewallEx2(char *title, char *exe)\r
+{\r
+       char *dir = MsGetExeDirName();\r
+       char tmp[MAX_PATH];\r
+       // 引数チェック\r
+       if (title == NULL || exe == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       ConbinePath(tmp, sizeof(tmp), dir, exe);\r
+\r
+       if (IsFileExists(tmp) == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       MsRegistWindowsFirewallEx(title, tmp);\r
+}\r
+void MsRegistWindowsFirewall(char *title)\r
+{\r
+       // 引数チェック\r
+       if (title == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       MsRegistWindowsFirewallEx(title, MsGetExeFileName());\r
+}\r
+void MsRegistWindowsFirewallEx(char *title, char *exe)\r
+{\r
+       char *data =\r
+               "Option Explicit\r\nConst NET_FW_PROFILE_DOMAIN = 0\r\nConst NET_FW_PROFILE_STANDARD = 1\r\n"\r
+               "Const NET_FW_SCOPE_ALL = 0\r\nConst NET_FW_IP_VERSION_ANY = 2\r\nDim fwMgr\r\n"\r
+               "Set fwMgr = CreateObject(\"HNetCfg.FwMgr\")\r\nDim profile\r\n"\r
+               "Set profile = fwMgr.LocalPolicy.CurrentProfile\r\nDim app\r\n"\r
+               "Set app = CreateObject(\"HNetCfg.FwAuthorizedApplication\")\r\n"\r
+               "app.ProcessImageFileName = \"$PATH$\"\r\napp.Name = \"$TITLE$\"\r\n"\r
+               "app.Scope = NET_FW_SCOPE_ALL\r\napp.IpVersion = NET_FW_IP_VERSION_ANY\r\n"\r
+               "app.Enabled = TRUE\r\nOn Error Resume Next\r\nprofile.AuthorizedApplications."\r
+               "Add app\r\n";\r
+       char *tmp;\r
+       UINT tmp_size;\r
+       char filename[MAX_PATH];\r
+       char cscript[MAX_PATH];\r
+       char arg[MAX_PATH];\r
+       UINT ostype;\r
+       IO *o;\r
+       char hash[MAX_PATH];\r
+       UCHAR hashbin[SHA1_SIZE];\r
+       // 引数チェック\r
+       if (title == NULL || exe == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // OS チェック (Windows XP, Windows Server 2003, Windows Vista, Windows 7 以外では実施しない)\r
+       ostype = GetOsInfo()->OsType;\r
+       if (OS_IS_WINDOWS_NT(ostype) == false)\r
+       {\r
+               return;\r
+       }\r
+       if (GET_KETA(ostype, 100) != 3 && GET_KETA(ostype, 100) != 4 && GET_KETA(ostype, 100) != 5 && GET_KETA(ostype, 100) != 6)\r
+       {\r
+               return;\r
+       }\r
+\r
+       tmp_size = StrLen(data) * 4;\r
+       tmp = ZeroMalloc(tmp_size);\r
+\r
+       HashSha1(hashbin, exe, StrLen(exe));\r
+       BinToStr(hash, sizeof(hash), hashbin, 6);\r
+\r
+       ReplaceStrEx(tmp, tmp_size, data, "$TITLE$", title, false);\r
+       ReplaceStrEx(tmp, tmp_size, tmp, "$PATH$", exe, false);\r
+\r
+       Format(filename, sizeof(filename), "%s\\winfire_%s.vbs", MsGetMyTempDir(), hash);\r
+       o = FileCreate(filename);\r
+       FileWrite(o, tmp, StrLen(tmp));\r
+       FileClose(o);\r
+\r
+       Format(cscript, sizeof(cscript), "%s\\cscript.exe", MsGetSystem32Dir());\r
+       Format(arg, sizeof(arg), "\"%s\"", filename);\r
+\r
+       Run(cscript, arg, true, false);\r
+\r
+       Debug("cscript %s\n", arg);\r
+\r
+       Free(tmp);\r
+}\r
+\r
+// Vista 用ドライバインストーラの実行\r
+bool MsExecDriverInstaller(char *arg)\r
+{\r
+       wchar_t tmp[MAX_PATH];\r
+       wchar_t hamcore_dst[MAX_PATH];\r
+       wchar_t hamcore_src[MAX_PATH];\r
+       HANDLE h;\r
+       UINT retcode;\r
+       SHELLEXECUTEINFOW info;\r
+       wchar_t *src_exe;\r
+       wchar_t *arg_w;\r
+       // 引数チェック\r
+       if (arg == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       UniFormat(hamcore_dst, sizeof(hamcore_dst), L"%s\\hamcore.utvpn", MsGetMyTempDirW());\r
+       UniFormat(hamcore_src, sizeof(hamcore_src), L"%s\\hamcore.utvpn", MsGetExeDirNameW());\r
+\r
+       // ファイル展開\r
+       src_exe = VISTA_DRIVER_INSTALLER_SRC;\r
+\r
+       if (MsIsX64())\r
+       {\r
+               src_exe = VISTA_DRIVER_INSTALLER_SRC_X64;\r
+       }\r
+       if (MsIsIA64())\r
+       {\r
+               src_exe = VISTA_DRIVER_INSTALLER_SRC_IA64;\r
+       }\r
+\r
+       UniFormat(tmp, sizeof(tmp), VISTA_DRIVER_INSTALLER_DST, MsGetMyTempDirW());\r
+\r
+       if (FileCopyW(src_exe, tmp) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (FileCopyW(hamcore_src, hamcore_dst) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       arg_w = CopyStrToUni(arg);\r
+\r
+       // 実行\r
+       Zero(&info, sizeof(info));\r
+       info.cbSize = sizeof(info);\r
+       info.lpVerb = L"open";\r
+       info.lpFile = tmp;\r
+       info.fMask = SEE_MASK_NOCLOSEPROCESS;\r
+       info.lpParameters = arg_w;\r
+       info.nShow = SW_SHOWNORMAL;\r
+       if (ShellExecuteExW(&info) == false)\r
+       {\r
+               Free(arg_w);\r
+               return false;\r
+       }\r
+\r
+       Free(arg_w);\r
+\r
+       h = info.hProcess;\r
+       retcode = 1;\r
+\r
+       while (true)\r
+       {\r
+               // 完了まで待機\r
+               WaitForSingleObject(h, INFINITE);\r
+\r
+               // 終了コードを取得\r
+               retcode = 1;\r
+               if (GetExitCodeProcess(h, &retcode) == false)\r
+               {\r
+                       break;\r
+               }\r
+\r
+               if (retcode != STILL_ACTIVE)\r
+               {\r
+                       break;\r
+               }\r
+       }\r
+\r
+       CloseHandle(h);\r
+\r
+       if (retcode & 1)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 現在のスレッドのロケールを取得\r
+UINT MsGetThreadLocale()\r
+{\r
+       return (UINT)GetThreadLocale();\r
+}\r
+\r
+// 現在のコンソールの横幅を設定する\r
+UINT MsSetConsoleWidth(UINT size)\r
+{\r
+       HANDLE h = GetStdHandle(STD_OUTPUT_HANDLE);\r
+       CONSOLE_SCREEN_BUFFER_INFO info;\r
+       COORD c;\r
+       UINT old_x, old_y;\r
+       // 引数チェック\r
+       if (size == 0)\r
+       {\r
+               return 0;\r
+       }\r
+       if (h == INVALID_HANDLE_VALUE)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       Zero(&info, sizeof(info));\r
+       if (GetConsoleScreenBufferInfo(h, &info) == false)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       old_x = info.dwSize.X;\r
+       old_y = info.dwSize.Y;\r
+\r
+       c.X = size;\r
+       c.Y = old_y;\r
+\r
+       SetConsoleScreenBufferSize(h, c);\r
+\r
+       return old_x;\r
+}\r
+\r
+// 現在のコンソールの横幅を取得する\r
+UINT MsGetConsoleWidth()\r
+{\r
+       HANDLE h = GetStdHandle(STD_OUTPUT_HANDLE);\r
+       CONSOLE_SCREEN_BUFFER_INFO info;\r
+\r
+       if (h == INVALID_HANDLE_VALUE)\r
+       {\r
+               return 80;\r
+       }\r
+\r
+       Zero(&info, sizeof(info));\r
+       if (GetConsoleScreenBufferInfo(h, &info) == false)\r
+       {\r
+               return 80;\r
+       }\r
+\r
+       return info.dwSize.X;\r
+}\r
+\r
+// MS-IME を無効にする\r
+bool MsDisableIme()\r
+{\r
+       HINSTANCE h;\r
+       bool ret = false;\r
+       char dll_name[MAX_PATH];\r
+       BOOL (WINAPI *_ImmDisableIME)(DWORD);\r
+\r
+       Format(dll_name, sizeof(dll_name), "%s\\imm32.dll", MsGetSystem32Dir());\r
+       h = MsLoadLibrary(dll_name);\r
+       if (h == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       _ImmDisableIME = (BOOL (__stdcall *)(DWORD))GetProcAddress(h, "ImmDisableIME");\r
+\r
+       if (_ImmDisableIME != NULL)\r
+       {\r
+               ret = _ImmDisableIME(-1);\r
+       }\r
+\r
+       FreeLibrary(h);\r
+\r
+       return ret;\r
+}\r
+\r
+// 現在時刻を表示する\r
+void MsPrintTick()\r
+{\r
+       UINT tick = timeGetTime();\r
+       static UINT tick_init = 0;\r
+       if (tick_init == 0)\r
+       {\r
+               tick_init = tick;\r
+               tick = 0;\r
+       }\r
+       else\r
+       {\r
+               tick -= tick_init;\r
+       }\r
+\r
+       printf("[%u]\n", tick);\r
+}\r
+\r
+// LoadLibrary の hamcore 対応版 (データファイルとして読み込み)\r
+void *MsLoadLibraryAsDataFileW(wchar_t *name)\r
+{\r
+       BUF *b;\r
+       wchar_t tmp_dll_name[MAX_SIZE];\r
+       char hash_str[MAX_SIZE];\r
+       UCHAR hash[SHA1_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Hash(hash, name, UniStrLen(name), true);\r
+\r
+       BinToStr(hash_str, sizeof(hash_str), hash, 4);\r
+\r
+       UniFormat(tmp_dll_name, sizeof(tmp_dll_name), L"%s\\%S.dll", MsGetMyTempDirW(), hash_str);\r
+\r
+       if (IsFileExistsW(tmp_dll_name) == false)\r
+       {\r
+               b = ReadDumpW(name);\r
+               if (b == NULL)\r
+               {\r
+                       return NULL;\r
+               }\r
+\r
+               DumpBufW(b, tmp_dll_name);\r
+               FreeBuf(b);\r
+       }\r
+\r
+       return LoadLibraryExW(tmp_dll_name, NULL, LOAD_LIBRARY_AS_DATAFILE);\r
+}\r
+void *MsLoadLibraryAsDataFile(char *name)\r
+{\r
+       wchar_t name_w[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       StrToUni(name_w, sizeof(name_w), name);\r
+\r
+       return MsLoadLibraryAsDataFileW(name_w);\r
+}\r
+\r
+// LoadLibrary の hamcore 対応版\r
+void *MsLoadLibraryW(wchar_t *name)\r
+{\r
+       BUF *b;\r
+       wchar_t tmp_dll_name[MAX_SIZE];\r
+       char hash_str[MAX_SIZE];\r
+       UCHAR hash[SHA1_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Hash(hash, name, UniStrSize(name), true);\r
+\r
+       BinToStr(hash_str, sizeof(hash_str), hash, 4);\r
+\r
+       UniFormat(tmp_dll_name, sizeof(tmp_dll_name), L"%s\\%S.dll", MsGetMyTempDirW(), hash_str);\r
+\r
+       if (IsFileExistsW(tmp_dll_name) == false)\r
+       {\r
+               b = ReadDumpW(name);\r
+               if (b == NULL)\r
+               {\r
+                       return NULL;\r
+               }\r
+\r
+               DumpBufW(b, tmp_dll_name);\r
+               FreeBuf(b);\r
+       }\r
+\r
+       if (IsNt())\r
+       {\r
+               return LoadLibraryW(tmp_dll_name);\r
+       }\r
+       else\r
+       {\r
+               char tmp_dll_name_a[MAX_SIZE];\r
+               HINSTANCE ret;\r
+\r
+               UniToStr(tmp_dll_name_a, sizeof(tmp_dll_name_a), tmp_dll_name);\r
+\r
+               ret = LoadLibraryA(tmp_dll_name_a);\r
+\r
+               return ret;\r
+       }\r
+}\r
+void *MsLoadLibrary(char *name)\r
+{\r
+       wchar_t name_w[MAX_SIZE];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       StrToUni(name_w, sizeof(name_w), name);\r
+\r
+       return MsLoadLibraryW(name_w);\r
+}\r
+\r
+// 単一のアダプタの取得\r
+MS_ADAPTER *MsGetAdapter(char *title)\r
+{\r
+       MS_ADAPTER_LIST *o;\r
+       MS_ADAPTER *ret = NULL;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (title == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       o = MsCreateAdapterList();\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       for (i = 0;i < o->Num;i++)\r
+       {\r
+               if (StrCmpi(o->Adapters[i]->Title, title) == 0)\r
+               {\r
+                       ret = MsCloneAdapter(o->Adapters[i]);\r
+                       break;\r
+               }\r
+       }\r
+\r
+       MsFreeAdapterList(o);\r
+\r
+       return ret;\r
+}\r
+\r
+// 32 ビットオーバーフローチェック\r
+#define        CHECK_32BIT_OVERFLOW(old_value, new_value)                              \\r
+{                                                                                                                              \\r
+       if ((old_value) > (new_value))                                                          \\r
+       {                                                                                                                       \\r
+               (new_value) += ((UINT64)4294967296ULL);                                 \\r
+       }                                                                                                                       \\r
+}\r
+\r
+// 指定したアダプタの TCP/IP 情報を取得する\r
+void MsGetAdapterTcpIpInformation(MS_ADAPTER *a)\r
+{\r
+       IP_ADAPTER_INFO *info, *info_top;\r
+       UINT info_size;\r
+       UINT ret;\r
+       // 引数チェック\r
+       if (a == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (w32net->GetAdaptersInfo == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       info_top = ZeroMalloc(sizeof(IP_ADAPTER_INFO));\r
+       info_size = sizeof(IP_ADAPTER_INFO);\r
+\r
+       ret = w32net->GetAdaptersInfo(info_top, &info_size);\r
+       if (ret == ERROR_INSUFFICIENT_BUFFER || ret == ERROR_BUFFER_OVERFLOW)\r
+       {\r
+               Free(info_top);\r
+               info_size *= 2;\r
+               info_top = ZeroMalloc(info_size);\r
+\r
+               if (w32net->GetAdaptersInfo(info_top, &info_size) != NO_ERROR)\r
+               {\r
+                       Free(info_top);\r
+                       return;\r
+               }\r
+       }\r
+       else if (ret != NO_ERROR)\r
+       {\r
+               Free(info_top);\r
+               return;\r
+       }\r
+\r
+       // 自分のエントリを検索する\r
+       info = info_top;\r
+\r
+       while (info != NULL)\r
+       {\r
+               if (info->Index == a->Index)\r
+               {\r
+                       IP_ADDR_STRING *s;\r
+\r
+                       // IP アドレス\r
+                       a->NumIpAddress = 0;\r
+                       s = &info->IpAddressList;\r
+                       while (s != NULL)\r
+                       {\r
+                               if (a->NumIpAddress < MAX_MS_ADAPTER_IP_ADDRESS)\r
+                               {\r
+                                       StrToIP(&a->IpAddresses[a->NumIpAddress], s->IpAddress.String);\r
+                                       StrToIP(&a->SubnetMasks[a->NumIpAddress], s->IpMask.String);\r
+                                       a->NumIpAddress++;\r
+                               }\r
+                               s = s->Next;\r
+                       }\r
+\r
+                       // ゲートウェイ\r
+                       a->NumGateway = 0;\r
+                       s = &info->GatewayList;\r
+                       while (s != NULL)\r
+                       {\r
+                               if (a->NumGateway < MAX_MS_ADAPTER_IP_ADDRESS)\r
+                               {\r
+                                       StrToIP(&a->Gateways[a->NumGateway], s->IpAddress.String);\r
+                                       a->NumGateway++;\r
+                               }\r
+                               s = s->Next;\r
+                       }\r
+\r
+                       // DHCP サーバー\r
+                       a->UseDhcp = (info->DhcpEnabled == 0 ? false : true);\r
+                       if (a->UseDhcp)\r
+                       {\r
+                               SYSTEMTIME st;\r
+\r
+                               StrToIP(&a->DhcpServer, info->DhcpServer.IpAddress.String);\r
+                               TimeToSystem(&st, info->LeaseObtained);\r
+                               a->DhcpLeaseStart = SystemToUINT64(&st);\r
+\r
+                               TimeToSystem(&st, info->LeaseExpires);\r
+                               a->DhcpLeaseExpires = SystemToUINT64(&st);\r
+                       }\r
+\r
+                       // WINS サーバー\r
+                       a->UseWins = info->HaveWins;\r
+                       if (a->UseWins)\r
+                       {\r
+                               StrToIP(&a->PrimaryWinsServer, info->PrimaryWinsServer.IpAddress.String);\r
+                               StrToIP(&a->SecondaryWinsServer, info->SecondaryWinsServer.IpAddress.String);\r
+                       }\r
+\r
+                       StrCpy(a->Guid, sizeof(a->Guid), info->AdapterName);\r
+\r
+                       a->Info = true;\r
+\r
+                       break;\r
+               }\r
+\r
+               info = info->Next;\r
+       }\r
+\r
+       Free(info_top);\r
+}\r
+\r
+// アダプタリストの生成\r
+MS_ADAPTER_LIST *MsCreateAdapterList()\r
+{\r
+       return MsCreateAdapterListEx(false);\r
+}\r
+MS_ADAPTER_LIST *MsCreateAdapterListEx(bool no_info)\r
+{\r
+       MS_ADAPTER_LIST *ret;\r
+\r
+       if (no_info)\r
+       {\r
+               ret = MsCreateAdapterListInnerEx(true);\r
+\r
+               return ret;\r
+       }\r
+\r
+       Lock(lock_adapter_list);\r
+       {\r
+               MS_ADAPTER_LIST *old = last_adapter_list;\r
+               UINT i;\r
+\r
+               // 新しくアダプタリストを取ってくる\r
+               ret = MsCreateAdapterListInner();\r
+\r
+               if (ret == NULL)\r
+               {\r
+                       Unlock(lock_adapter_list);\r
+                       return NULL;\r
+               }\r
+\r
+               // 取ってきたアダプタリストの各エントリについて、前回取得したものが\r
+               // 存在するかどうかチェックする\r
+               for (i = 0;i < ret->Num;i++)\r
+               {\r
+                       UINT j;\r
+                       for (j = 0;j < old->Num;j++)\r
+                       {\r
+                               MS_ADAPTER *o = old->Adapters[j];\r
+                               MS_ADAPTER *n = ret->Adapters[i];\r
+\r
+                               if (StrCmpi(o->Title, n->Title) == 0)\r
+                               {\r
+                                       // 古いもののほうが値が小さい場合、インクリメントする\r
+                                       CHECK_32BIT_OVERFLOW(o->RecvBytes, n->RecvBytes);\r
+                                       CHECK_32BIT_OVERFLOW(o->RecvPacketsBroadcast, n->RecvPacketsBroadcast);\r
+                                       CHECK_32BIT_OVERFLOW(o->RecvPacketsUnicast, n->RecvPacketsUnicast);\r
+                                       CHECK_32BIT_OVERFLOW(o->SendBytes, n->SendBytes);\r
+                                       CHECK_32BIT_OVERFLOW(o->SendPacketsBroadcast, n->SendPacketsBroadcast);\r
+                                       CHECK_32BIT_OVERFLOW(o->SendPacketsUnicast, n->SendPacketsUnicast);\r
+                                       break;\r
+                               }\r
+                       }\r
+               }\r
+\r
+               // 古いアダプタリストを解放する\r
+               MsFreeAdapterList(old);\r
+\r
+               // 新しく取得したアダプタリストのクローンを保存しておく\r
+               last_adapter_list = MsCloneAdapterList(ret);\r
+       }\r
+       Unlock(lock_adapter_list);\r
+\r
+       return ret;\r
+}\r
+\r
+// アダプタリストモジュールの初期化\r
+void MsInitAdapterListModule()\r
+{\r
+       lock_adapter_list = NewLock(NULL);\r
+\r
+       last_adapter_list = MsCreateAdapterListInner();\r
+}\r
+\r
+// アダプタリストモジュールの解放\r
+void MsFreeAdapterListModule()\r
+{\r
+       if (last_adapter_list != NULL)\r
+       {\r
+               MsFreeAdapterList(last_adapter_list);\r
+               last_adapter_list = NULL;\r
+       }\r
+\r
+       DeleteLock(lock_adapter_list);\r
+       lock_adapter_list = NULL;\r
+}\r
+\r
+// アダプタリストのクローン\r
+MS_ADAPTER_LIST *MsCloneAdapterList(MS_ADAPTER_LIST *o)\r
+{\r
+       MS_ADAPTER_LIST *ret;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = ZeroMalloc(sizeof(MS_ADAPTER_LIST));\r
+       ret->Num = o->Num;\r
+       ret->Adapters = ZeroMalloc(sizeof(MS_ADAPTER *) * ret->Num);\r
+\r
+       for (i = 0;i < ret->Num;i++)\r
+       {\r
+               ret->Adapters[i] = ZeroMalloc(sizeof(MS_ADAPTER));\r
+               Copy(ret->Adapters[i], o->Adapters[i], sizeof(MS_ADAPTER));\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// アダプタのクローン\r
+MS_ADAPTER *MsCloneAdapter(MS_ADAPTER *a)\r
+{\r
+       MS_ADAPTER *ret;\r
+       // 引数チェック\r
+       if (a == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = ZeroMalloc(sizeof(MS_ADAPTER));\r
+       Copy(ret, a, sizeof(MS_ADAPTER));\r
+\r
+       return ret;\r
+}\r
+\r
+// アダプタリストの作成\r
+MS_ADAPTER_LIST *MsCreateAdapterListInner()\r
+{\r
+       return MsCreateAdapterListInnerEx(false);\r
+}\r
+MS_ADAPTER_LIST *MsCreateAdapterListInnerEx(bool no_info)\r
+{\r
+       LIST *o;\r
+       UINT i;\r
+       UINT retcode;\r
+       MIB_IFTABLE *table;\r
+       UINT table_size = sizeof(MIB_IFTABLE);\r
+       MS_ADAPTER_LIST *ret;\r
+\r
+       table = ZeroMalloc(table_size);\r
+\r
+       if (w32net->GetIfTable == NULL)\r
+       {\r
+               return ZeroMalloc(sizeof(MS_ADAPTER_LIST));\r
+       }\r
+\r
+       retcode = w32net->GetIfTable(table, &table_size, TRUE);\r
+       if (retcode == ERROR_INSUFFICIENT_BUFFER || retcode == ERROR_BUFFER_OVERFLOW)\r
+       {\r
+               Free(table);\r
+               table_size *= 2;\r
+               table = ZeroMalloc(table_size);\r
+               if (w32net->GetIfTable(table, &table_size, TRUE) != NO_ERROR)\r
+               {\r
+                       Free(table);\r
+                       return NULL;\r
+               }\r
+       }\r
+       else if (retcode != NO_ERROR)\r
+       {\r
+               Free(table);\r
+               return NULL;\r
+       }\r
+\r
+       o = NewListFast(NULL);\r
+\r
+       for (i = 0;i < table->dwNumEntries;i++)\r
+       {\r
+               MIB_IFROW *r = &table->table[i];\r
+               char title[MAX_PATH];\r
+               UINT num = 0;\r
+               MS_ADAPTER *a;\r
+               UINT j;\r
+\r
+               //if (r->dwOperStatus == MIB_IF_OPER_STATUS_CONNECTED || r->dwOperStatus == MIB_IF_OPER_STATUS_OPERATIONAL)\r
+               {\r
+                       //if (r->dwType & IF_TYPE_ETHERNET_CSMACD)\r
+                       {\r
+                               for (j = 1;;j++)\r
+                               {\r
+                                       UINT k;\r
+                                       bool exists;\r
+                                       if (j == 1)\r
+                                       {\r
+                                               StrCpy(title, sizeof(title), (char *)r->bDescr);\r
+                                       }\r
+                                       else\r
+                                       {\r
+                                               Format(title, sizeof(title), "%s (%u)", (char *)r->bDescr, j);\r
+                                       }\r
+\r
+                                       exists = false;\r
+\r
+                                       for (k = 0;k < LIST_NUM(o);k++)\r
+                                       {\r
+                                               MS_ADAPTER *a = LIST_DATA(o, k);\r
+\r
+                                               if (StrCmpi(a->Title, title) == 0)\r
+                                               {\r
+                                                       exists = true;\r
+                                                       break;\r
+                                               }\r
+                                       }\r
+\r
+                                       if (exists == false)\r
+                                       {\r
+                                               break;\r
+                                       }\r
+                               }\r
+\r
+                               a = ZeroMalloc(sizeof(MS_ADAPTER));\r
+\r
+                               // アダプタ情報作成\r
+                               StrCpy(a->Title, sizeof(a->Title), title);\r
+                               a->Index = r->dwIndex;\r
+                               a->Type = r->dwType;\r
+                               a->Status = r->dwOperStatus;\r
+                               a->Mtu = r->dwMtu;\r
+                               a->Speed = r->dwSpeed;\r
+                               a->AddressSize = MIN(sizeof(a->Address), r->dwPhysAddrLen);\r
+                               Copy(a->Address, r->bPhysAddr, a->AddressSize);\r
+                               a->RecvBytes = r->dwInOctets;\r
+                               a->RecvPacketsBroadcast = r->dwInNUcastPkts;\r
+                               a->RecvPacketsUnicast = r->dwInUcastPkts;\r
+                               a->SendBytes = r->dwOutOctets;\r
+                               a->SendPacketsBroadcast = r->dwOutNUcastPkts;\r
+                               a->SendPacketsUnicast = r->dwOutUcastPkts;\r
+\r
+                               // TCP/IP 情報取得\r
+                               if (no_info == false)\r
+                               {\r
+                                       MsGetAdapterTcpIpInformation(a);\r
+                               }\r
+\r
+                               Add(o, a);\r
+                       }\r
+               }\r
+       }\r
+\r
+       ret = ZeroMalloc(sizeof(MS_ADAPTER_LIST));\r
+       ret->Num = LIST_NUM(o);\r
+       ret->Adapters = ToArray(o);\r
+\r
+       ReleaseList(o);\r
+       Free(table);\r
+\r
+       return ret;\r
+}\r
+\r
+// アダプタリストの解放\r
+void MsFreeAdapterList(MS_ADAPTER_LIST *o)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < o->Num;i++)\r
+       {\r
+               MsFreeAdapter(o->Adapters[i]);\r
+       }\r
+       Free(o->Adapters);\r
+\r
+       Free(o);\r
+}\r
+\r
+// アダプタ情報の解放\r
+void MsFreeAdapter(MS_ADAPTER *a)\r
+{\r
+       // 引数チェック\r
+       if (a == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Free(a);\r
+}\r
+\r
+// アダプタの状態文字列を取得する\r
+wchar_t *MsGetAdapterStatusStr(UINT status)\r
+{\r
+       wchar_t *ret;\r
+\r
+       switch (status)\r
+       {\r
+       case MIB_IF_OPER_STATUS_NON_OPERATIONAL:\r
+               ret = _UU("MS_NON_OPERATIONAL");\r
+               break;\r
+\r
+       case MIB_IF_OPER_STATUS_UNREACHABLE:\r
+               ret = _UU("MS_UNREACHABLE");\r
+               break;\r
+\r
+       case MIB_IF_OPER_STATUS_DISCONNECTED:\r
+               ret = _UU("MS_DISCONNECTED");\r
+               break;\r
+\r
+       case MIB_IF_OPER_STATUS_CONNECTING:\r
+               ret = _UU("MS_CONNECTING");\r
+               break;\r
+\r
+       case MIB_IF_OPER_STATUS_CONNECTED:\r
+               ret = _UU("MS_CONNECTED");\r
+               break;\r
+\r
+       default:\r
+               ret = _UU("MS_OPERATIONAL");\r
+               break;\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// アダプタの種類文字列を取得する\r
+wchar_t *MsGetAdapterTypeStr(UINT type)\r
+{\r
+       wchar_t *ret;\r
+\r
+       switch (type)\r
+       {\r
+       case MIB_IF_TYPE_ETHERNET:\r
+               ret = _UU("MS_ETHERNET");\r
+               break;\r
+\r
+       case MIB_IF_TYPE_TOKENRING:\r
+               ret = _UU("MS_TOKENRING");\r
+               break;\r
+\r
+       case MIB_IF_TYPE_FDDI:\r
+               ret = _UU("MS_FDDI");\r
+               break;\r
+\r
+       case MIB_IF_TYPE_PPP:\r
+               ret = _UU("MS_PPP");\r
+               break;\r
+\r
+       case MIB_IF_TYPE_LOOPBACK:\r
+               ret = _UU("MS_LOOPBACK");\r
+               break;\r
+\r
+       case MIB_IF_TYPE_SLIP:\r
+               ret = _UU("MS_SLIP");\r
+               break;\r
+\r
+       default:\r
+               ret = _UU("MS_OTHER");\r
+               break;\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// 自分自身の EXE の自分以外のインスタンスをすべて終了する\r
+void MsKillOtherInstance()\r
+{\r
+       MsKillOtherInstanceEx(NULL);\r
+}\r
+void MsKillOtherInstanceEx(char *exclude_svcname)\r
+{\r
+       UINT me, i;\r
+       wchar_t me_path[MAX_PATH];\r
+       wchar_t me_path_short[MAX_PATH];\r
+       LIST *o = MsGetProcessList();\r
+       UINT e_procid = 0;\r
+       UINT e_procid2 = 0;\r
+\r
+       if (exclude_svcname != NULL)\r
+       {\r
+               e_procid = MsReadCallingServiceManagerProcessId(exclude_svcname, false);\r
+               e_procid2 = MsReadCallingServiceManagerProcessId(exclude_svcname, true);\r
+       }\r
+\r
+       me = MsGetProcessId();\r
+\r
+       MsGetCurrentProcessExeNameW(me_path, sizeof(me_path));\r
+       MsGetShortPathNameW(me_path, me_path_short, sizeof(me_path_short));\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               MS_PROCESS *p = LIST_DATA(o, i);\r
+               if (p->ProcessId != me)\r
+               {\r
+                       if ((e_procid == 0 || (e_procid != p->ProcessId)) && (e_procid2 == 0 || (e_procid2 != p->ProcessId)))\r
+                       {\r
+                               wchar_t tmp[MAX_PATH];\r
+                               MsGetShortPathNameW(p->ExeFilenameW, tmp, sizeof(tmp));\r
+                               if (UniStrCmpi(me_path_short, tmp) == 0)\r
+                               {\r
+                                       MsKillProcess(p->ProcessId);\r
+                               }\r
+                       }\r
+               }\r
+       }\r
+\r
+       MsFreeProcessList(o);\r
+}\r
+\r
+// 短いファイル名を取得する\r
+bool MsGetShortPathNameA(char *long_path, char *short_path, UINT short_path_size)\r
+{\r
+       // 引数チェック\r
+       if (long_path == NULL || short_path == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GetShortPathNameA(long_path, short_path, short_path_size) == 0)\r
+       {\r
+               StrCpy(short_path, short_path_size, long_path);\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+bool MsGetShortPathNameW(wchar_t *long_path, wchar_t *short_path, UINT short_path_size)\r
+{\r
+       // 引数チェック\r
+       if (long_path == NULL || short_path == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char short_path_a[MAX_SIZE];\r
+               char long_path_a[MAX_SIZE];\r
+               bool ret;\r
+\r
+               UniToStr(long_path_a, sizeof(long_path_a), long_path);\r
+\r
+               ret = MsGetShortPathNameA(long_path_a, short_path_a, sizeof(short_path_a));\r
+\r
+               StrToUni(short_path, short_path_size, short_path_a);\r
+\r
+               return ret;\r
+       }\r
+\r
+       if (GetShortPathNameW(long_path, short_path, short_path_size) == 0)\r
+       {\r
+               UniStrCpy(short_path, short_path_size, long_path);\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 指定したプロセスの強制終了\r
+bool MsKillProcess(UINT id)\r
+{\r
+       HANDLE h;\r
+       // 引数チェック\r
+       if (id == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       h = OpenProcess(PROCESS_TERMINATE, FALSE, id);\r
+       if (h == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (TerminateProcess(h, 0) == FALSE)\r
+       {\r
+               CloseHandle(h);\r
+               return false;\r
+       }\r
+\r
+       CloseHandle(h);\r
+\r
+       return true;\r
+}\r
+\r
+// 現在の EXE ファイル名を取得\r
+void MsGetCurrentProcessExeName(char *name, UINT size)\r
+{\r
+       UINT id;\r
+       LIST *o;\r
+       MS_PROCESS *p;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       id = MsGetCurrentProcessId();\r
+       o = MsGetProcessList();\r
+       p = MsSearchProcessById(o, id);\r
+       if (p != NULL)\r
+       {\r
+               p = MsSearchProcessById(o, id);\r
+               StrCpy(name, size, p->ExeFilename);\r
+       }\r
+       else\r
+       {\r
+               StrCpy(name, size, MsGetExeFileName());\r
+       }\r
+       MsFreeProcessList(o);\r
+}\r
+void MsGetCurrentProcessExeNameW(wchar_t *name, UINT size)\r
+{\r
+       UINT id;\r
+       LIST *o;\r
+       MS_PROCESS *p;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       id = MsGetCurrentProcessId();\r
+       o = MsGetProcessList();\r
+       p = MsSearchProcessById(o, id);\r
+       if (p != NULL)\r
+       {\r
+               p = MsSearchProcessById(o, id);\r
+               UniStrCpy(name, size, p->ExeFilenameW);\r
+       }\r
+       else\r
+       {\r
+               UniStrCpy(name, size, MsGetExeFileNameW());\r
+       }\r
+       MsFreeProcessList(o);\r
+}\r
+\r
+// プロセスをプロセス ID から検索する\r
+MS_PROCESS *MsSearchProcessById(LIST *o, UINT id)\r
+{\r
+       MS_PROCESS *p, t;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Zero(&t, sizeof(t));\r
+       t.ProcessId = id;\r
+\r
+       p = Search(o, &t);\r
+\r
+       return p;\r
+}\r
+\r
+// プロセスリスト比較\r
+int MsCompareProcessList(void *p1, void *p2)\r
+{\r
+       MS_PROCESS *e1, *e2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       e1 = *(MS_PROCESS **)p1;\r
+       e2 = *(MS_PROCESS **)p2;\r
+       if (e1 == NULL || e2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (e1->ProcessId > e2->ProcessId)\r
+       {\r
+               return 1;\r
+       }\r
+       else if (e1->ProcessId < e2->ProcessId)\r
+       {\r
+               return -1;\r
+       }\r
+       else\r
+       {\r
+               return 0;\r
+       }\r
+}\r
+\r
+// プロセスリストの表示\r
+void MsPrintProcessList(LIST *o)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               MS_PROCESS *p = LIST_DATA(o, i);\r
+               UniPrint(L"%-4u: %s\n", p->ProcessId, p->ExeFilenameW);\r
+       }\r
+}\r
+\r
+// プロセスリストの解放\r
+void MsFreeProcessList(LIST *o)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               MS_PROCESS *p = LIST_DATA(o, i);\r
+               Free(p);\r
+       }\r
+\r
+       ReleaseList(o);\r
+}\r
+\r
+// プロセスリストの取得 (WinNT 用)\r
+LIST *MsGetProcessListNt()\r
+{\r
+       LIST *o;\r
+       UINT max = 16384;\r
+       DWORD *processes;\r
+       UINT needed, num;\r
+       UINT i;\r
+\r
+       o = NewListFast(MsCompareProcessList);\r
+\r
+       if (ms->nt->EnumProcesses == NULL)\r
+       {\r
+               return o;\r
+       }\r
+\r
+       processes = ZeroMalloc(sizeof(DWORD) * max);\r
+\r
+       if (ms->nt->EnumProcesses(processes, sizeof(DWORD) * max, &needed) == FALSE)\r
+       {\r
+               Free(processes);\r
+               return NULL;\r
+       }\r
+\r
+       num = needed / sizeof(DWORD);\r
+\r
+       for (i = 0;i < num;i++)\r
+       {\r
+               UINT id = processes[i];\r
+               HANDLE h = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ,\r
+                       false, id);\r
+\r
+               if (h != NULL)\r
+               {\r
+                       HINSTANCE hInst = NULL;\r
+                       DWORD needed;\r
+                       if (ms->nt->EnumProcessModules(h, &hInst, sizeof(hInst), &needed))\r
+                       {\r
+                               MS_PROCESS *p = ZeroMalloc(sizeof(MS_PROCESS));\r
+                               ms->nt->GetModuleFileNameExA(h, hInst, p->ExeFilename, sizeof(p->ExeFilename) - 1);\r
+                               ms->nt->GetModuleFileNameExW(h, hInst, p->ExeFilenameW, sizeof(p->ExeFilenameW) / sizeof(wchar_t) - 1);\r
+                               p->ProcessId = id;\r
+                               Add(o, p);\r
+                       }\r
+                       CloseHandle(h);\r
+               }\r
+       }\r
+\r
+       Sort(o);\r
+\r
+       Free(processes);\r
+\r
+       return o;\r
+}\r
+\r
+// プロセスリストの取得 (Win9x 用)\r
+LIST *MsGetProcessList9x()\r
+{\r
+       HANDLE h;\r
+       LIST *o;\r
+       HANDLE (WINAPI *CreateToolhelp32Snapshot)(DWORD, DWORD);\r
+       BOOL (WINAPI *Process32First)(HANDLE, LPPROCESSENTRY32);\r
+       BOOL (WINAPI *Process32Next)(HANDLE, LPPROCESSENTRY32);\r
+\r
+       CreateToolhelp32Snapshot =\r
+               (HANDLE (__stdcall *)(DWORD,DWORD))\r
+               GetProcAddress(ms->hKernel32, "CreateToolhelp32Snapshot");\r
+       Process32First =\r
+               (BOOL (__stdcall *)(HANDLE,LPPROCESSENTRY32))\r
+               GetProcAddress(ms->hKernel32, "Process32First");\r
+       Process32Next =\r
+               (BOOL (__stdcall *)(HANDLE,LPPROCESSENTRY32))\r
+               GetProcAddress(ms->hKernel32, "Process32Next");\r
+\r
+       o = NewListFast(MsCompareProcessList);\r
+\r
+       if (CreateToolhelp32Snapshot != NULL && Process32First != NULL && Process32Next != NULL)\r
+       {\r
+               h = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);\r
+               if (h != INVALID_HANDLE_VALUE)\r
+               {\r
+                       PROCESSENTRY32 e;\r
+                       Zero(&e, sizeof(e));\r
+                       e.dwSize = sizeof(e);\r
+\r
+                       if (Process32First(h, &e))\r
+                       {\r
+                               while (true)\r
+                               {\r
+                                       MS_PROCESS *p = ZeroMalloc(sizeof(MS_PROCESS));\r
+                                       StrCpy(p->ExeFilename, sizeof(p->ExeFilename), e.szExeFile);\r
+                                       StrToUni(p->ExeFilenameW, sizeof(p->ExeFilenameW), p->ExeFilename);\r
+                                       p->ProcessId = e.th32ProcessID;\r
+                                       Add(o, p);\r
+                                       if (Process32Next(h, &e) == false)\r
+                                       {\r
+                                               break;\r
+                                       }\r
+                               }\r
+                       }\r
+                       CloseHandle(h);\r
+               }\r
+       }\r
+\r
+       Sort(o);\r
+\r
+       return o;\r
+}\r
+\r
+// プロセスリストの取得\r
+LIST *MsGetProcessList()\r
+{\r
+       if (MsIsNt() == false)\r
+       {\r
+               // Windows 9x\r
+               return MsGetProcessList9x();\r
+       }\r
+       else\r
+       {\r
+               // Windows NT, 2000, XP\r
+               return MsGetProcessListNt();\r
+       }\r
+}\r
+\r
+// 現在のスレッドを 1 つの CPU で動作するように強制する\r
+void MsSetThreadSingleCpu()\r
+{\r
+       SetThreadAffinityMask(GetCurrentThread(), 1);\r
+}\r
+\r
+// サウンドの再生\r
+void MsPlaySound(char *name)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       char wav[MAX_SIZE];\r
+       char *temp;\r
+       BUF *b;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp), "|%s", name);\r
+\r
+       b = ReadDump(tmp);\r
+       if (b == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       temp = MsGetMyTempDir();\r
+       Format(wav, sizeof(tmp), "%s\\%s", temp, name);\r
+       DumpBuf(b, wav);\r
+\r
+       PlaySound(wav, NULL, SND_ASYNC | SND_FILENAME | SND_NODEFAULT);\r
+\r
+       FreeBuf(b);\r
+}\r
+\r
+// タスクトレイにアイコンを表示する\r
+void MsShowIconOnTray(HWND hWnd, HICON icon, wchar_t *tooltip, UINT msg)\r
+{\r
+       // 引数チェック\r
+       if (hWnd == NULL || icon == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               Zero(&nid, sizeof(nid));\r
+               nid.cbSize = sizeof(nid);\r
+               nid.hWnd = hWnd;\r
+               nid.uID = 1;\r
+               nid.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP | NIF_INFO;\r
+               nid.uCallbackMessage = msg;\r
+               nid.hIcon = icon;\r
+               UniToStr(nid.szTip, sizeof(nid.szTip), tooltip);\r
+               Shell_NotifyIcon(NIM_ADD, &nid);\r
+       }\r
+       else\r
+       {\r
+               Zero(&nid_nt, sizeof(nid_nt));\r
+               nid_nt.cbSize = sizeof(nid_nt);\r
+               nid_nt.hWnd = hWnd;\r
+               nid_nt.uID = 1;\r
+               nid_nt.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP | NIF_INFO;\r
+               nid_nt.uCallbackMessage = msg;\r
+               nid_nt.hIcon = icon;\r
+               UniStrCpy(nid_nt.szTip, sizeof(nid_nt.szTip), tooltip);\r
+               Shell_NotifyIconW(NIM_ADD, &nid_nt);\r
+       }\r
+\r
+       tray_inited = true;\r
+}\r
+\r
+// タスクトレイが初期化されているかどうか確認する\r
+bool MsIsTrayInited()\r
+{\r
+       return tray_inited;\r
+}\r
+\r
+// タスクトレイのアイコンを復元する\r
+void MsRestoreIconOnTray()\r
+{\r
+       if (tray_inited == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               Shell_NotifyIcon(NIM_ADD, &nid);\r
+       }\r
+       else\r
+       {\r
+               Shell_NotifyIconW(NIM_ADD, &nid_nt);\r
+       }\r
+}\r
+\r
+// タスクトレイのアイコンを変更する (いけー)\r
+void MsChangeIconOnTrayEx2(void *icon, wchar_t *tooltip, wchar_t *info_title, wchar_t *info, UINT info_flags)\r
+{\r
+       MsChangeIconOnTrayEx((HICON)icon, tooltip, info_title, info, info_flags);\r
+}\r
+\r
+// タスクトレイのアイコンを変更する\r
+void MsChangeIconOnTray(HICON icon, wchar_t *tooltip)\r
+{\r
+       MsChangeIconOnTrayEx(icon, tooltip, NULL, NULL, NIIF_NONE);\r
+}\r
+void MsChangeIconOnTrayEx(HICON icon, wchar_t *tooltip, wchar_t *info_title, wchar_t *info, UINT info_flags)\r
+{\r
+       bool changed = false;\r
+\r
+       if (tray_inited == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (icon != NULL)\r
+       {\r
+               if (MsIsNt() == false)\r
+               {\r
+                       if (nid.hIcon != icon)\r
+                       {\r
+                               changed = true;\r
+                               nid.hIcon = icon;\r
+                       }\r
+               }\r
+               else\r
+               {\r
+                       if (nid_nt.hIcon != icon)\r
+                       {\r
+                               changed = true;\r
+                               nid_nt.hIcon = icon;\r
+                       }\r
+               }\r
+       }\r
+\r
+       if (tooltip != NULL)\r
+       {\r
+               if (MsIsNt() == false)\r
+               {\r
+                       char tmp[MAX_SIZE];\r
+\r
+                       UniToStr(tmp, sizeof(tmp), tooltip);\r
+\r
+                       if (StrCmp(nid.szTip, tmp) != 0)\r
+                       {\r
+                               StrCpy(nid.szTip, sizeof(nid.szTip), tmp);\r
+                               changed = true;\r
+                       }\r
+               }\r
+               else\r
+               {\r
+                       wchar_t tmp[MAX_SIZE];\r
+\r
+                       UniStrCpy(tmp, sizeof(tmp), tooltip);\r
+\r
+                       if (UniStrCmp(nid_nt.szTip, tmp) != 0)\r
+                       {\r
+                               UniStrCpy(nid_nt.szTip, sizeof(nid_nt.szTip), tmp);\r
+                               changed = true;\r
+                       }\r
+               }\r
+       }\r
+\r
+       if (info_title != NULL && info != NULL)\r
+       {\r
+               if (MsIsNt() == false)\r
+               {\r
+                       char tmp1[MAX_SIZE];\r
+                       char tmp2[MAX_PATH];\r
+\r
+                       UniToStr(tmp1, sizeof(tmp1), info_title);\r
+                       UniToStr(tmp2, sizeof(tmp2), info);\r
+\r
+                       if (StrCmp(nid.szInfo, tmp1) != 0 ||\r
+                               StrCmp(nid.szInfoTitle, tmp2) != 0)\r
+                       {\r
+                               StrCpy(nid.szInfo, sizeof(nid.szInfo), tmp1);\r
+                               StrCpy(nid.szInfoTitle, sizeof(nid.szInfoTitle), tmp2);\r
+                               nid.dwInfoFlags = info_flags;\r
+\r
+                               changed = true;\r
+                       }\r
+               }\r
+               else\r
+               {\r
+                       wchar_t tmp1[MAX_SIZE];\r
+                       wchar_t tmp2[MAX_PATH];\r
+\r
+                       UniStrCpy(tmp1, sizeof(tmp1), info_title);\r
+                       UniStrCpy(tmp2, sizeof(tmp2), info);\r
+\r
+                       if (UniStrCmp(nid_nt.szInfo, tmp1) != 0 ||\r
+                               UniStrCmp(nid_nt.szInfoTitle, tmp2) != 0)\r
+                       {\r
+                               UniStrCpy(nid_nt.szInfo, sizeof(nid_nt.szInfo), tmp1);\r
+                               UniStrCpy(nid_nt.szInfoTitle, sizeof(nid_nt.szInfoTitle), tmp2);\r
+                               nid_nt.dwInfoFlags = info_flags;\r
+\r
+                               changed = true;\r
+                       }\r
+               }\r
+       }\r
+\r
+       if (changed)\r
+       {\r
+               if (MsIsNt() == false)\r
+               {\r
+                       Shell_NotifyIcon(NIM_MODIFY, &nid);\r
+               }\r
+               else\r
+               {\r
+                       Shell_NotifyIconW(NIM_MODIFY, &nid_nt);\r
+               }\r
+       }\r
+}\r
+\r
+// タスクトレイのアイコンを削除する\r
+void MsHideIconOnTray()\r
+{\r
+       if (MsIsNt() == false)\r
+       {\r
+               Shell_NotifyIcon(NIM_DELETE, &nid);\r
+       }\r
+       else\r
+       {\r
+               Shell_NotifyIconW(NIM_DELETE, &nid_nt);\r
+       }\r
+\r
+       tray_inited = false;\r
+}\r
+\r
+// メニュー項目の挿入\r
+bool MsInsertMenu(HMENU hMenu, UINT pos, UINT flags, UINT_PTR id_new_item, wchar_t *lp_new_item)\r
+{\r
+       bool ret;\r
+\r
+       if (MsIsNt())\r
+       {\r
+               ret = InsertMenuW(hMenu, pos, flags, id_new_item, lp_new_item);\r
+       }\r
+       else\r
+       {\r
+               char *s = CopyUniToStr(lp_new_item);\r
+               ret = InsertMenuA(hMenu, pos, flags, id_new_item, s);\r
+               Free(s);\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// メニュー項目の追加\r
+bool MsAppendMenu(HMENU hMenu, UINT flags, UINT_PTR id, wchar_t *str)\r
+{\r
+       bool ret;\r
+\r
+       if (MsIsNt())\r
+       {\r
+               ret = AppendMenuW(hMenu, flags, id, str);\r
+       }\r
+       else\r
+       {\r
+               char *s = CopyUniToStr(str);\r
+               ret = AppendMenuA(hMenu, flags, id, s);\r
+               Free(s);\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// メニュー表示\r
+void MsUserModeTrayMenu(HWND hWnd)\r
+{\r
+       HMENU h;\r
+       POINT p;\r
+       wchar_t tmp[MAX_SIZE];\r
+       wchar_t caption[MAX_SIZE];\r
+       // 引数チェック\r
+       if (hWnd == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // メニューを作成する\r
+       h = CreatePopupMenu();\r
+       MsAppendMenu(h, MF_ENABLED | MF_STRING, 10001, _UU("SVC_USERMODE_MENU_1"));\r
+       MsAppendMenu(h, MF_SEPARATOR, 10002, NULL);\r
+\r
+       if (MsIsNt())\r
+       {\r
+               GetWindowTextW(hWnd, caption, sizeof(caption));\r
+       }\r
+       else\r
+       {\r
+               char tmp[MAX_SIZE];\r
+               GetWindowTextA(hWnd, tmp, sizeof(tmp));\r
+               StrToUni(caption, sizeof(caption), tmp);\r
+       }\r
+\r
+       UniFormat(tmp, sizeof(tmp), _UU("SVC_USERMODE_MENU_2"), caption);\r
+       MsAppendMenu(h, MF_ENABLED | MF_STRING, 10003, tmp);\r
+\r
+       // メニューを表示する\r
+       GetCursorPos(&p);\r
+\r
+       SetForegroundWindow(hWnd);\r
+       TrackPopupMenu(h, TPM_LEFTALIGN, p.x, p.y, 0, hWnd, NULL);\r
+       PostMessage(hWnd, WM_NULL, 0, 0);\r
+\r
+       DestroyMenu(h);\r
+}\r
+\r
+// ユーザーモード用ウインドウプロシージャ\r
+LRESULT CALLBACK MsUserModeWindowProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)\r
+{\r
+       wchar_t tmp[MAX_SIZE];\r
+       char title[MAX_SIZE];\r
+       wchar_t title_w[MAX_SIZE];\r
+       char value_name[MAX_SIZE];\r
+       static UINT taskbar_msg = 0;\r
+       // 引数チェック\r
+       if (hWnd == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       if (msg == taskbar_msg && taskbar_msg != 0)\r
+       {\r
+               // タスクバーが再生成された\r
+               if (MsRegReadInt(REG_CURRENT_USER, SVC_USERMODE_SETTING_KEY, value_name) == 0 &&\r
+                       service_for_9x_mode == false)\r
+               {\r
+                       MsRestoreIconOnTray();\r
+               }\r
+       }\r
+\r
+       switch (msg)\r
+       {\r
+       case WM_ENDSESSION:\r
+               // 再開\r
+               if (wParam == false)\r
+               {\r
+                       break;\r
+               }\r
+       case WM_CREATE:\r
+               // 開始\r
+               exiting = false;\r
+               g_start();\r
+               GetWindowText(hWnd, title, sizeof(title));\r
+               StrToUni(title_w, sizeof(title_w), title);\r
+               UniFormat(tmp, sizeof(tmp), _UU("SVC_TRAY_TOOLTIP"), title);\r
+\r
+               if (taskbar_msg == 0)\r
+               {\r
+                       taskbar_msg = RegisterWindowMessage("TaskbarCreated");\r
+               }\r
+\r
+               Format(value_name, sizeof(value_name), SVC_HIDETRAY_REG_VALUE, title_w);\r
+               if (MsRegReadInt(REG_CURRENT_USER, SVC_USERMODE_SETTING_KEY, value_name) == 0 &&\r
+                       service_for_9x_mode == false)\r
+               {\r
+                       MsShowIconOnTray(hWnd, tray_icon, tmp, WM_APP + 33);\r
+               }\r
+\r
+               break;\r
+       case WM_APP + 33:\r
+               if (wParam == 1)\r
+               {\r
+                       // タスクトレイのアイコンに対する操作\r
+                       switch (lParam)\r
+                       {\r
+                       case WM_RBUTTONDOWN:\r
+                               // 右クリック\r
+                               MsUserModeTrayMenu(hWnd);\r
+                               break;\r
+                       case WM_LBUTTONDBLCLK:\r
+                               // 左ダブルクリック\r
+                               break;\r
+                       }\r
+               }\r
+               break;\r
+       case WM_LBUTTONDOWN:\r
+               MsUserModeTrayMenu(hWnd);\r
+               break;\r
+       case WM_QUERYENDSESSION:\r
+               if (exiting == false)\r
+               {\r
+                       exiting = true;\r
+                       MsHideIconOnTray();\r
+                       g_stop();\r
+                       DestroyWindow(hWnd);\r
+               }\r
+               return TRUE;\r
+       case WM_CLOSE:\r
+               // 停止\r
+               if (exiting == false)\r
+               {\r
+                       exiting = true;\r
+                       g_stop();\r
+                       MsHideIconOnTray();\r
+                       DestroyWindow(hWnd);\r
+               }\r
+               break;\r
+       case WM_DESTROY:\r
+               wnd_end = true;\r
+               break;\r
+       case WM_COMMAND:\r
+               switch (wParam)\r
+               {\r
+               case 10001:\r
+                       GetWindowText(hWnd, title, sizeof(title));\r
+                       StrToUni(title_w, sizeof(title_w), title);\r
+                       // 確認メッセージの表示\r
+                       if (MsgBoxEx(hWnd, MB_ICONINFORMATION | MB_OKCANCEL | MB_DEFBUTTON2 |\r
+                               MB_SYSTEMMODAL, _UU("SVC_HIDE_TRAY_MSG"), title, title) == IDOK)\r
+                       {\r
+                               char tmp[MAX_SIZE];\r
+                               Format(tmp, sizeof(tmp), SVC_HIDETRAY_REG_VALUE, title_w);\r
+                               // レジストリに書き込む\r
+                               MsRegWriteInt(REG_CURRENT_USER, SVC_USERMODE_SETTING_KEY, tmp, 1);\r
+                               // アイコンを消す\r
+                               MsHideIconOnTray();\r
+                       }\r
+                       break;\r
+               case 10003:\r
+                       SendMessage(hWnd, WM_CLOSE, 0, 0);\r
+                       break;\r
+               }\r
+               break;\r
+       }\r
+       return DefWindowProc(hWnd, msg, wParam, lParam);\r
+}\r
+\r
+// PenCore.dll の名前の取得\r
+char *MsGetPenCoreDllFileName()\r
+{\r
+       return PENCORE_DLL_NAME;\r
+}\r
+\r
+// これがユーザーモードかどうか取得\r
+bool MsIsUserMode()\r
+{\r
+       return is_usermode;\r
+}\r
+\r
+// サービス側からユーザーモードの終了を指示\r
+void MsStopUserModeFromService()\r
+{\r
+       if (hWndUsermode != NULL)\r
+       {\r
+               PostMessage(hWndUsermode, WM_CLOSE, 0, 0);\r
+       }\r
+}\r
+\r
+// テストのみ実行 (デバッグ用)\r
+void MsTestOnly()\r
+{\r
+       g_start();\r
+       GetLine(NULL, 0);\r
+       g_stop();\r
+\r
+       _exit(0);\r
+}\r
+\r
+// ユーザーモードとして起動\r
+void MsUserMode(char *title, SERVICE_FUNCTION *start, SERVICE_FUNCTION *stop, UINT icon)\r
+{\r
+       wchar_t *title_w = CopyStrToUni(title);\r
+\r
+       MsUserModeW(title_w, start, stop, icon);\r
+\r
+       Free(title_w);\r
+}\r
+void MsUserModeW(wchar_t *title, SERVICE_FUNCTION *start, SERVICE_FUNCTION *stop, UINT icon)\r
+{\r
+       WNDCLASS wc;\r
+       HINSTANCE hDll;\r
+       HWND hWnd;\r
+       MSG msg;\r
+       INSTANCE *inst;\r
+       char title_a[MAX_PATH];\r
+       // 引数チェック\r
+       if (title == NULL || start == NULL || stop == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       UniToStr(title_a, sizeof(title_a), title);\r
+\r
+       is_usermode = true;\r
+       g_start = start;\r
+       g_stop = stop;\r
+\r
+       inst = NewSingleInstance(NULL);\r
+       if (inst == NULL)\r
+       {\r
+               if (service_for_9x_mode == false)\r
+               {\r
+                       // Win9x サービスモードの場合はエラーを表示しない\r
+                       MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVC_USERMODE_MUTEX"), ms->ExeFileNameW);\r
+               }\r
+               return;\r
+       }\r
+\r
+       if (Is64())\r
+       {\r
+               hDll = MsLoadLibraryAsDataFile(MsGetPenCoreDllFileName());\r
+       }\r
+       else\r
+       {\r
+               hDll = MsLoadLibrary(MsGetPenCoreDllFileName());\r
+       }\r
+\r
+       // アイコン読み込み\r
+       tray_icon = LoadImage(hDll, MAKEINTRESOURCE(icon), IMAGE_ICON, 16, 16,\r
+               (MsIsNt() ? LR_SHARED : 0) | LR_VGACOLOR);\r
+\r
+       // メインウインドウの作成\r
+       Zero(&wc, sizeof(wc));\r
+       wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);\r
+       wc.hCursor = LoadCursor(NULL,IDC_ARROW);\r
+       wc.hIcon = LoadIcon(hDll, MAKEINTRESOURCE(icon));\r
+       wc.hInstance = ms->hInst;\r
+       wc.lpfnWndProc = MsUserModeWindowProc;\r
+       wc.lpszClassName = title_a;\r
+       if (RegisterClass(&wc) == 0)\r
+       {\r
+               return;\r
+       }\r
+\r
+       hWnd = CreateWindow(title_a, title_a, WS_OVERLAPPEDWINDOW,\r
+               CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,\r
+               NULL, NULL, ms->hInst, NULL);\r
+\r
+       if (hWnd == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       hWndUsermode = hWnd;\r
+\r
+       wnd_end = false;\r
+       // ウインドウループ\r
+       while (wnd_end == false)\r
+       {\r
+               GetMessage(&msg, NULL, 0, 0);\r
+               TranslateMessage(&msg);\r
+               DispatchMessage(&msg);\r
+       }\r
+\r
+       FreeSingleInstance(inst);\r
+\r
+       hWndUsermode = NULL;\r
+\r
+       // 強制終了して良い\r
+       _exit(0);\r
+}\r
+\r
+// サービス停止処理メインスレッド\r
+void MsServiceStoperMainThread(THREAD *t, void *p)\r
+{\r
+       // 停止処理\r
+       g_stop();\r
+}\r
+\r
+// サービス停止処理用スレッド\r
+void MsServiceStoperThread(THREAD *t, void *p)\r
+{\r
+       THREAD *thread;\r
+       UINT64 selfkill_timeout = Tick64() + SVC_SELFKILL_TIMEOUT;\r
+\r
+       thread = NewThread(MsServiceStoperMainThread, NULL);\r
+\r
+       // まだ開始中の場合は開始スレッドの終了を待つ\r
+       while (WaitThread(starter_thread, 250) == false)\r
+       {\r
+               if (Tick64() >= selfkill_timeout)\r
+               {\r
+                       // フリーズ時用の自殺\r
+                       _exit(0);\r
+               }\r
+               // 開始処理が完了するまでの間、一定時間ごとに SetServiceStatus を呼び出す\r
+               status.dwWin32ExitCode = 0;\r
+               status.dwWaitHint = 100000;\r
+               status.dwCheckPoint++;\r
+               status.dwCurrentState = SERVICE_STOP_PENDING;\r
+               ms->nt->SetServiceStatus(ssh, &status);\r
+       }\r
+\r
+       ReleaseThread(starter_thread);\r
+       starter_thread = NULL;\r
+\r
+       while (WaitThread(thread, 250) == false)\r
+       {\r
+               if (Tick64() >= selfkill_timeout)\r
+               {\r
+                       // フリーズ時用の自殺\r
+                       _exit(0);\r
+               }\r
+               // 停止処理が完了するまでの間、一定時間ごとに SetServiceStatus を呼び出す\r
+               status.dwWin32ExitCode = 0;\r
+               status.dwWaitHint = 100000;\r
+               status.dwCheckPoint++;\r
+               status.dwCurrentState = SERVICE_STOP_PENDING;\r
+               ms->nt->SetServiceStatus(ssh, &status);\r
+       }\r
+\r
+       ReleaseThread(thread);\r
+\r
+       // 停止が完了したことを報告する\r
+       status.dwWin32ExitCode = 0;\r
+       status.dwWaitHint = 0;\r
+       status.dwCheckPoint = 0;\r
+       status.dwCurrentState = SERVICE_STOPPED;\r
+       ms->nt->SetServiceStatus(ssh, &status);\r
+\r
+       Set(server_stopped_event);\r
+}\r
+\r
+// サービスハンドラ\r
+void CALLBACK MsServiceHandler(DWORD opcode)\r
+{\r
+       switch (opcode)\r
+       {\r
+       case SERVICE_CONTROL_SHUTDOWN:\r
+       case SERVICE_CONTROL_STOP:\r
+               // 停止要求\r
+               status.dwWin32ExitCode = 0;\r
+               status.dwWaitHint = 100000;\r
+               status.dwCheckPoint = 0;\r
+               status.dwCurrentState = SERVICE_STOP_PENDING;\r
+\r
+               // 停止用スレッドを立てる\r
+               service_stopper_thread = NewThread(MsServiceStoperThread, NULL);\r
+               break;\r
+       }\r
+\r
+       ms->nt->SetServiceStatus(ssh, &status);\r
+}\r
+\r
+// サービス開始用スレッド\r
+void MsServiceStarterMainThread(THREAD *t, void *p)\r
+{\r
+       // 開始\r
+       g_start();\r
+}\r
+\r
+// サービスのディスパッチ関数\r
+void CALLBACK MsServiceDispatcher(DWORD argc, LPTSTR *argv)\r
+{\r
+       // サービスの準備\r
+       Zero(&status, sizeof(status));\r
+       status.dwServiceType = SERVICE_WIN32;\r
+       status.dwCurrentState = SERVICE_START_PENDING;\r
+       status.dwControlsAccepted = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_SHUTDOWN;\r
+\r
+       ssh = ms->nt->RegisterServiceCtrlHandler(g_service_name, MsServiceHandler);\r
+\r
+       if (ssh == NULL)\r
+       {\r
+               Alert("RegisterServiceCtrlHandler() Failed.", "MsServiceDispatcher()");\r
+               return;\r
+       }\r
+\r
+       status.dwWaitHint = 10000;\r
+       status.dwCheckPoint = 0;\r
+       status.dwCurrentState = SERVICE_START_PENDING;\r
+       ms->nt->SetServiceStatus(ssh, &status);\r
+\r
+       // サービス開始用スレッドを作成する\r
+       starter_thread = NewThread(MsServiceStarterMainThread, NULL);\r
+\r
+       // 開始完了を報告する\r
+       status.dwWaitHint = 0;\r
+       status.dwCheckPoint = 0;\r
+       status.dwCurrentState = SERVICE_RUNNING;\r
+       ms->nt->SetServiceStatus(ssh, &status);\r
+}\r
+\r
+// サービスとして動作\r
+void MsServiceMode(SERVICE_FUNCTION *start, SERVICE_FUNCTION *stop)\r
+{\r
+       SERVICE_TABLE_ENTRY dispatch_table[] =\r
+       {\r
+               {"", MsServiceDispatcher},\r
+               {NULL, NULL},\r
+       };\r
+       INSTANCE *inst;\r
+       // 引数チェック\r
+       if (start == NULL || stop == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       MsSetErrorModeToSilent();\r
+\r
+       g_start = start;\r
+       g_stop = stop;\r
+\r
+       server_stopped_event = NewEvent();\r
+\r
+       inst = NewSingleInstance(NULL);\r
+       if (inst == NULL)\r
+       {\r
+               MsgBoxEx(NULL, MB_SETFOREGROUND | MB_TOPMOST | MB_SERVICE_NOTIFICATION | MB_OK | MB_ICONEXCLAMATION,\r
+                       _UU("SVC_SERVICE_MUTEX"), g_service_name, ms->ExeFileNameW);\r
+               return;\r
+       }\r
+\r
+       // サービス設定を更新する\r
+       MsUpdateServiceConfig(g_service_name);\r
+\r
+       if (ms->nt->StartServiceCtrlDispatcher(dispatch_table) == false)\r
+       {\r
+               Alert("StartServiceCtrlDispatcher() Failed.", "MsServiceMode()");\r
+               return;\r
+       }\r
+\r
+       MsUpdateServiceConfig(g_service_name);\r
+\r
+       FreeSingleInstance(inst);\r
+\r
+       // サービス終了後は直ちにプロセスを終了する\r
+       Wait(server_stopped_event, INFINITE);\r
+       ReleaseEvent(server_stopped_event);\r
+       WaitThread(service_stopper_thread, INFINITE);\r
+       ReleaseThread(service_stopper_thread);\r
+       server_stopped_event = NULL;\r
+\r
+       _exit(0);\r
+}\r
+\r
+// テストモードとして起動\r
+void MsTestMode(char *title, SERVICE_FUNCTION *start, SERVICE_FUNCTION *stop)\r
+{\r
+       wchar_t *title_w = CopyStrToUni(title);\r
+\r
+       MsTestModeW(title_w, start, stop);\r
+       Free(title_w);\r
+}\r
+void MsTestModeW(wchar_t *title, SERVICE_FUNCTION *start, SERVICE_FUNCTION *stop)\r
+{\r
+       INSTANCE *inst;\r
+       // 引数チェック\r
+       if (title == NULL || start == NULL || stop == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       is_usermode = true;\r
+\r
+       inst = NewSingleInstance(NULL);\r
+       if (inst == NULL)\r
+       {\r
+               // すでに起動している\r
+               MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVC_TEST_MUTEX"), ms->ExeFileNameW);\r
+               return;\r
+       }\r
+\r
+       // 起動\r
+       start();\r
+\r
+       // メッセージ表示\r
+       MsgBoxEx(NULL, MB_ICONINFORMATION | MB_SYSTEMMODAL, _UU("SVC_TEST_MSG"), title);\r
+\r
+       // 停止\r
+       stop();\r
+\r
+       FreeSingleInstance(inst);\r
+}\r
+\r
+// サービスマネージャを呼び出し中のプロセスのプロセス ID を書き込む\r
+void MsWriteCallingServiceManagerProcessId(char *svcname, UINT pid)\r
+{\r
+       char tmp[MAX_PATH];\r
+\r
+       Format(tmp, sizeof(tmp), SVC_CALLING_SM_PROCESS_ID_KEY, svcname);\r
+\r
+       if (pid != 0)\r
+       {\r
+               MsRegWriteInt(REG_LOCAL_MACHINE, tmp, SVC_CALLING_SM_PROCESS_ID_VALUE, pid);\r
+               MsRegWriteInt(REG_CURRENT_USER, tmp, SVC_CALLING_SM_PROCESS_ID_VALUE, pid);\r
+       }\r
+       else\r
+       {\r
+               MsRegDeleteValue(REG_LOCAL_MACHINE, tmp, SVC_CALLING_SM_PROCESS_ID_VALUE);\r
+               MsRegDeleteKey(REG_LOCAL_MACHINE, tmp);\r
+\r
+               MsRegDeleteValue(REG_CURRENT_USER, tmp, SVC_CALLING_SM_PROCESS_ID_VALUE);\r
+               MsRegDeleteKey(REG_CURRENT_USER, tmp);\r
+       }\r
+}\r
+\r
+// サービスマネージャを呼び出し中のプロセス ID を取得する\r
+UINT MsReadCallingServiceManagerProcessId(char *svcname, bool current_user)\r
+{\r
+       char tmp[MAX_PATH];\r
+       // 引数チェック\r
+       if (svcname == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp), SVC_CALLING_SM_PROCESS_ID_KEY, svcname);\r
+\r
+       return MsRegReadInt(current_user ? REG_CURRENT_USER : REG_LOCAL_MACHINE, tmp, SVC_CALLING_SM_PROCESS_ID_VALUE);\r
+}\r
+\r
+// サービスメイン関数\r
+UINT MsService(char *name, SERVICE_FUNCTION *start, SERVICE_FUNCTION *stop, UINT icon)\r
+{\r
+       UINT mode;\r
+       UINT ret = 0;\r
+       char *arg;\r
+       wchar_t *arg_w;\r
+       TOKEN_LIST *t = NULL;\r
+       UNI_TOKEN_LIST *ut = NULL;\r
+       char *service_name;\r
+       wchar_t *service_title;\r
+       wchar_t *service_description;\r
+       wchar_t *service_title_uni;\r
+       char tmp[MAX_SIZE];\r
+       bool restoreReg = false;\r
+       bool silent = false;\r
+       // 引数チェック\r
+       if (name == NULL || start == NULL || stop == NULL)\r
+       {\r
+               return ret;\r
+       }\r
+\r
+       // Mayaqua の開始\r
+       InitMayaqua(false, false, 0, NULL);\r
+\r
+       // MS-IME の停止\r
+       MsDisableIme();\r
+\r
+       // サービスに関する情報を string table から取得\r
+       Format(tmp, sizeof(tmp), SVC_NAME, name);\r
+       service_name = _SS(tmp);\r
+       Format(tmp, sizeof(tmp), SVC_TITLE, name);\r
+       service_title = _UU(tmp);\r
+       service_title_uni = _UU(tmp);\r
+       Format(tmp, sizeof(tmp), SVC_DESCRIPT, name);\r
+       service_description = _UU(tmp);\r
+\r
+       if (StrLen(service_name) == 0 || UniStrLen(service_title) == 0)\r
+       {\r
+               // サービス情報が見つからない\r
+               MsgBoxEx(NULL, MB_ICONSTOP, _UU("SVC_NOT_FOUND"), name);\r
+       }\r
+       else\r
+       {\r
+               wchar_t path[MAX_SIZE];\r
+               // 引数のチェック\r
+               mode = SVC_MODE_NONE;\r
+\r
+               t = GetCommandLineToken();\r
+               arg = NULL;\r
+\r
+               ut = GetCommandLineUniToken();\r
+               arg_w = NULL;\r
+\r
+               if (t->NumTokens >= 1)\r
+               {\r
+                       arg = t->Token[0];\r
+               }\r
+               if(t->NumTokens >= 2)\r
+               {\r
+                       if(StrCmpi(t->Token[1], SVC_ARG_SILENT) == 0)\r
+                       {\r
+                               silent = true;\r
+                       }\r
+               }\r
+\r
+               if (ut->NumTokens >= 1)\r
+               {\r
+                       arg_w = ut->Token[0];\r
+               }\r
+\r
+               if (arg != NULL)\r
+               {\r
+                       if (StrCmpi(arg, SVC_ARG_INSTALL) == 0)\r
+                       {\r
+                               mode = SVC_MODE_INSTALL;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_UNINSTALL) == 0)\r
+                       {\r
+                               mode = SVC_MODE_UNINSTALL;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_START) == 0)\r
+                       {\r
+                               mode = SVC_MODE_START;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_STOP) == 0)\r
+                       {\r
+                               mode = SVC_MODE_STOP;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_TEST) == 0)\r
+                       {\r
+                               mode = SVC_MODE_TEST;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_USERMODE) == 0)\r
+                       {\r
+                               mode = SVC_MODE_USERMODE;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_SETUP_INSTALL) == 0)\r
+                       {\r
+                               mode = SVC_MODE_SETUP_INSTALL;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_SETUP_UNINSTALL) == 0)\r
+                       {\r
+                               mode = SVC_MODE_SETUP_UNINSTALL;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_WIN9X_SERVICE) == 0)\r
+                       {\r
+                               mode = SVC_MODE_WIN9X_SERVICE;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_WIN9X_INSTALL) == 0)\r
+                       {\r
+                               mode = SVC_MODE_WIN9X_INSTALL;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_WIN9X_UNINSTALL) == 0)\r
+                       {\r
+                               mode = SVC_MODE_WIN9X_UNINSTALL;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_TCP) == 0)\r
+                       {\r
+                               mode = SVC_MODE_TCP;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_TCP_SETUP) == 0)\r
+                       {\r
+                               mode = SVC_MODE_TCPSETUP;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_TRAFFIC) == 0)\r
+                       {\r
+                               mode = SVC_MODE_TRAFFIC;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_UIHELP) == 0)\r
+                       {\r
+                               mode = SVC_MODE_UIHELP;\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_USERMODE_SHOWTRAY) == 0)\r
+                       {\r
+                               char tmp[MAX_SIZE];\r
+                               mode = SVC_MODE_USERMODE;\r
+                               Format(tmp, sizeof(tmp), SVC_HIDETRAY_REG_VALUE, service_title);\r
+                               MsRegDeleteValue(REG_CURRENT_USER, SVC_USERMODE_SETTING_KEY, tmp);\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_USERMODE_HIDETRAY) == 0)\r
+                       {\r
+                               char tmp[MAX_SIZE];\r
+                               mode = SVC_MODE_USERMODE;\r
+                               Format(tmp, sizeof(tmp), SVC_HIDETRAY_REG_VALUE, service_title);\r
+                               MsRegWriteInt(REG_CURRENT_USER, SVC_USERMODE_SETTING_KEY, tmp, 1);\r
+                       }\r
+                       if (StrCmpi(arg, SVC_ARG_SERVICE) == 0)\r
+                       {\r
+                               mode = SVC_MODE_SERVICE;\r
+                       }\r
+\r
+                       if (mode != SVC_MODE_NONE)\r
+                       {\r
+                               // Network Config\r
+                               MsInitGlobalNetworkConfig();\r
+                       }\r
+               }\r
+\r
+               // サービスとして実行する際のコマンドライン名を取得する\r
+               UniFormat(path, sizeof(path), SVC_RUN_COMMANDLINE, ms->ExeFileNameW);\r
+\r
+               if ((mode == SVC_MODE_INSTALL || mode == SVC_MODE_UNINSTALL || mode == SVC_MODE_START ||\r
+                       mode == SVC_MODE_STOP || mode == SVC_MODE_SERVICE) &&\r
+                       (ms->IsNt == false))\r
+               {\r
+                       // Windows NT 以外で NT 系のコマンドを使用しようとした\r
+                       MsgBox(NULL, MB_ICONSTOP, _UU("SVC_NT_ONLY"));\r
+               }\r
+               else if ((mode == SVC_MODE_INSTALL || mode == SVC_MODE_UNINSTALL || mode == SVC_MODE_START ||\r
+                       mode == SVC_MODE_STOP || mode == SVC_MODE_SERVICE) &&\r
+                       (ms->IsAdmin == false))\r
+               {\r
+                       // Administrators 権限が無い\r
+                       MsgBox(NULL, MB_ICONEXCLAMATION, _UU("SVC_NOT_ADMIN"));\r
+               }\r
+               else\r
+               {\r
+                       // モードごとに処理を行う\r
+                       switch (mode)\r
+                       {\r
+                       case SVC_MODE_NONE:\r
+                               // 案内メッセージを表示して終了する\r
+                               if (arg_w != NULL && UniEndWith(arg_w, L".uvpn"))\r
+                               {\r
+                                       if (MsgBox(NULL, MB_ICONQUESTION | MB_YESNO, _UU("CM_VPN_FILE_CLICKED")) == IDYES)\r
+                                       {\r
+                                               wchar_t vpncmgr[MAX_PATH];\r
+                                               wchar_t filename[MAX_PATH];\r
+\r
+                                               UniFormat(filename, sizeof(filename), L"\"%s\"", arg_w);\r
+\r
+                                               if (Is64() == false)\r
+                                               {\r
+                                                       UniFormat(vpncmgr, sizeof(vpncmgr), L"%s\\utvpncmgr.exe", MsGetExeDirNameW());\r
+                                               }\r
+                                               else\r
+                                               {\r
+                                                       UniFormat(vpncmgr, sizeof(vpncmgr), L"%s\\utvpncmgr_x64.exe", MsGetExeDirNameW());\r
+                                               }\r
+\r
+                                               RunW(vpncmgr, filename, false, false);\r
+                                       }\r
+                               }\r
+                               else\r
+                               {\r
+                                       MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVC_HELP"),\r
+                                               service_title, service_name, service_title, service_title, service_name, service_title, service_name, service_title, service_name, service_title, service_name, service_title, service_title);\r
+                               }\r
+                               break;\r
+\r
+                       case SVC_MODE_SETUP_INSTALL:\r
+                               // setup.exe インストール モード\r
+                               // 古いものをアンインストールする\r
+                               MsWriteCallingServiceManagerProcessId(service_name, MsGetCurrentProcessId());\r
+                               restoreReg = true;\r
+\r
+                               if (MsIsServiceInstalled(service_name))\r
+                               {\r
+                                       if (MsIsServiceRunning(service_name))\r
+                                       {\r
+                                               MsStopService(service_name);\r
+                                       }\r
+                                       MsUninstallService(service_name);\r
+                               }\r
+                               if (MsInstallServiceW(service_name, service_title, service_description, path) == false)\r
+                               {\r
+                                       ret = 1;\r
+                               }\r
+                               MsStartService(service_name);\r
+                               MsWriteCallingServiceManagerProcessId(service_name, 0);\r
+                               break;\r
+\r
+                       case SVC_MODE_SETUP_UNINSTALL:\r
+                               // setup.exe アンインストール モード\r
+                               MsWriteCallingServiceManagerProcessId(service_name, MsGetCurrentProcessId());\r
+                               restoreReg = true;\r
+\r
+                               if (MsIsServiceInstalled(service_name))\r
+                               {\r
+                                       if (MsIsServiceRunning(service_name))\r
+                                       {\r
+                                               MsStopService(service_name);\r
+                                       }\r
+                                       if (MsUninstallService(service_name) == false)\r
+                                       {\r
+                                               ret = 1;\r
+                                       }\r
+                               }\r
+                               break;\r
+\r
+                       case SVC_MODE_INSTALL:\r
+                               // サービスのインストール\r
+                               // すでにインストールされているかどうか確認する\r
+                               MsWriteCallingServiceManagerProcessId(service_name, MsGetCurrentProcessId());\r
+                               restoreReg = true;\r
+\r
+                               if (MsIsServiceInstalled(service_name))\r
+                               {\r
+                                       // すでにインストールされている\r
+                                       // アンインストールするかどうか確認のメッセージを表示する\r
+                                       if (silent == false)\r
+                                       {\r
+                                               if (MsgBoxEx(NULL, MB_ICONQUESTION | MB_YESNO | MB_DEFBUTTON2, _UU("SVC_ALREADY_INSTALLED"),\r
+                                                       service_title, service_name) == IDNO)\r
+                                               {\r
+                                                       // 処理をキャンセルする\r
+                                                       break;\r
+                                               }\r
+                                       }\r
+                                       // 既存のサービスが動作しているか?\r
+                                       if (MsIsServiceRunning(service_name))\r
+                                       {\r
+                                               // 停止を試みる\r
+                                               if (MsStopService(service_name) == false)\r
+                                               {\r
+                                                       // 停止に失敗した\r
+                                                       MsgBoxEx(NULL, MB_ICONSTOP, _UU("SVC_STOP_FAILED"),\r
+                                                               service_title, service_name);\r
+                                                       break;\r
+                                               }\r
+                                       }\r
+                                       // アンインストールする\r
+                                       if (MsUninstallService(service_name) == false)\r
+                                       {\r
+                                               // アンインストールに失敗した\r
+                                               MsgBoxEx(NULL, MB_ICONSTOP, _UU("SVC_UNINSTALL_FAILED"),\r
+                                                       service_title, service_name);\r
+                                               break;\r
+                                       }\r
+                               }\r
+\r
+                               // インストールを行う\r
+                               if (MsInstallServiceW(service_name, service_title, service_description, path) == false)\r
+                               {\r
+                                       // インストールに失敗した\r
+                                       MsgBoxEx(NULL, MB_ICONSTOP, _UU("SVC_INSTALL_FAILED"),\r
+                                               service_title, service_name);\r
+                                       break;\r
+                               }\r
+\r
+                               // サービスを開始する\r
+                               if (MsStartService(service_name) == false)\r
+                               {\r
+                                       // 開始に失敗した\r
+                                       MsgBoxEx(NULL, MB_ICONEXCLAMATION, _UU("SVC_INSTALL_FAILED_2"),\r
+                                               service_title, service_name, path);\r
+                                       break;\r
+                               }\r
+\r
+                               // すべて成功した\r
+                               if(silent == false)\r
+                               {\r
+                                       MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVC_INSTALL_OK"),\r
+                                               service_title, service_name, path);\r
+                               }\r
+                               break;\r
+\r
+                       case SVC_MODE_UNINSTALL:\r
+                               // サービスのアンインストール\r
+                               // すでにインストールされているかどうか確認する\r
+                               MsWriteCallingServiceManagerProcessId(service_name, MsGetCurrentProcessId());\r
+                               restoreReg = true;\r
+\r
+                               if (MsIsServiceInstalled(service_name) == false)\r
+                               {\r
+                                       if(silent == false)\r
+                                       {\r
+                                               MsgBoxEx(NULL, MB_ICONEXCLAMATION, _UU("SVC_NOT_INSTALLED"),\r
+                                                       service_title, service_name, path);\r
+                                       }\r
+                                       break;\r
+                               }\r
+\r
+                               // サービスが起動中の場合は停止する\r
+                               if (MsIsServiceRunning(service_name))\r
+                               {\r
+                                       // サービスを停止する\r
+                                       if (MsStopService(service_name) == false)\r
+                                       {\r
+                                               // 停止に失敗した\r
+                                               MsgBoxEx(NULL, MB_ICONSTOP, _UU("SVC_STOP_FAILED"),\r
+                                                       service_title, service_name);\r
+                                               break;\r
+                                       }\r
+                               }\r
+\r
+                               // サービスをアンインストールする\r
+                               if (MsUninstallService(service_name) == false)\r
+                               {\r
+                                       MsgBoxEx(NULL, MB_ICONSTOP, _UU("SVC_UNINSTALL_FAILED"),\r
+                                               service_title, service_name);\r
+                                       break;\r
+                               }\r
+\r
+                               // すべて成功した\r
+                               if(silent == false)\r
+                               {\r
+                                       MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVC_UNINSTALL_OK"),\r
+                                               service_title, service_name);\r
+                               }\r
+                               break;\r
+\r
+                       case SVC_MODE_START:\r
+                               // サービスの開始\r
+                               MsWriteCallingServiceManagerProcessId(service_name, MsGetCurrentProcessId());\r
+                               restoreReg = true;\r
+\r
+                               if (MsIsServiceInstalled(service_name) == false)\r
+                               {\r
+                                       // サービスはインストールされていない\r
+                                       MsgBoxEx(NULL, MB_ICONEXCLAMATION, _UU("SVC_NOT_INSTALLED"),\r
+                                               service_title, service_name);\r
+                                       break;\r
+                               }\r
+\r
+                               // サービスが起動中かどうか確認する\r
+                               if (MsIsServiceRunning(service_name))\r
+                               {\r
+                                       // サービスが起動中\r
+                                       if(silent == false)\r
+                                       {\r
+                                               MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVR_ALREADY_START"),\r
+                                                       service_title, service_name);\r
+                                       }\r
+                                       break;\r
+                               }\r
+\r
+                               // サービスを起動する\r
+                               if (MsStartService(service_name) == false)\r
+                               {\r
+                                       // 開始に失敗した\r
+                                       MsgBoxEx(NULL, MB_ICONEXCLAMATION, _UU("SVC_START_FAILED"),\r
+                                               service_title, service_name);\r
+                                       break;\r
+                               }\r
+\r
+                               // すべて成功した\r
+                               if(silent == false)\r
+                               {\r
+                                       MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVC_START_OK"),\r
+                                               service_title, service_name);\r
+                               }\r
+                               break;\r
+\r
+                       case SVC_MODE_STOP:\r
+                               // サービスの停止\r
+                               MsWriteCallingServiceManagerProcessId(service_name, MsGetCurrentProcessId());\r
+                               restoreReg = true;\r
+\r
+                               if (MsIsServiceInstalled(service_name) == false)\r
+                               {\r
+                                       // サービスはインストールされていない\r
+                                       if(silent == false)\r
+                                       {\r
+                                               MsgBoxEx(NULL, MB_ICONEXCLAMATION, _UU("SVC_NOT_INSTALLED"),\r
+                                                       service_title, service_name);\r
+                                       }\r
+                                       break;\r
+                               }\r
+\r
+                               // サービスが起動中かどうか確認する\r
+                               if (MsIsServiceRunning(service_name) == false)\r
+                               {\r
+                                       // サービスが停止中\r
+                                       if(silent == false)\r
+                                       {\r
+                                               MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVC_ALREADY_STOP"),\r
+                                                       service_title, service_name);\r
+                                       }\r
+                                       break;\r
+                               }\r
+                               // サービスを停止する\r
+                               if (MsStopService(service_name) == false)\r
+                               {\r
+                                       // 停止に失敗した\r
+                                       MsgBoxEx(NULL, MB_ICONEXCLAMATION, _UU("SVC_STOP_FAILED"),\r
+                                               service_title, service_name);\r
+                                       break;\r
+                               }\r
+\r
+                               // すべて成功した\r
+                               if(silent == false)\r
+                               {\r
+                                       MsgBoxEx(NULL, MB_ICONINFORMATION, _UU("SVC_STOP_OK"),\r
+                                               service_title, service_name);\r
+                               }\r
+                               break;\r
+\r
+                       case SVC_MODE_TEST:\r
+                               // テストモード\r
+                               MsTestModeW(service_title, start, stop);\r
+                               break;\r
+\r
+                       case SVC_MODE_WIN9X_SERVICE:\r
+                               // Win9x サービスモード\r
+                               // (タスクトレイのアイコンを無条件で非表示にする)\r
+                               if (MsIsNt())\r
+                               {\r
+                                       // Windows 2000 以降では動作させない\r
+                                       break;\r
+                               }\r
+                               service_for_9x_mode = true;\r
+                       case SVC_MODE_USERMODE:\r
+                               // ユーザーモード\r
+                               MsUserModeW(service_title, start, stop, icon);\r
+                               break;\r
+\r
+                       case SVC_MODE_WIN9X_INSTALL:\r
+                               // Win9x インストールモード\r
+                               MsWriteCallingServiceManagerProcessId(service_name, MsGetCurrentProcessId());\r
+                               restoreReg = true;\r
+\r
+                               if (MsIsNt() == false)\r
+                               {\r
+                                       // レジストリキーの追加\r
+                                       char cmdline[MAX_PATH];\r
+                                       Format(cmdline, sizeof(cmdline), "\"%s\" %s",\r
+                                               MsGetExeFileName(), SVC_ARG_WIN9X_SERVICE);\r
+                                       MsRegWriteStr(REG_LOCAL_MACHINE, WIN9X_SVC_REGKEY_1,\r
+                                               name, cmdline);\r
+                                       MsRegWriteStr(REG_LOCAL_MACHINE, WIN9X_SVC_REGKEY_2,\r
+                                               name, cmdline);\r
+\r
+                                       // 実行\r
+                                       Run(MsGetExeFileName(), SVC_ARG_WIN9X_SERVICE, false, false);\r
+                               }\r
+                               break;\r
+\r
+                       case SVC_MODE_WIN9X_UNINSTALL:\r
+                               // Win9x アンインストールモード\r
+                               MsWriteCallingServiceManagerProcessId(service_name, MsGetCurrentProcessId());\r
+                               restoreReg = true;\r
+\r
+                               if (MsIsNt() == false)\r
+                               {\r
+                                       // レジストリキーの削除\r
+                                       MsRegDeleteValue(REG_LOCAL_MACHINE, WIN9X_SVC_REGKEY_1,\r
+                                               name);\r
+                                       MsRegDeleteValue(REG_LOCAL_MACHINE, WIN9X_SVC_REGKEY_2,\r
+                                               name);\r
+\r
+                                       // 自分以外のすべてのプロセスを終了\r
+                                       MsKillOtherInstance();\r
+                               }\r
+                               break;\r
+\r
+                       case SVC_MODE_SERVICE:\r
+                               // サービスとして動作\r
+                               StrCpy(g_service_name, sizeof(g_service_name), service_name);\r
+                               MsServiceMode(start, stop);\r
+                               break;\r
+\r
+                       case SVC_MODE_TCP:\r
+                               // TCP ユーティリティ\r
+                               InitCedar();\r
+                               InitWinUi(service_title_uni, NULL, 0);\r
+                               ShowTcpIpConfigUtil(NULL, true);\r
+                               FreeWinUi();\r
+                               FreeCedar();\r
+                               break;\r
+\r
+                       case SVC_MODE_TCPSETUP:\r
+                               // TCP 最適化モード (インストーラから呼ばれる)\r
+                               InitCedar();\r
+                               InitWinUi(service_title_uni, NULL, 0);\r
+                               ShowTcpIpConfigUtil(NULL, false);\r
+                               FreeWinUi();\r
+                               FreeCedar();\r
+                               break;\r
+\r
+                       case SVC_MODE_TRAFFIC:\r
+                               // 通信スループット測定ツール\r
+                               InitCedar();\r
+                               InitWinUi(service_title_uni, NULL, 0);\r
+                               CmTraffic(NULL);\r
+                               FreeWinUi();\r
+                               FreeCedar();\r
+                               break;\r
+\r
+                       case SVC_MODE_UIHELP:\r
+                               // UI Helper の起動\r
+                               CnStart();\r
+                               break;\r
+                       }\r
+\r
+               }\r
+               FreeToken(t);\r
+               UniFreeToken(ut);\r
+\r
+               if (restoreReg)\r
+               {\r
+                       MsWriteCallingServiceManagerProcessId(service_name, 0);\r
+               }\r
+       }\r
+\r
+       FreeMayaqua();\r
+\r
+       return 0;\r
+}\r
+\r
+// 指定したセッションのユーザー名を取得する\r
+wchar_t *MsGetSessionUserName(UINT session_id)\r
+{\r
+       if (MsIsTerminalServiceInstalled() || MsIsUserSwitchingInstalled())\r
+       {\r
+               wchar_t *ret;\r
+               wchar_t *name;\r
+               UINT size = 0;\r
+               if (ms->nt->WTSQuerySessionInformation(WTS_CURRENT_SERVER_HANDLE, session_id,\r
+                       WTSUserName, (wchar_t *)&name, &size) == false)\r
+               {\r
+                       return NULL;\r
+               }\r
+\r
+               if (name == NULL || UniStrLen(name) == 0)\r
+               {\r
+                       ret = NULL;\r
+               }\r
+               else\r
+               {\r
+                       ret = UniCopyStr(name);\r
+               }\r
+\r
+               ms->nt->WTSFreeMemory(name);\r
+\r
+               return ret;\r
+       }\r
+       return NULL;\r
+}\r
+\r
+// 現在のデスクトップが VNC で利用可能かどうか取得する\r
+bool MsIsCurrentDesktopAvailableForVnc()\r
+{\r
+       if (MsIsNt() == false)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (MsIsCurrentTerminalSessionActive() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (ms->nt->OpenDesktopA == NULL ||\r
+               ms->nt->CloseDesktop == NULL ||\r
+               ms->nt->SwitchDesktop == NULL)\r
+       {\r
+               return true;\r
+       }\r
+       else\r
+       {\r
+               HDESK hDesk = ms->nt->OpenDesktopA("default", 0, false, DESKTOP_SWITCHDESKTOP);\r
+               bool ret;\r
+\r
+               if (hDesk == NULL)\r
+               {\r
+                       return false;\r
+               }\r
+\r
+               ret = ms->nt->SwitchDesktop(hDesk);\r
+               ms->nt->CloseDesktop(hDesk);\r
+\r
+               return ret;\r
+       }\r
+}\r
+\r
+// 現在のターミナルセッションがアクティブかどうか取得する\r
+bool MsIsCurrentTerminalSessionActive()\r
+{\r
+       return MsIsTerminalSessionActive(MsGetCurrentTerminalSessionId());\r
+}\r
+\r
+// 指定したターミナルセッションがアクティブかどうか取得する\r
+bool MsIsTerminalSessionActive(UINT session_id)\r
+{\r
+       if (MsIsTerminalServiceInstalled() || MsIsUserSwitchingInstalled())\r
+       {\r
+               UINT *status = NULL;\r
+               UINT size = sizeof(status);\r
+               bool active = true;\r
+\r
+               if (ms->nt->WTSQuerySessionInformation(WTS_CURRENT_SERVER_HANDLE, session_id,\r
+                       WTSConnectState, (wchar_t *)&status, &size) == false)\r
+               {\r
+                       return true;\r
+               }\r
+\r
+               switch (*status)\r
+               {\r
+               case WTSDisconnected:\r
+               case WTSShadow:\r
+               case WTSIdle:\r
+               case WTSDown:\r
+               case WTSReset:\r
+                       active = false;\r
+                       break;\r
+               }\r
+\r
+               ms->nt->WTSFreeMemory(status);\r
+\r
+               return active;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 現在のターミナルセッション ID を取得する\r
+UINT MsGetCurrentTerminalSessionId()\r
+{\r
+       if (MsIsTerminalServiceInstalled() || MsIsUserSwitchingInstalled())\r
+       {\r
+               UINT ret;\r
+               UINT *session_id = NULL;\r
+               UINT size = sizeof(session_id);\r
+               if (ms->nt->WTSQuerySessionInformation(WTS_CURRENT_SERVER_HANDLE, WTS_CURRENT_SESSION,\r
+                       WTSSessionId, (wchar_t *)&session_id, &size) == false)\r
+               {\r
+                       return 0;\r
+               }\r
+\r
+               ret = *session_id;\r
+\r
+               ms->nt->WTSFreeMemory(session_id);\r
+\r
+               return ret;\r
+       }\r
+\r
+       return 0;\r
+}\r
+\r
+// ターミナルサービスがインストールされていて複数セッションがログイン可能かどうか調べる\r
+bool MsIsTerminalServiceMultiUserInstalled()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+       OSVERSIONINFOEX i;\r
+       if (MsIsTerminalServiceInstalled() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (OS_IS_SERVER(info->OsType) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&i, sizeof(i));\r
+       i.dwOSVersionInfoSize = sizeof(i);\r
+       if (GetVersionEx((OSVERSIONINFO *)&i) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (i.wSuiteMask & VER_SUITE_SINGLEUSERTS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// ユーザー切り替えがインストールされているかどうか調べる\r
+bool MsIsUserSwitchingInstalled()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+       OSVERSIONINFOEX i;\r
+\r
+       if (OS_IS_WINDOWS_NT(info->OsType) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (ms->nt->WTSDisconnectSession == NULL ||\r
+               ms->nt->WTSFreeMemory == NULL ||\r
+               ms->nt->WTSQuerySessionInformation == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) < 2)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&i, sizeof(i));\r
+       i.dwOSVersionInfoSize = sizeof(i);\r
+       if (GetVersionEx((OSVERSIONINFO *)&i) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (i.wSuiteMask & VER_SUITE_SINGLEUSERTS)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// リモートデスクトップを有効にする\r
+bool MsEnableRemoteDesktop()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+\r
+       if (MsIsRemoteDesktopAvailable() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsRemoteDesktopEnabled())\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) == 2)\r
+       {\r
+               // Windows 2000\r
+               return false;\r
+       }\r
+\r
+       if (MsRegWriteInt(REG_LOCAL_MACHINE,\r
+               "SYSTEM\\CurrentControlSet\\Control\\Terminal Server",\r
+               "fDenyTSConnections", 0) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsVista())\r
+       {\r
+               if (MsRegWriteInt(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Terminal Server\\WinStations\\RDP-Tcp",\r
+                       "UserAuthentication", 0) == false)\r
+               {\r
+                       return false;\r
+               }\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// リモートデスクトップが有効かどうか調べる\r
+bool MsIsRemoteDesktopEnabled()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+\r
+       if (MsIsRemoteDesktopAvailable() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) == 2)\r
+       {\r
+               // Windows 2000\r
+               return MsIsServiceRunning("TermService");\r
+       }\r
+       else\r
+       {\r
+               // Windows XP 以降\r
+               bool b = MsRegReadInt(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Terminal Server",\r
+                       "fDenyTSConnections");\r
+\r
+               if (MsIsVista() == false)\r
+               {\r
+                       return b ? false : true;\r
+               }\r
+               else\r
+               {\r
+                       if (b)\r
+                       {\r
+                               return false;\r
+                       }\r
+                       else\r
+                       {\r
+                               if (MsRegReadInt(REG_LOCAL_MACHINE,\r
+                                       "SYSTEM\\CurrentControlSet\\Control\\Terminal Server\\WinStations\\RDP-Tcp",\r
+                                       "UserAuthentication"))\r
+                               {\r
+                                       return false;\r
+                               }\r
+                               else\r
+                               {\r
+                                       return true;\r
+                               }\r
+                       }\r
+               }\r
+       }\r
+}\r
+\r
+// レジストリ操作によってリモートデスクトップが利用可能になるかどうか調べる\r
+bool MsIsRemoteDesktopCanEnableByRegistory()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+       if (MsIsRemoteDesktopAvailable() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) == 2)\r
+       {\r
+               // Windows 2000\r
+               return false;\r
+       }\r
+       else\r
+       {\r
+               // それ以外\r
+               return true;\r
+       }\r
+}\r
+\r
+// Windows 2000 かどうか調べる\r
+bool MsIsWin2000()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+\r
+       if (OS_IS_WINDOWS_NT(info->OsType) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) == 2)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// Windows 2000 以降かどうか調べる\r
+bool MsIsWin2000OrGreater()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+\r
+       if (OS_IS_WINDOWS_NT(info->OsType) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) >= 2)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// リモートデスクトップが利用可能かどうか調べる\r
+bool MsIsRemoteDesktopAvailable()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+       if (MsIsTerminalServiceInstalled() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) == 2)\r
+       {\r
+               // Windows 2000\r
+               if (info->OsType == 2200)\r
+               {\r
+                       // Windows 2000 Professional\r
+                       return false;\r
+               }\r
+               else\r
+               {\r
+                       // Windows 2000 サーバー系\r
+                       return true;\r
+               }\r
+       }\r
+       else if (GET_KETA(info->OsType, 100) == 3)\r
+       {\r
+               // Windows XP\r
+               if (info->OsType == OSTYPE_WINDOWS_XP_HOME)\r
+               {\r
+                       // Home Edition\r
+                       return false;\r
+               }\r
+               else\r
+               {\r
+                       // Professional Edition\r
+                       return true;\r
+               }\r
+       }\r
+       else if (GET_KETA(info->OsType, 100) == 4)\r
+       {\r
+               // Windows Server 2003\r
+               return true;\r
+       }\r
+       else if (GET_KETA(info->OsType, 100) >= 5)\r
+       {\r
+               // Windows Vista 以降\r
+               OSVERSIONINFOEX i;\r
+\r
+               Zero(&i, sizeof(i));\r
+               i.dwOSVersionInfoSize = sizeof(i);\r
+               if (GetVersionEx((OSVERSIONINFO *)&i) == false)\r
+               {\r
+                       return false;\r
+               }\r
+\r
+               if (i.wSuiteMask & VER_SUITE_PERSONAL)\r
+               {\r
+                       // Home 系\r
+                       return false;\r
+               }\r
+               else\r
+               {\r
+                       return true;\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// ターミナルサービスがインストールされているかどうか調べる\r
+bool MsIsTerminalServiceInstalled()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+       OSVERSIONINFOEX i;\r
+\r
+       if (OS_IS_WINDOWS_NT(info->OsType) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (ms->nt->WTSDisconnectSession == NULL ||\r
+               ms->nt->WTSFreeMemory == NULL ||\r
+               ms->nt->WTSQuerySessionInformation == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(info->OsType, 100) < 2)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&i, sizeof(i));\r
+       i.dwOSVersionInfoSize = sizeof(i);\r
+       if (GetVersionEx((OSVERSIONINFO *)&i) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (i.wSuiteMask & VER_SUITE_TERMINAL || i.wSuiteMask & VER_SUITE_SINGLEUSERTS)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// サービスを停止する\r
+bool MsStopService(char *name)\r
+{\r
+       SC_HANDLE sc, service;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (ms->IsNt == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       sc = ms->nt->OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);\r
+       if (sc == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       service = ms->nt->OpenService(sc, name, SERVICE_ALL_ACCESS);\r
+       if (service != NULL)\r
+       {\r
+               SERVICE_STATUS st;\r
+               ret = ms->nt->ControlService(service, SERVICE_CONTROL_STOP, &st);\r
+\r
+               ms->nt->CloseServiceHandle(service);\r
+       }\r
+\r
+       if (ret)\r
+       {\r
+               UINT64 end = Tick64() + 10000ULL;\r
+               while (Tick64() < end)\r
+               {\r
+                       if (MsIsServiceRunning(name) == false)\r
+                       {\r
+                               break;\r
+                       }\r
+\r
+                       SleepThread(250);\r
+               }\r
+       }\r
+\r
+       ms->nt->CloseServiceHandle(sc);\r
+       return ret;\r
+}\r
+\r
+// サービスを起動する\r
+bool MsStartService(char *name)\r
+{\r
+       return MsStartServiceEx(name, NULL);\r
+}\r
+bool MsStartServiceEx(char *name, UINT *error_code)\r
+{\r
+       SC_HANDLE sc, service;\r
+       bool ret = false;\r
+       static UINT dummy = 0;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (ms->IsNt == false)\r
+       {\r
+               return false;\r
+       }\r
+       if (error_code == NULL)\r
+       {\r
+               error_code = &dummy;\r
+       }\r
+\r
+       *error_code = 0;\r
+\r
+       sc = ms->nt->OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);\r
+       if (sc == NULL)\r
+       {\r
+               *error_code = GetLastError();\r
+               return false;\r
+       }\r
+\r
+       service = ms->nt->OpenService(sc, name, SERVICE_ALL_ACCESS);\r
+       if (service != NULL)\r
+       {\r
+               ret = ms->nt->StartService(service, 0, NULL);\r
+\r
+               ms->nt->CloseServiceHandle(service);\r
+       }\r
+       else\r
+       {\r
+               *error_code = GetLastError();\r
+       }\r
+\r
+       if (ret)\r
+       {\r
+               UINT64 end = Tick64() + 10000ULL;\r
+               while (Tick64() < end)\r
+               {\r
+                       if (MsIsServiceRunning(name))\r
+                       {\r
+                               break;\r
+                       }\r
+\r
+                       SleepThread(250);\r
+               }\r
+       }\r
+\r
+       ms->nt->CloseServiceHandle(sc);\r
+       return ret;\r
+}\r
+\r
+// サービスが起動しているかどうか取得する\r
+bool MsIsServiceRunning(char *name)\r
+{\r
+       SC_HANDLE sc, service;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (name == NULL || IsEmptyStr(name))\r
+       {\r
+               return false;\r
+       }\r
+       if (ms->IsNt == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       sc = ms->nt->OpenSCManager(NULL, NULL, GENERIC_READ);\r
+       if (sc == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       service = ms->nt->OpenService(sc, name, GENERIC_READ);\r
+       if (service != NULL)\r
+       {\r
+               SERVICE_STATUS st;\r
+               Zero(&st, sizeof(st));\r
+               if (ms->nt->QueryServiceStatus(service, &st))\r
+               {\r
+                       switch (st.dwCurrentState)\r
+                       {\r
+                       case SERVICE_CONTINUE_PENDING:\r
+                       case SERVICE_PAUSE_PENDING:\r
+                       case SERVICE_PAUSED:\r
+                       case SERVICE_RUNNING:\r
+                       case SERVICE_START_PENDING:\r
+                       case SERVICE_STOP_PENDING:\r
+                               ret = true;\r
+                               break;\r
+                       }\r
+               }\r
+\r
+               ms->nt->CloseServiceHandle(service);\r
+       }\r
+\r
+       ms->nt->CloseServiceHandle(sc);\r
+       return ret;\r
+}\r
+\r
+// サービスをアンインストールする\r
+bool MsUninstallService(char *name)\r
+{\r
+       SC_HANDLE sc, service;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (ms->IsNt == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       MsStopService(name);\r
+\r
+       sc = ms->nt->OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);\r
+       if (sc == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       service = ms->nt->OpenService(sc, name, SERVICE_ALL_ACCESS);\r
+       if (service != NULL)\r
+       {\r
+               if (ms->nt->DeleteService(service))\r
+               {\r
+                       ret = true;\r
+               }\r
+               ms->nt->CloseServiceHandle(service);\r
+       }\r
+\r
+       ms->nt->CloseServiceHandle(sc);\r
+\r
+       if (ret)\r
+       {\r
+               SleepThread(2000);\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// サービス設定を更新する\r
+bool MsUpdateServiceConfig(char *name)\r
+{\r
+       SC_HANDLE sc, service;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // Windows 起動直後かどうか (デッドロック防止)\r
+       if (timeGetTime() <= (60 * 30 * 1000))\r
+       {\r
+               if (MsRegReadInt(REG_LOCAL_MACHINE, "Software\\SoftEther Corporation\\Update Service Config", name) != 0)\r
+               {\r
+                       return false;\r
+               }\r
+       }\r
+\r
+       sc = ms->nt->OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);\r
+       if (sc == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       service = ms->nt->OpenService(sc, name, SERVICE_ALL_ACCESS);\r
+       if (service != NULL)\r
+       {\r
+               if (GET_KETA(GetOsInfo()->OsType, 100) >= 2)\r
+               {\r
+                       SERVICE_FAILURE_ACTIONS action;\r
+                       SC_ACTION *e;\r
+                       Zero(&action, sizeof(action));\r
+                       e = ZeroMalloc(sizeof(SC_ACTION) * 3);\r
+                       e[0].Delay = 10000; e[0].Type = SC_ACTION_RESTART;\r
+                       e[1].Delay = 10000; e[1].Type = SC_ACTION_RESTART;\r
+                       e[2].Delay = 10000; e[2].Type = SC_ACTION_RESTART;\r
+                       action.cActions = 3;\r
+                       action.lpsaActions = e;\r
+                       action.dwResetPeriod = 1 * 60 * 60 * 24;\r
+                       ms->nt->ChangeServiceConfig2(service, SERVICE_CONFIG_FAILURE_ACTIONS, &action);\r
+\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "Software\\SoftEther Corporation\\Update Service Config", name, 1);\r
+               }\r
+               ms->nt->CloseServiceHandle(service);\r
+       }\r
+\r
+       ms->nt->CloseServiceHandle(sc);\r
+\r
+       return true;\r
+}\r
+\r
+// サービスをインストールする\r
+bool MsInstallService(char *name, char *title, wchar_t *description, char *path)\r
+{\r
+       wchar_t title_w[MAX_PATH];\r
+       wchar_t path_w[MAX_PATH];\r
+       // 引数チェック\r
+       if (name == NULL || title == NULL || path == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       StrToUni(title_w, sizeof(title_w), title);\r
+       StrToUni(path_w, sizeof(path_w), path);\r
+\r
+       return MsInstallServiceW(name, title_w, description, path_w);\r
+}\r
+bool MsInstallServiceW(char *name, wchar_t *title, wchar_t *description, wchar_t *path)\r
+{\r
+       return MsInstallServiceExW(name, title, description, path, NULL);\r
+}\r
+bool MsInstallServiceExW(char *name, wchar_t *title, wchar_t *description, wchar_t *path, UINT *error_code)\r
+{\r
+       SC_HANDLE sc, service;\r
+       bool ret = false;\r
+       wchar_t name_w[MAX_SIZE];\r
+       static UINT temp_int = 0;\r
+       // 引数チェック\r
+       if (name == NULL || title == NULL || path == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (ms->IsNt == false)\r
+       {\r
+               return false;\r
+       }\r
+       if (error_code == NULL)\r
+       {\r
+               error_code = &temp_int;\r
+       }\r
+\r
+       *error_code = 0;\r
+\r
+       StrToUni(name_w, sizeof(name_w), name);\r
+\r
+       sc = ms->nt->OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);\r
+       if (sc == NULL)\r
+       {\r
+               *error_code = GetLastError();\r
+               return false;\r
+       }\r
+\r
+       service = ms->nt->CreateServiceW(sc, name_w, title, SERVICE_ALL_ACCESS,\r
+               SERVICE_WIN32_OWN_PROCESS | SERVICE_INTERACTIVE_PROCESS , SERVICE_AUTO_START,\r
+               SERVICE_ERROR_NORMAL, path, NULL, NULL, NULL, NULL, NULL);\r
+\r
+       if (service != NULL)\r
+       {\r
+               ret = true;\r
+\r
+               if (GET_KETA(GetOsInfo()->OsType, 100) >= 2)\r
+               {\r
+                       SERVICE_DESCRIPTIONW d;\r
+                       SERVICE_FAILURE_ACTIONS action;\r
+                       SC_ACTION *e;\r
+                       Zero(&d, sizeof(d));\r
+                       d.lpDescription = description;\r
+                       ms->nt->ChangeServiceConfig2(service, SERVICE_CONFIG_DESCRIPTION, &d);\r
+                       Zero(&action, sizeof(action));\r
+                       e = ZeroMalloc(sizeof(SC_ACTION) * 3);\r
+                       e[0].Delay = 10000; e[0].Type = SC_ACTION_RESTART;\r
+                       e[1].Delay = 10000; e[1].Type = SC_ACTION_RESTART;\r
+                       e[2].Delay = 10000; e[2].Type = SC_ACTION_RESTART;\r
+                       action.cActions = 3;\r
+                       action.lpsaActions = e;\r
+                       action.dwResetPeriod = 1 * 60 * 60 * 24;\r
+                       ms->nt->ChangeServiceConfig2(service, SERVICE_CONFIG_FAILURE_ACTIONS, &action);\r
+\r
+                       Free(e);\r
+               }\r
+\r
+               ms->nt->CloseServiceHandle(service);\r
+       }\r
+       else\r
+       {\r
+               *error_code = GetLastError();\r
+       }\r
+\r
+       ms->nt->CloseServiceHandle(sc);\r
+\r
+       if (ret)\r
+       {\r
+               SleepThread(2000);\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// 指定したサービスがインストールされているかどうか調べる\r
+bool MsIsServiceInstalled(char *name)\r
+{\r
+       SC_HANDLE sc;\r
+       SC_HANDLE service;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (ms->IsNt == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       sc = ms->nt->OpenSCManager(NULL, NULL, GENERIC_READ);\r
+       if (sc == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       service = ms->nt->OpenService(sc, name, GENERIC_READ);\r
+       if (service != NULL)\r
+       {\r
+               ret = true;\r
+       }\r
+\r
+       ms->nt->CloseServiceHandle(service);\r
+       ms->nt->CloseServiceHandle(sc);\r
+\r
+       return ret;\r
+}\r
+\r
+// プロセスの強制終了\r
+void MsTerminateProcess()\r
+{\r
+       TerminateProcess(GetCurrentProcess(), 0);\r
+       _exit(0);\r
+}\r
+\r
+// プロセス ID の取得\r
+UINT MsGetProcessId()\r
+{\r
+       return GetCurrentProcessId();\r
+}\r
+\r
+// MS 構造体の取得\r
+MS *MsGetMs()\r
+{\r
+       return ms;\r
+}\r
+\r
+// スレッドの優先順位を最低にする\r
+void MsSetThreadPriorityIdle()\r
+{\r
+       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_IDLE);\r
+}\r
+\r
+// スレッドの優先順位を上げる\r
+void MsSetThreadPriorityHigh()\r
+{\r
+       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_ABOVE_NORMAL);\r
+}\r
+\r
+// スレッドの優先順位を下げる\r
+void MsSetThreadPriorityLow()\r
+{\r
+       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_BELOW_NORMAL);\r
+}\r
+\r
+// スレッドの優先順位を最高にする\r
+void MsSetThreadPriorityRealtime()\r
+{\r
+       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_ABOVE_NORMAL);\r
+       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_HIGHEST);\r
+       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_TIME_CRITICAL);\r
+}\r
+\r
+// スレッドの優先順位を戻す\r
+void MsRestoreThreadPriority()\r
+{\r
+       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_NORMAL);\r
+}\r
+\r
+// TCP 設定アプリケーションを表示するべきかどうかチェックする\r
+bool MsIsShouldShowTcpConfigApp()\r
+{\r
+       MS_TCP tcp1, tcp2;\r
+       if (MsIsTcpConfigSupported() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       MsGetTcpConfig(&tcp1);\r
+       if (MsLoadTcpConfigReg(&tcp2) == false)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       if (Cmp(&tcp1, &tcp2, sizeof(MS_TCP) != 0))\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// レジストリの一時設定内容データを Windows の TCP パラメータに適用する\r
+void MsApplyTcpConfig()\r
+{\r
+       if (MsIsTcpConfigSupported())\r
+       {\r
+               MS_TCP tcp;\r
+\r
+               if (MsLoadTcpConfigReg(&tcp))\r
+               {\r
+                       MsSetTcpConfig(&tcp);\r
+               }\r
+       }\r
+}\r
+\r
+// 現在の状態で TCP の動的構成がサポートされているかどうかチェックする\r
+bool MsIsTcpConfigSupported()\r
+{\r
+       if (MsIsNt() && MsIsAdmin())\r
+       {\r
+               UINT type = GetOsInfo()->OsType;\r
+\r
+               if (GET_KETA(type, 100) >= 2)\r
+               {\r
+                       return true;\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// TCP 設定をレジストリ設定から読み込む\r
+bool MsLoadTcpConfigReg(MS_TCP *tcp)\r
+{\r
+       // 引数チェック\r
+       if (tcp == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt())\r
+       {\r
+               Zero(tcp, sizeof(MS_TCP));\r
+\r
+               if (MsRegIsValueEx(REG_LOCAL_MACHINE, MS_REG_TCP_SETTING_KEY, "RecvWindowSize", true) == false ||\r
+                       MsRegIsValueEx(REG_LOCAL_MACHINE, MS_REG_TCP_SETTING_KEY, "SendWindowSize", true) == false)\r
+               {\r
+                       return false;\r
+               }\r
+\r
+               tcp->RecvWindowSize = MsRegReadIntEx(REG_LOCAL_MACHINE, MS_REG_TCP_SETTING_KEY, "RecvWindowSize", true);\r
+               tcp->SendWindowSize = MsRegReadIntEx(REG_LOCAL_MACHINE, MS_REG_TCP_SETTING_KEY, "SendWindowSize", true);\r
+\r
+               return true;\r
+       }\r
+       else\r
+       {\r
+               return false;\r
+       }\r
+}\r
+\r
+// TCP 設定をレジストリから削除する\r
+void MsDeleteTcpConfigReg()\r
+{\r
+       if (MsIsNt() && MsIsAdmin())\r
+       {\r
+               MsRegDeleteKeyEx(REG_LOCAL_MACHINE, MS_REG_TCP_SETTING_KEY, true);\r
+       }\r
+}\r
+\r
+// TCP 設定をレジストリ設定に書き込む\r
+void MsSaveTcpConfigReg(MS_TCP *tcp)\r
+{\r
+       // 引数チェック\r
+       if (tcp == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (MsIsNt() && MsIsAdmin())\r
+       {\r
+               MsRegWriteIntEx(REG_LOCAL_MACHINE, MS_REG_TCP_SETTING_KEY, "RecvWindowSize", tcp->RecvWindowSize, true);\r
+               MsRegWriteIntEx(REG_LOCAL_MACHINE, MS_REG_TCP_SETTING_KEY, "SendWindowSize", tcp->SendWindowSize, true);\r
+       }\r
+}\r
+\r
+// 現在の TCP 設定を取得する\r
+void MsGetTcpConfig(MS_TCP *tcp)\r
+{\r
+       // 引数チェック\r
+       if (tcp == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Zero(tcp, sizeof(MS_TCP));\r
+\r
+       if (MsIsNt())\r
+       {\r
+               // ネットワーク設定初期化\r
+               MsInitGlobalNetworkConfig();\r
+\r
+               // GlobalMaxTcpWindowSize または TcpWindowSize の値が存在すれば読み込む\r
+               tcp->RecvWindowSize = MAX(tcp->RecvWindowSize, MsRegReadInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters", "TcpWindowSize"));\r
+               tcp->RecvWindowSize = MAX(tcp->RecvWindowSize, MsRegReadInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters", "GlobalMaxTcpWindowSize"));\r
+               tcp->RecvWindowSize = MAX(tcp->RecvWindowSize, MsRegReadInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters", "DefaultReceiveWindow"));\r
+\r
+               // DefaultSendWindow の値が存在すれば読み込む\r
+               tcp->SendWindowSize = MsRegReadInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters", "DefaultSendWindow");\r
+       }\r
+}\r
+\r
+// TCP 設定を書き込む\r
+void MsSetTcpConfig(MS_TCP *tcp)\r
+{\r
+       // 引数チェック\r
+       if (tcp == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (MsIsNt() && MsIsAdmin())\r
+       {\r
+               bool window_scaling = false;\r
+               UINT tcp1323opts;\r
+\r
+               if (tcp->RecvWindowSize >= 65536 || tcp->SendWindowSize >= 65536)\r
+               {\r
+                       window_scaling = true;\r
+               }\r
+\r
+               // Tcp1323Opts の設定\r
+               tcp1323opts = MsRegReadInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters", "Tcp1323Opts");\r
+               if (window_scaling)\r
+               {\r
+                       if (tcp1323opts == 0)\r
+                       {\r
+                               tcp1323opts = 1;\r
+                       }\r
+                       if (tcp1323opts == 2)\r
+                       {\r
+                               tcp1323opts = 3;\r
+                       }\r
+               }\r
+               else\r
+               {\r
+                       if (tcp1323opts == 1)\r
+                       {\r
+                               tcp1323opts = 0;\r
+                       }\r
+                       if (tcp1323opts == 3)\r
+                       {\r
+                               tcp1323opts = 2;\r
+                       }\r
+               }\r
+               MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters", "Tcp1323Opts", tcp1323opts);\r
+\r
+               // 受信ウインドウの設定\r
+               if (tcp->RecvWindowSize == 0)\r
+               {\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters",\r
+                               "DefaultReceiveWindow");\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "TcpWindowSize");\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "GlobalMaxTcpWindowSize");\r
+               }\r
+               else\r
+               {\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters",\r
+                               "DefaultReceiveWindow", tcp->RecvWindowSize);\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "TcpWindowSize", tcp->RecvWindowSize);\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "GlobalMaxTcpWindowSize", tcp->RecvWindowSize);\r
+               }\r
+\r
+               // 送信ウインドウの設定\r
+               if (tcp->SendWindowSize == 0)\r
+               {\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters",\r
+                               "DefaultSendWindow");\r
+               }\r
+               else\r
+               {\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters",\r
+                               "DefaultSendWindow", tcp->SendWindowSize);\r
+               }\r
+       }\r
+}\r
+\r
+// グローバルなネットワーク設定を初期化する\r
+void MsInitGlobalNetworkConfig()\r
+{\r
+       if (MsIsNt())\r
+       {\r
+               UINT current_window_size;\r
+\r
+               if (MsRegReadInt(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                       "packetix_no_optimize") == 0)\r
+\r
+               {\r
+                       // TCP コネクション数を最大にする\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "TcpNumConnections", TCP_MAX_NUM_CONNECTIONS);\r
+\r
+                       // タスク オフロードを無効化する\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "DisableTaskOffload", 1);\r
+               }\r
+\r
+               current_window_size = MsRegReadInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters", "TcpWindowSize");\r
+\r
+               if (current_window_size == 65535 || current_window_size == 5980160 ||\r
+                       current_window_size == 16777216 || current_window_size == 16777214)\r
+               {\r
+                       // 古いバージョンの VPN が書き込んでしまった変な値を削除する\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters",\r
+                               "DefaultReceiveWindow");\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters",\r
+                               "DefaultSendWindow");\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "Tcp1323Opts");\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "TcpWindowSize");\r
+                       MsRegDeleteValue(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "GlobalMaxTcpWindowSize");\r
+\r
+                       // vpn_no_change = true にする\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters", "vpn_no_change", 1);\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\AFD\\Parameters", "vpn_no_change", 1);\r
+               }\r
+       }\r
+       else\r
+       {\r
+               if (MsRegReadInt(REG_LOCAL_MACHINE,\r
+                       "System\\CurrentControlSet\\Services\\VxD\\MSTCP",\r
+                       "packetix_no_optimize") == 0)\r
+               {\r
+                       // DeadGWDetect を無効にする\r
+                       MsRegWriteStr(REG_LOCAL_MACHINE, "System\\CurrentControlSet\\Services\\VxD\\MSTCP",\r
+                               "DeadGWDetect", "0");\r
+               }\r
+       }\r
+\r
+       MsApplyTcpConfig();\r
+}\r
+\r
+// 仮想 LAN カードをアップグレードする\r
+bool MsUpgradeVLan(char *tag_name, char *connection_tag_name, char *instance_name)\r
+{\r
+       wchar_t infpath[MAX_PATH];\r
+       char hwid[MAX_PATH];\r
+       wchar_t hwid_w[MAX_PATH];\r
+       bool ret = false;\r
+       bool need_reboot;\r
+       bool before_status;\r
+       UCHAR old_mac_address[6];\r
+       char *s;\r
+       NO_WARNING *nw;\r
+       char sen_sys[MAX_PATH];\r
+       // 引数チェック\r
+       if (instance_name == NULL || tag_name == NULL || connection_tag_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               // Windows 9x ではアップグレードできない\r
+               return false;\r
+       }\r
+\r
+       Zero(hwid, sizeof(hwid));\r
+       Format(hwid, sizeof(hwid), DRIVER_DEVICE_ID_TAG, instance_name);\r
+       StrToUni(hwid_w, sizeof(hwid_w), hwid);\r
+\r
+       // 指定された名前の仮想 LAN カードがすでに登録されているかどうかを調べる\r
+       if (MsIsVLanExists(tag_name, instance_name) == false)\r
+       {\r
+               // 登録されていない\r
+               return false;\r
+       }\r
+\r
+       // 現在使用している .sys ファイル名を取得する\r
+       if (MsGetSenDeiverFilename(sen_sys, sizeof(sen_sys), instance_name) == false)\r
+       {\r
+               // 不明なので新しいファイル名を作成する\r
+               if (MsMakeNewSenDriverFilename(sen_sys, sizeof(sen_sys)) == false)\r
+               {\r
+                       // 失敗\r
+                       return false;\r
+               }\r
+       }\r
+\r
+       // 現在の動作状況を取得する\r
+       before_status = MsIsVLanEnabled(instance_name);\r
+\r
+       // 以前の MAC アドレスを取得する\r
+       s = MsGetMacAddress(tag_name, instance_name);\r
+       if (s == NULL)\r
+       {\r
+               Zero(old_mac_address, 6);\r
+       }\r
+       else\r
+       {\r
+               BUF *b;\r
+               b = StrToBin(s);\r
+               Free(s);\r
+\r
+               if (b->Size == 6)\r
+               {\r
+                       Copy(old_mac_address, b->Buf, b->Size);\r
+               }\r
+               else\r
+               {\r
+                       Zero(old_mac_address, 6);\r
+               }\r
+\r
+               FreeBuf(b);\r
+       }\r
+\r
+       // インストール開始\r
+       if (MsStartDriverInstall(instance_name, IsZero(old_mac_address, 6) ? NULL : old_mac_address, sen_sys) == false)\r
+       {\r
+               return false;\r
+       }\r
+       MsGetDriverPath(instance_name, NULL, NULL, infpath, NULL, sen_sys);\r
+\r
+       nw = NULL;\r
+\r
+       //if (MsIsVista() == false)\r
+       {\r
+               nw = MsInitNoWarning();\r
+       }\r
+\r
+       // インストールを行う\r
+       if (ms->nt->UpdateDriverForPlugAndPlayDevicesW(\r
+               NULL, hwid_w, infpath, 1, &need_reboot))\r
+       {\r
+               ret = true;\r
+       }\r
+       MsFreeNoWarning(nw);\r
+\r
+       // インストール完了\r
+       MsFinishDriverInstall(instance_name, sen_sys);\r
+\r
+       MsInitNetworkConfig(tag_name, instance_name, connection_tag_name);\r
+\r
+       // 動作を復元する\r
+       if (before_status)\r
+       {\r
+               MsEnableVLan(instance_name);\r
+       }\r
+       else\r
+       {\r
+               MsDisableVLan(instance_name);\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// Windows 9x 用テスト\r
+void MsWin9xTest()\r
+{\r
+}\r
+\r
+// 仮想 LAN カードの CompatibleIDs を更新する\r
+void MsUpdateCompatibleIDs(char *instance_name)\r
+{\r
+       TOKEN_LIST *t;\r
+       char id[MAX_SIZE];\r
+       char device_title[MAX_SIZE];\r
+       char device_title_old[MAX_SIZE];\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Format(id, sizeof(id), DRIVER_DEVICE_ID_TAG, instance_name);\r
+       Format(device_title, sizeof(device_title), VLAN_ADAPTER_NAME_TAG, instance_name);\r
+       Format(device_title_old, sizeof(device_title_old), "---dummy-string-ut--", instance_name);\r
+\r
+       t = MsRegEnumKey(REG_LOCAL_MACHINE, "Enum\\Root\\Net");\r
+       if (t != NULL)\r
+       {\r
+               UINT i;\r
+               for (i = 0;i < t->NumTokens;i++)\r
+               {\r
+                       char keyname[MAX_PATH];\r
+                       char *str;\r
+                       char *title;\r
+\r
+                       Format(keyname, sizeof(keyname), "Enum\\Root\\Net\\%s", t->Token[i]);\r
+\r
+                       title = MsRegReadStr(REG_LOCAL_MACHINE, keyname, "DeviceDesc");\r
+\r
+                       if (title != NULL)\r
+                       {\r
+                               if (StrCmpi(title, device_title) == 0 || StrCmpi(title, device_title_old) == 0)\r
+                               {\r
+                                       Format(keyname, sizeof(keyname), "Enum\\Root\\Net\\%s",t->Token[i]);\r
+                                       str = MsRegReadStr(REG_LOCAL_MACHINE, keyname, "CompatibleIDs");\r
+                                       if (str != NULL)\r
+                                       {\r
+                                               Free(str);\r
+                                       }\r
+                                       else\r
+                                       {\r
+                                               MsRegWriteStr(REG_LOCAL_MACHINE, keyname, "CompatibleIDs", id);\r
+                                       }\r
+                               }\r
+                               Free(title);\r
+                       }\r
+               }\r
+\r
+               FreeToken(t);\r
+       }\r
+\r
+       MsRegWriteStr(REG_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Setup", "SourcePath",\r
+               ms->System32Dir);\r
+}\r
+\r
+// 仮想 LAN カードをインストールする (Win9x 用)\r
+bool MsInstallVLan9x(char *instance_name)\r
+{\r
+       char sysdir[MAX_PATH];\r
+       char infdir[MAX_PATH];\r
+       char otherdir[MAX_PATH];\r
+       char syspath[MAX_PATH];\r
+       char syspath2[MAX_PATH];\r
+       char infpath[MAX_PATH];\r
+       char vpn16[MAX_PATH];\r
+       char infpath_src[MAX_PATH];\r
+       char syspath_src[MAX_PATH];\r
+       char sen_sys[MAX_PATH];\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       StrCpy(sysdir, sizeof(sysdir), MsGetSystem32Dir());\r
+       Format(infdir, sizeof(infdir), "%s\\inf", MsGetWindowsDir());\r
+       Format(otherdir, sizeof(otherdir), "%s\\other", infdir);\r
+       Format(syspath, sizeof(syspath), "%s\\Sen_%s.sys", sysdir, instance_name);\r
+       Format(syspath2, sizeof(syspath2), "%s\\Sen_%s.sys", infdir, instance_name);\r
+       Format(infpath, sizeof(infpath), "%s\\Sen_%s.inf", infdir, instance_name);\r
+       Format(vpn16, sizeof(vpn16), "%s\\vpn16.exe", MsGetMyTempDir());\r
+\r
+       MakeDir(otherdir);\r
+\r
+       Format(sen_sys, sizeof(sen_sys), DRIVER_INSTALL_SYS_NAME_TAG, instance_name);\r
+\r
+       // vpn16.exe のコピー\r
+       FileCopy("|vpn16.exe", vpn16);\r
+\r
+       // インストール開始\r
+       if (MsStartDriverInstall(instance_name, NULL, sen_sys) == false)\r
+       {\r
+               return false;\r
+       }\r
+       MsGetDriverPathA(instance_name, NULL, NULL, infpath_src, syspath_src, sen_sys);\r
+\r
+       // inf ファイルのコピー\r
+       FileCopy(infpath_src, infpath);\r
+\r
+       // sys ファイルのコピー\r
+       FileCopy(syspath_src, syspath);\r
+\r
+       // デバイスドライバのインストール\r
+       if (Run(vpn16, instance_name, false, true) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // CompatibleIDs の更新\r
+       MsUpdateCompatibleIDs(instance_name);\r
+\r
+       return true;\r
+}\r
+\r
+// 子ウインドウ列挙プロシージャ\r
+bool CALLBACK MsEnumChildWindowProc(HWND hWnd, LPARAM lParam)\r
+{\r
+       LIST *o = (LIST *)lParam;\r
+\r
+       if (o != NULL)\r
+       {\r
+               MsEnumChildWindows(o, hWnd);\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 指定したウインドウとその子ウインドウをすべて列挙する\r
+LIST *MsEnumChildWindows(LIST *o, HWND hWnd)\r
+{\r
+       // 引数チェック\r
+       if (hWnd == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (o == NULL)\r
+       {\r
+               o = NewListFast(NULL);\r
+       }\r
+\r
+       MsAddWindowToList(o, hWnd);\r
+\r
+       EnumChildWindows(hWnd, MsEnumChildWindowProc, (LPARAM)o);\r
+\r
+       return o;\r
+}\r
+\r
+// ウインドウをリストに追加する\r
+void MsAddWindowToList(LIST *o, HWND hWnd)\r
+{\r
+       // 引数チェック\r
+       if (o == NULL || hWnd == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (IsInList(o, hWnd) == false)\r
+       {\r
+               Add(o, hWnd);\r
+       }\r
+}\r
+\r
+// スレッドの所有するウインドウの列挙\r
+bool CALLBACK MsEnumThreadWindowProc(HWND hWnd, LPARAM lParam)\r
+{\r
+       LIST *o = (LIST *)lParam;\r
+\r
+       if (o == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       MsEnumChildWindows(o, hWnd);\r
+\r
+       return true;\r
+}\r
+\r
+// ウインドウ列挙プロシージャ\r
+BOOL CALLBACK EnumTopWindowProc(HWND hWnd, LPARAM lParam)\r
+{\r
+       LIST *o = (LIST *)lParam;\r
+       HWND hParent;\r
+       char c1[MAX_SIZE], c2[MAX_SIZE];\r
+       // 引数チェック\r
+       if (hWnd == NULL || o == NULL)\r
+       {\r
+               return TRUE;\r
+       }\r
+\r
+       Zero(c1, sizeof(c1));\r
+       Zero(c2, sizeof(c2));\r
+\r
+       hParent = GetParent(hWnd);\r
+\r
+       GetClassName(hWnd, c1, sizeof(c1));\r
+\r
+       if (hParent != NULL)\r
+       {\r
+               GetClassName(hParent, c2, sizeof(c2));\r
+       }\r
+\r
+       if (StrCmpi(c1, "SysIPAddress32") != 0 && (IsEmptyStr(c2) || StrCmpi(c2, "SysIPAddress32") != 0))\r
+       {\r
+               AddWindow(o, hWnd);\r
+       }\r
+\r
+       return TRUE;\r
+}\r
+\r
+// 子ウインドウ列挙プロシージャ\r
+BOOL CALLBACK EnumChildWindowProc(HWND hWnd, LPARAM lParam)\r
+{\r
+       ENUM_CHILD_WINDOW_PARAM *p = (ENUM_CHILD_WINDOW_PARAM *)lParam;\r
+       LIST *o;\r
+       HWND hParent;\r
+       char c1[MAX_SIZE], c2[MAX_SIZE];\r
+       // 引数チェック\r
+       if (hWnd == NULL || p == NULL)\r
+       {\r
+               return TRUE;\r
+       }\r
+\r
+       o = p->o;\r
+\r
+       Zero(c1, sizeof(c1));\r
+       Zero(c2, sizeof(c2));\r
+\r
+       hParent = GetParent(hWnd);\r
+\r
+       GetClassName(hWnd, c1, sizeof(c1));\r
+\r
+       if (hParent != NULL)\r
+       {\r
+               GetClassName(hParent, c2, sizeof(c2));\r
+       }\r
+\r
+       if (p->include_ipcontrol || (StrCmpi(c1, "SysIPAddress32") != 0 && (IsEmptyStr(c2) || StrCmpi(c2, "SysIPAddress32") != 0)))\r
+       {\r
+               AddWindow(o, hWnd);\r
+\r
+               if (p->no_recursion == false)\r
+               {\r
+                       EnumChildWindows(hWnd, EnumChildWindowProc, (LPARAM)p);\r
+               }\r
+       }\r
+\r
+       return TRUE;\r
+}\r
+LIST *EnumAllWindow()\r
+{\r
+       return EnumAllWindowEx(false, false);\r
+}\r
+LIST *EnumAllWindowEx(bool no_recursion, bool include_ipcontrol)\r
+{\r
+       ENUM_CHILD_WINDOW_PARAM p;\r
+       LIST *o = NewWindowList();\r
+\r
+       Zero(&p, sizeof(p));\r
+       p.o = o;\r
+       p.no_recursion = no_recursion;\r
+       p.include_ipcontrol = include_ipcontrol;\r
+\r
+       EnumWindows(EnumChildWindowProc, (LPARAM)&p);\r
+\r
+       return o;\r
+}\r
+LIST *EnumAllTopWindow()\r
+{\r
+       LIST *o = NewWindowList();\r
+\r
+       EnumWindows(EnumTopWindowProc, (LPARAM)o);\r
+\r
+       return o;\r
+}\r
+\r
+// 特定のウインドウの中にあるすべての子ウインドウを列挙する\r
+LIST *EnumAllChildWindow(HWND hWnd)\r
+{\r
+       return EnumAllChildWindowEx(hWnd, false, false, false);\r
+}\r
+LIST *EnumAllChildWindowEx(HWND hWnd, bool no_recursion, bool include_ipcontrol, bool no_self)\r
+{\r
+       ENUM_CHILD_WINDOW_PARAM p;\r
+       LIST *o = NewWindowList();\r
+\r
+       Zero(&p, sizeof(p));\r
+       p.include_ipcontrol = include_ipcontrol;\r
+       p.no_recursion = no_recursion;\r
+       p.o = o;\r
+\r
+       if (no_self == false)\r
+       {\r
+               AddWindow(o, hWnd);\r
+       }\r
+\r
+       EnumChildWindows(hWnd, EnumChildWindowProc, (LPARAM)&p);\r
+\r
+       return o;\r
+}\r
+\r
+// ウインドウリストの解放\r
+void FreeWindowList(LIST *o)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               HWND *e = LIST_DATA(o, i);\r
+\r
+               Free(e);\r
+       }\r
+\r
+       ReleaseList(o);\r
+}\r
+\r
+// ウインドウリストにウインドウを追加\r
+void AddWindow(LIST *o, HWND hWnd)\r
+{\r
+       HWND t, *e;\r
+       // 引数チェック\r
+       if (o == NULL || hWnd == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       t = hWnd;\r
+\r
+       if (Search(o, &t) != NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       e = ZeroMalloc(sizeof(HWND));\r
+       *e = hWnd;\r
+\r
+       Insert(o, e);\r
+}\r
+\r
+// ウインドウリストの比較\r
+int CmpWindowList(void *p1, void *p2)\r
+{\r
+       HWND *h1, *h2;\r
+       if (p1 == NULL || p2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       h1 = *(HWND **)p1;\r
+       h2 = *(HWND **)p2;\r
+       if (h1 == NULL || h2 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       return Cmp(h1, h2, sizeof(HWND));\r
+}\r
+\r
+// 新しいウインドウリストの作成\r
+LIST *NewWindowList()\r
+{\r
+       return NewListFast(CmpWindowList);\r
+}\r
+\r
+// Windows Vista かどうか判別\r
+bool MsIsVista()\r
+{\r
+       OS_INFO *info = GetOsInfo();\r
+\r
+       if (info == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (OS_IS_WINDOWS_NT(info->OsType))\r
+       {\r
+               if (GET_KETA(info->OsType, 100) >= 5)\r
+               {\r
+                       return true;\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// ウインドウの所有者のプロセスパスを取得する\r
+bool MsGetWindowOwnerProcessExeName(char *path, UINT size, HWND hWnd)\r
+{\r
+       DWORD procId = 0;\r
+       // 引数チェック\r
+       if (path == NULL || hWnd == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       GetWindowThreadProcessId(hWnd, &procId);\r
+       if (procId == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsGetProcessExeName(path, size, procId) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+bool MsGetWindowOwnerProcessExeNameW(wchar_t *path, UINT size, HWND hWnd)\r
+{\r
+       DWORD procId = 0;\r
+       // 引数チェック\r
+       if (path == NULL || hWnd == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       GetWindowThreadProcessId(hWnd, &procId);\r
+       if (procId == 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsGetProcessExeNameW(path, size, procId) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// プロセス ID からプロセスパスを取得する\r
+bool MsGetProcessExeName(char *path, UINT size, UINT id)\r
+{\r
+       LIST *o;\r
+       MS_PROCESS *proc;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (path == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       o = MsGetProcessList();\r
+       proc = MsSearchProcessById(o, id);\r
+\r
+       if (proc != NULL)\r
+       {\r
+               ret = true;\r
+               StrCpy(path, size, proc->ExeFilename);\r
+       }\r
+\r
+       MsFreeProcessList(o);\r
+\r
+       return ret;\r
+}\r
+bool MsGetProcessExeNameW(wchar_t *path, UINT size, UINT id)\r
+{\r
+       LIST *o;\r
+       MS_PROCESS *proc;\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (path == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       o = MsGetProcessList();\r
+       proc = MsSearchProcessById(o, id);\r
+\r
+       if (proc != NULL)\r
+       {\r
+               ret = true;\r
+               UniStrCpy(path, size, proc->ExeFilenameW);\r
+       }\r
+\r
+       MsFreeProcessList(o);\r
+\r
+       return ret;\r
+}\r
+\r
+// 警告ダイアログを閉じる\r
+bool MsCloseWarningWindow(UINT thread_id)\r
+{\r
+       UINT i;\r
+       LIST *o;\r
+       bool ret = false;\r
+\r
+       if (MsIsVista() == false)\r
+       {\r
+               o = NewListFast(NULL);\r
+               EnumThreadWindows(thread_id, MsEnumThreadWindowProc, (LPARAM)o);\r
+       }\r
+       else\r
+       {\r
+               o = EnumAllTopWindow();\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               HWND hWnd;\r
+               \r
+               if (MsIsVista() == false)\r
+               {\r
+                       hWnd = LIST_DATA(o, i);\r
+               }\r
+               else\r
+               {\r
+                       hWnd = *((HWND *)LIST_DATA(o, i));\r
+               }\r
+\r
+               if (hWnd != NULL)\r
+               {\r
+                       OS_INFO *info = GetOsInfo();\r
+\r
+                       if (MsIsNt())\r
+                       {\r
+                               // このウインドウがドライバの警告画面かどうかを取得する\r
+                               if (MsIsVista() == false)\r
+                               {\r
+                                       // Windows Vista 以外\r
+                                       HWND hStatic, hOk, hCancel, hDetail;\r
+\r
+                                       hStatic = GetDlgItem(hWnd, 0x14C1);\r
+                                       hOk = GetDlgItem(hWnd, 0x14B7);\r
+                                       hCancel = GetDlgItem(hWnd, 0x14BA);\r
+                                       hDetail = GetDlgItem(hWnd, 0x14B9);\r
+\r
+                                       if ((hStatic != NULL || hDetail != NULL) && hOk != NULL && hCancel != NULL)\r
+                                       {\r
+                                               char tmp[MAX_SIZE];\r
+                                               bool b = false;\r
+\r
+                                               if (GetClassName(hStatic, tmp, sizeof(tmp)) != 0)\r
+                                               {\r
+                                                       if (StrCmpi(tmp, "static") == 0)\r
+                                                       {\r
+                                                               b = true;\r
+                                                       }\r
+                                               }\r
+\r
+                                               if (GetClassName(hDetail, tmp, sizeof(tmp)) != 0)\r
+                                               {\r
+                                                       if (StrCmpi(tmp, "button") == 0)\r
+                                                       {\r
+                                                               b = true;\r
+                                                       }\r
+                                               }\r
+\r
+                                               if (b)\r
+                                               {\r
+                                                       if (GetClassName(hOk, tmp, sizeof(tmp)) != 0)\r
+                                                       {\r
+                                                               if (StrCmpi(tmp, "button") == 0)\r
+                                                               {\r
+                                                                       if (GetClassName(hCancel, tmp, sizeof(tmp)) != 0)\r
+                                                                       {\r
+                                                                               if (StrCmpi(tmp, "button") == 0)\r
+                                                                               {\r
+                                                                                       // 発見したので OK ボタンを押す\r
+                                                                                       PostMessage(hWnd, WM_COMMAND, 0x14B7, 0);\r
+\r
+                                                                                       ret = true;\r
+                                                                               }\r
+                                                                       }\r
+                                                               }\r
+                                                       }\r
+                                               }\r
+                                       }\r
+                               }\r
+                               else\r
+                               {\r
+                                       // Windows Vista\r
+                                       char exe[MAX_PATH];\r
+\r
+                                       if (MsGetWindowOwnerProcessExeName(exe, sizeof(exe), hWnd))\r
+                                       {\r
+                                               if (EndWith(exe, "rundll32.exe"))\r
+                                               {\r
+                                                       LIST *o;\r
+                                                       HWND h;\r
+                                                       UINT i;\r
+\r
+                                                       o = EnumAllChildWindow(hWnd);\r
+\r
+                                                       if (o != NULL)\r
+                                                       {\r
+                                                               for (i = 0;i < LIST_NUM(o);i++)\r
+                                                               {\r
+                                                                       char tmp[MAX_SIZE];\r
+\r
+                                                                       h = *((HWND *)LIST_DATA(o, i));\r
+\r
+                                                                       Zero(tmp, sizeof(tmp));\r
+                                                                       GetClassNameA(h, tmp, sizeof(tmp));\r
+\r
+                                                                       if (StrCmpi(tmp, "DirectUIHWND") == 0)\r
+                                                                       {\r
+                                                                               LIST *o = EnumAllChildWindow(h);\r
+\r
+                                                                               if (o != NULL)\r
+                                                                               {\r
+                                                                                       UINT j;\r
+                                                                                       UINT numDirectUIHWND = 0;\r
+                                                                                       UINT numButton = 0;\r
+                                                                                       HWND hButton1 = NULL;\r
+                                                                                       HWND hButton2 = NULL;\r
+\r
+                                                                                       for (j = 0;j < LIST_NUM(o);j++)\r
+                                                                                       {\r
+                                                                                               HWND hh;\r
+                                                                                               char tmp[MAX_SIZE];\r
+\r
+                                                                                               hh = *((HWND *)LIST_DATA(o, j));\r
+\r
+                                                                                               Zero(tmp, sizeof(tmp));\r
+                                                                                               GetClassNameA(hh, tmp, sizeof(tmp));\r
+\r
+                                                                                               if (StrCmpi(tmp, "DirectUIHWND") == 0)\r
+                                                                                               {\r
+                                                                                                       numDirectUIHWND++;\r
+                                                                                               }\r
+\r
+                                                                                               if (StrCmpi(tmp, "button") == 0)\r
+                                                                                               {\r
+                                                                                                       numButton++;\r
+                                                                                                       if (hButton1 == NULL)\r
+                                                                                                       {\r
+                                                                                                               hButton1 = hh;\r
+                                                                                                       }\r
+                                                                                                       else\r
+                                                                                                       {\r
+                                                                                                               hButton2 = hh;\r
+                                                                                                       }\r
+                                                                                               }\r
+                                                                                       }\r
+\r
+                                                                                       if (numDirectUIHWND == 1 && numButton == 2)\r
+                                                                                       {\r
+                                                                                               if (hButton1 != NULL && hButton2 != NULL)\r
+                                                                                               {\r
+                                                                                                       HWND hButton;\r
+                                                                                                       HWND hParent;\r
+                                                                                                       RECT r1, r2;\r
+\r
+                                                                                                       GetWindowRect(hButton1, &r1);\r
+                                                                                                       GetWindowRect(hButton2, &r2);\r
+\r
+                                                                                                       hButton = hButton1;\r
+\r
+                                                                                                       if (r1.top < r2.top)\r
+                                                                                                       {\r
+                                                                                                               hButton = hButton2;\r
+                                                                                                       }\r
+\r
+                                                                                                       hParent = GetParent(hButton);\r
+\r
+                                                                                                       // 発見したので OK ボタンを押す\r
+                                                                                                       PostMessage(hParent, WM_COMMAND, 1, 0);\r
+\r
+                                                                                                       ret = true;\r
+                                                                                               }\r
+                                                                                       }\r
+\r
+                                                                                       FreeWindowList(o);\r
+                                                                               }\r
+                                                                       }\r
+                                                               }\r
+\r
+                                                               FreeWindowList(o);\r
+                                                       }\r
+                                               }\r
+                                       }\r
+                               }\r
+                       }\r
+               }\r
+       }\r
+\r
+       if (MsIsVista() == false)\r
+       {\r
+               ReleaseList(o);\r
+       }\r
+       else\r
+       {\r
+               FreeWindowList(o);\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// 警告を出さないようにするためのスレッド\r
+void MsNoWarningThreadProc(THREAD *thread, void *param)\r
+{\r
+       NO_WARNING *nw;\r
+       UINT interval;\r
+       UINT i;\r
+       bool found0 = false;\r
+       // 引数チェック\r
+       if (thread == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       nw = (NO_WARNING *)param;\r
+\r
+       nw->NoWarningThread = thread;\r
+       AddRef(thread->ref);\r
+\r
+       NoticeThreadInit(thread);\r
+\r
+       interval = 50;\r
+\r
+       if (MsIsVista())\r
+       {\r
+               interval = 1000;\r
+       }\r
+\r
+       i = 0;\r
+\r
+       while (nw->Halt == false)\r
+       {\r
+               bool found;\r
+\r
+               // 警告ダイアログを閉じる\r
+               found = MsCloseWarningWindow(nw->ThreadId);\r
+               if (i == 0)\r
+               {\r
+                       found0 = found;\r
+               }\r
+               else\r
+               {\r
+                       if (found0 == false && found)\r
+                       {\r
+                               break;\r
+                       }\r
+               }\r
+               i++;\r
+\r
+               // 親スレッドが指示するまでループする\r
+               Wait(nw->HaltEvent, interval);\r
+       }\r
+}\r
+\r
+// 警告音を消す処理の初期化\r
+char *MsNoWarningSoundInit()\r
+{\r
+       char *ret = MsRegReadStr(REG_CURRENT_USER, "AppEvents\\Schemes\\Apps\\.Default\\SystemAsterisk\\.Current", "");\r
+\r
+       if (IsEmptyStr(ret))\r
+       {\r
+               Free(ret);\r
+               ret = NULL;\r
+       }\r
+       else\r
+       {\r
+               MsRegWriteStr(REG_CURRENT_USER,\r
+                       "AppEvents\\Schemes\\Apps\\.Default\\SystemAsterisk\\.Current",\r
+                       "", "");\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// 警告音を消す処理の解放\r
+void MsNoWarningSoundFree(char *s)\r
+{\r
+       // 引数チェック\r
+       if (s == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       MsRegWriteStrExpand(REG_CURRENT_USER,\r
+               "AppEvents\\Schemes\\Apps\\.Default\\SystemAsterisk\\.Current",\r
+               "", s);\r
+\r
+       Free(s);\r
+}\r
+\r
+// 警告を出さないようにする処理の開始\r
+NO_WARNING *MsInitNoWarning()\r
+{\r
+       wchar_t *tmp;\r
+       THREAD *thread;\r
+       NO_WARNING *nw = ZeroMalloc(sizeof(NO_WARNING));\r
+\r
+       // 現在のサウンドファイル名を取得する\r
+       tmp = MsRegReadStrW(REG_CURRENT_USER, "AppEvents\\Schemes\\Apps\\.Default\\SystemAsterisk\\.Current", "");\r
+       if (UniIsEmptyStr(tmp) == false)\r
+       {\r
+               nw->SoundFileName = CopyUniStr(tmp);\r
+\r
+               MsRegWriteStrW(REG_CURRENT_USER,\r
+                       "AppEvents\\Schemes\\Apps\\.Default\\SystemAsterisk\\.Current",\r
+                       "", L"");\r
+       }\r
+\r
+       Free(tmp);\r
+\r
+       nw->ThreadId = GetCurrentThreadId();\r
+       nw->HaltEvent = NewEvent();\r
+\r
+       thread = NewThread(MsNoWarningThreadProc, nw);\r
+       WaitThreadInit(thread);\r
+\r
+       ReleaseThread(thread);\r
+\r
+       return nw;\r
+}\r
+\r
+// 警告を出さないようにする処理の終了\r
+void MsFreeNoWarning(NO_WARNING *nw)\r
+{\r
+       // 引数チェック\r
+       if (nw == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       nw->Halt = true;\r
+       Set(nw->HaltEvent);\r
+\r
+       WaitThread(nw->NoWarningThread, INFINITE);\r
+       ReleaseThread(nw->NoWarningThread);\r
+\r
+       ReleaseEvent(nw->HaltEvent);\r
+\r
+       if (nw->SoundFileName != NULL)\r
+       {\r
+               MsRegWriteStrExpandW(REG_CURRENT_USER,\r
+                       "AppEvents\\Schemes\\Apps\\.Default\\SystemAsterisk\\.Current",\r
+                       "", nw->SoundFileName);\r
+\r
+               Free(nw->SoundFileName);\r
+       }\r
+\r
+       Free(nw);\r
+}\r
+\r
+// 仮想 LAN カードをインストールする\r
+bool MsInstallVLan(char *tag_name, char *connection_tag_name, char *instance_name)\r
+{\r
+       wchar_t infpath[MAX_PATH];\r
+       wchar_t inf_class_name[MAX_PATH];\r
+       GUID inf_class_guid;\r
+       HDEVINFO device_info;\r
+       SP_DEVINFO_DATA device_info_data;\r
+       char hwid[MAX_PATH];\r
+       wchar_t hwid_w[MAX_PATH];\r
+       bool ret = false;\r
+       bool need_reboot;\r
+       char sen_sys[MAX_PATH];\r
+       UINT i;\r
+       // 引数チェック\r
+       if (instance_name == NULL || tag_name == NULL || connection_tag_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               // Windows 9x 用\r
+               return MsInstallVLan9x(instance_name);\r
+       }\r
+\r
+       Zero(hwid, sizeof(hwid));\r
+       Format(hwid, sizeof(hwid), DRIVER_DEVICE_ID_TAG, instance_name);\r
+       StrToUni(hwid_w, sizeof(hwid_w), hwid);\r
+\r
+       // 指定された名前の仮想 LAN カードがすでに登録されているかどうかを調べる\r
+       if (MsIsVLanExists(tag_name, instance_name))\r
+       {\r
+               // すでに登録されている\r
+               return false;\r
+       }\r
+\r
+       // インストール先 .sys ファイル名の決定\r
+       if (MsMakeNewSenDriverFilename(sen_sys, sizeof(sen_sys)) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // インストール開始\r
+       if (MsStartDriverInstall(instance_name, NULL, sen_sys) == false)\r
+       {\r
+               return false;\r
+       }\r
+       MsGetDriverPath(instance_name, NULL, NULL, infpath, NULL, sen_sys);\r
+\r
+       // inf ファイルのクラス GUID を取得する\r
+       if (SetupDiGetINFClassW(infpath, &inf_class_guid, inf_class_name, sizeof(inf_class_name), NULL))\r
+       {\r
+               // デバイス情報セットを取得する\r
+               device_info = SetupDiCreateDeviceInfoList(&inf_class_guid, NULL);\r
+               if (device_info != INVALID_HANDLE_VALUE)\r
+               {\r
+                       // Windows 2000 以降\r
+                       Zero(&device_info_data, sizeof(device_info_data));\r
+                       device_info_data.cbSize = sizeof(device_info_data);\r
+                       if (SetupDiCreateDeviceInfoW(device_info, inf_class_name, &inf_class_guid,\r
+                               NULL, NULL, DICD_GENERATE_ID, &device_info_data))\r
+                       {\r
+                               // レジストリ情報を設定する\r
+                               if (SetupDiSetDeviceRegistryProperty(device_info, &device_info_data,\r
+                                       SPDRP_HARDWAREID, (BYTE *)hwid, sizeof(hwid)))\r
+                               {\r
+                                       NO_WARNING *nw = NULL;\r
+                                       \r
+                                       //if (MsIsVista() == false)\r
+                                       {\r
+                                               nw = MsInitNoWarning();\r
+                                       }\r
+\r
+                                       // クラスインストーラを起動する\r
+                                       if (SetupDiCallClassInstaller(DIF_REGISTERDEVICE, device_info,\r
+                                               &device_info_data))\r
+                                       {\r
+                                               // インストールを行う\r
+                                               if (ms->nt->UpdateDriverForPlugAndPlayDevicesW(\r
+                                                       NULL, hwid_w, infpath, 1, &need_reboot))\r
+                                               {\r
+                                                       ret = true;\r
+                                               }\r
+                                               else\r
+                                               {\r
+                                                       // インストール失敗\r
+                                                       SetupDiCallClassInstaller(DIF_REMOVE, device_info,\r
+                                                               &device_info_data);\r
+                                               }\r
+                                       }\r
+                                       else\r
+                                       {\r
+                                               Debug("SetupDiCallClassInstaller Error: %X\n", GetLastError());\r
+                                       }\r
+\r
+                                       MsFreeNoWarning(nw);\r
+                               }\r
+                       }\r
+                       // デバイス情報セットを削除する\r
+                       SetupDiDestroyDeviceInfoList(device_info);\r
+               }\r
+       }\r
+\r
+       // インストール完了\r
+       MsFinishDriverInstall(instance_name, sen_sys);\r
+\r
+       for (i = 0;i < 5;i++)\r
+       {\r
+               MsInitNetworkConfig(tag_name, instance_name, connection_tag_name);\r
+               SleepThread(MsIsVista() ? 1000 : 300);\r
+       }\r
+\r
+       if (ret)\r
+       {\r
+               MsDisableVLan(instance_name);\r
+               MsEnableVLan(instance_name);\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// デバイス ID からデバイス情報を取得する\r
+HDEVINFO MsGetDevInfoFromDeviceId(SP_DEVINFO_DATA *dev_info_data, char *device_id)\r
+{\r
+       HDEVINFO dev_info;\r
+       SP_DEVINFO_LIST_DETAIL_DATA detail_data;\r
+       SP_DEVINFO_DATA data;\r
+       UINT i;\r
+       bool found;\r
+       char target_name[MAX_SIZE];\r
+       // 引数チェック\r
+       if (dev_info_data == NULL || device_id == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       StrCpy(target_name, sizeof(target_name), device_id);\r
+\r
+       // デバイス情報リストを作成\r
+       dev_info = SetupDiGetClassDevsEx(NULL, NULL, NULL, DIGCF_ALLCLASSES | DIGCF_PRESENT, NULL, NULL, NULL);\r
+       if (dev_info == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Zero(&detail_data, sizeof(detail_data));\r
+       detail_data.cbSize = sizeof(detail_data);\r
+       if (SetupDiGetDeviceInfoListDetail(dev_info, &detail_data) == false)\r
+       {\r
+               MsDestroyDevInfo(dev_info);\r
+               return NULL;\r
+       }\r
+\r
+       Zero(&data, sizeof(data));\r
+       data.cbSize = sizeof(data);\r
+\r
+       // 列挙開始\r
+       found = false;\r
+       for (i = 0;SetupDiEnumDeviceInfo(dev_info, i, &data);i++)\r
+       {\r
+               char *buffer;\r
+               UINT buffer_size = 8092;\r
+               DWORD data_type;\r
+\r
+               buffer = ZeroMalloc(buffer_size);\r
+\r
+               if (SetupDiGetDeviceRegistryProperty(dev_info, &data, SPDRP_HARDWAREID, &data_type, (PBYTE)buffer, buffer_size, NULL))\r
+               {\r
+                       if (StrCmpi(buffer, target_name) == 0)\r
+                       {\r
+                               // 発見\r
+                               found = true;\r
+                       }\r
+               }\r
+\r
+               Free(buffer);\r
+\r
+               if (found)\r
+               {\r
+                       break;\r
+               }\r
+       }\r
+\r
+       if (found == false)\r
+       {\r
+               MsDestroyDevInfo(dev_info);\r
+               return NULL;\r
+       }\r
+       else\r
+       {\r
+               Copy(dev_info_data, &data, sizeof(data));\r
+               return dev_info;\r
+       }\r
+}\r
+\r
+// 指定したデバイスが動作中かどうかを調べる\r
+bool MsIsDeviceRunning(HDEVINFO info, SP_DEVINFO_DATA *dev_info_data)\r
+{\r
+       SP_DEVINFO_LIST_DETAIL_DATA detail;\r
+       UINT status = 0, problem = 0;\r
+       // 引数チェック\r
+       if (info == NULL || dev_info_data == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&detail, sizeof(detail));\r
+       detail.cbSize = sizeof(detail);\r
+\r
+       if (SetupDiGetDeviceInfoListDetail(info, &detail) == false ||\r
+               ms->nt->CM_Get_DevNode_Status_Ex(&status, &problem, dev_info_data->DevInst,\r
+               0, detail.RemoteMachineHandle) != CR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (status & 8)\r
+       {\r
+               return true;\r
+       }\r
+       else\r
+       {\r
+               return false;\r
+       }\r
+}\r
+\r
+// 指定したデバイスを開始させる\r
+bool MsStartDevice(HDEVINFO info, SP_DEVINFO_DATA *dev_info_data)\r
+{\r
+       SP_PROPCHANGE_PARAMS p;\r
+       // 引数チェック\r
+       if (info == NULL || dev_info_data == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&p, sizeof(p));\r
+       p.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);\r
+       p.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;\r
+       p.StateChange = DICS_ENABLE;\r
+       p.Scope = DICS_FLAG_GLOBAL;\r
+       if (SetupDiSetClassInstallParams(info, dev_info_data, &p.ClassInstallHeader, sizeof(p)))\r
+       {\r
+               SetupDiCallClassInstaller(DIF_PROPERTYCHANGE, info, dev_info_data);\r
+       }\r
+\r
+       Zero(&p, sizeof(p));\r
+       p.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);\r
+       p.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;\r
+       p.StateChange = DICS_ENABLE;\r
+       p.Scope = DICS_FLAG_CONFIGSPECIFIC;\r
+\r
+       if (SetupDiSetClassInstallParams(info, dev_info_data, &p.ClassInstallHeader, sizeof(p)) == false ||\r
+               SetupDiCallClassInstaller(DIF_PROPERTYCHANGE, info, dev_info_data) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 指定したデバイスを停止させる\r
+bool MsStopDevice(HDEVINFO info, SP_DEVINFO_DATA *dev_info_data)\r
+{\r
+       SP_PROPCHANGE_PARAMS p;\r
+       // 引数チェック\r
+       if (info == NULL || dev_info_data == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&p, sizeof(p));\r
+       p.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);\r
+       p.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;\r
+       p.StateChange = DICS_DISABLE;\r
+       p.Scope = DICS_FLAG_CONFIGSPECIFIC;\r
+\r
+       if (SetupDiSetClassInstallParams(info, dev_info_data, &p.ClassInstallHeader, sizeof(p)) == false ||\r
+               SetupDiCallClassInstaller(DIF_PROPERTYCHANGE, info, dev_info_data) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 指定したデバイスを削除する\r
+bool MsDeleteDevice(HDEVINFO info, SP_DEVINFO_DATA *dev_info_data)\r
+{\r
+       SP_REMOVEDEVICE_PARAMS p;\r
+       SP_DEVINFO_LIST_DETAIL_DATA detail;\r
+       char device_id[MAX_PATH];\r
+       // 引数チェック\r
+       if (info == NULL || dev_info_data == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&detail, sizeof(detail));\r
+       detail.cbSize = sizeof(detail);\r
+\r
+       if (SetupDiGetDeviceInfoListDetail(info, &detail) == false ||\r
+               ms->nt->CM_Get_Device_ID_Ex(dev_info_data->DevInst, device_id, sizeof(device_id),\r
+               0, detail.RemoteMachineHandle) != CR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Zero(&p, sizeof(p));\r
+       p.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);\r
+       p.ClassInstallHeader.InstallFunction = DIF_REMOVE;\r
+       p.Scope = DI_REMOVEDEVICE_GLOBAL;\r
+\r
+       if (SetupDiSetClassInstallParams(info, dev_info_data, &p.ClassInstallHeader, sizeof(p)) == false)\r
+       {\r
+               Debug("SetupDiSetClassInstallParams Failed. Err=%u\n", GetLastError());\r
+               return false;\r
+       }\r
+\r
+       if (SetupDiCallClassInstaller(DIF_REMOVE, info, dev_info_data) == false)\r
+       {\r
+               Debug("SetupDiCallClassInstaller Failed. Err=%u\n", GetLastError());\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 仮想 LAN カードを有効にする\r
+bool MsEnableVLan(char *instance_name)\r
+{\r
+       char tmp[MAX_PATH];\r
+       HDEVINFO h;\r
+       bool ret;\r
+       SP_DEVINFO_DATA data;\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp), DRIVER_DEVICE_ID_TAG, instance_name);\r
+\r
+       h = MsGetDevInfoFromDeviceId(&data, tmp);\r
+       if (h == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       ret = MsStartDevice(h, &data);\r
+\r
+       MsDestroyDevInfo(h);\r
+\r
+       return ret;\r
+}\r
+\r
+// 仮想 LAN カードを無効にする\r
+bool MsDisableVLan(char *instance_name)\r
+{\r
+       char tmp[MAX_PATH];\r
+       HDEVINFO h;\r
+       bool ret;\r
+       SP_DEVINFO_DATA data;\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp), DRIVER_DEVICE_ID_TAG, instance_name);\r
+\r
+       h = MsGetDevInfoFromDeviceId(&data, tmp);\r
+       if (h == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       ret = MsStopDevice(h, &data);\r
+\r
+       MsDestroyDevInfo(h);\r
+\r
+       return ret;\r
+}\r
+\r
+// 仮想 LAN カードを再起動する\r
+void MsRestartVLan(char *instance_name)\r
+{\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (MsIsVLanEnabled(instance_name) == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       MsDisableVLan(instance_name);\r
+       MsEnableVLan(instance_name);\r
+}\r
+\r
+// 仮想 LAN カードが動作しているかどうか取得する\r
+bool MsIsVLanEnabled(char *instance_name)\r
+{\r
+       char tmp[MAX_PATH];\r
+       HDEVINFO h;\r
+       bool ret;\r
+       SP_DEVINFO_DATA data;\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp), DRIVER_DEVICE_ID_TAG, instance_name);\r
+\r
+       h = MsGetDevInfoFromDeviceId(&data, tmp);\r
+       if (h == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       ret = MsIsDeviceRunning(h, &data);\r
+\r
+       MsDestroyDevInfo(h);\r
+\r
+       return ret;\r
+}\r
+\r
+// 仮想 LAN カードをアンインストールする\r
+bool MsUninstallVLan(char *instance_name)\r
+{\r
+       char tmp[MAX_PATH];\r
+       HDEVINFO h;\r
+       bool ret;\r
+       SP_DEVINFO_DATA data;\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp), DRIVER_DEVICE_ID_TAG, instance_name);\r
+\r
+       h = MsGetDevInfoFromDeviceId(&data, tmp);\r
+       if (h == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       ret = MsDeleteDevice(h, &data);\r
+\r
+       MsDestroyDevInfo(h);\r
+\r
+       return ret;\r
+}\r
+\r
+// 汎用テスト関数\r
+void MsTest()\r
+{\r
+}\r
+\r
+// デバイス情報の破棄\r
+void MsDestroyDevInfo(HDEVINFO info)\r
+{\r
+       // 引数チェック\r
+       if (info == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       SetupDiDestroyDeviceInfoList(info);\r
+}\r
+\r
+// ドライバインストールの開始\r
+bool MsStartDriverInstall(char *instance_name, UCHAR *mac_address, char *sen_sys)\r
+{\r
+       wchar_t src_inf[MAX_PATH];\r
+       wchar_t src_sys[MAX_PATH];\r
+       wchar_t dest_inf[MAX_PATH];\r
+       wchar_t dest_sys[MAX_PATH];\r
+       UCHAR mac_address_bin[6];\r
+       char mac_address_str[32];\r
+       UINT size;\r
+       char *tmp;\r
+       BUF *b;\r
+       IO *io;\r
+       // 引数チェック\r
+       if (instance_name == NULL || sen_sys == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       MsGetDriverPath(instance_name, src_inf, src_sys, dest_inf, dest_sys, sen_sys);\r
+\r
+       // INF ファイルの処理\r
+       io = FileOpenW(src_inf, false);\r
+       if (io == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       size = FileSize(io);\r
+       tmp = ZeroMalloc(size * 2);\r
+       if (FileRead(io, tmp, size) == false)\r
+       {\r
+               FileClose(io);\r
+               Free(tmp);\r
+               return false;\r
+       }\r
+\r
+       FileClose(io);\r
+\r
+       if (mac_address == NULL)\r
+       {\r
+               MsGenMacAddress(mac_address_bin);\r
+       }\r
+       else\r
+       {\r
+               Copy(mac_address_bin, mac_address, 6);\r
+       }\r
+\r
+       BinToStr(mac_address_str, sizeof(mac_address_str), mac_address_bin, sizeof(mac_address_bin));\r
+\r
+       //ReplaceStrEx(tmp, size * 2, tmp, "$TAG_DRIVER_VER$", DRIVER_VER_STR, false);\r
+       ReplaceStrEx(tmp, size * 2, tmp, "$TAG_INSTANCE_NAME$", instance_name, false);\r
+       ReplaceStrEx(tmp, size * 2, tmp, "$TAG_MAC_ADDRESS$", mac_address_str, false);\r
+       ReplaceStrEx(tmp, size * 2, tmp, "$TAG_SYS_NAME$", sen_sys, false);\r
+\r
+       if (MsIsVista())\r
+       {\r
+               //ReplaceStrEx(tmp, size * 2, tmp, "\"100\"", "\"2000\"", false);\r
+       }\r
+\r
+       io = FileCreateW(dest_inf);\r
+       if (io == NULL)\r
+       {\r
+               Free(tmp);\r
+               return false;\r
+       }\r
+\r
+       FileWrite(io, tmp, StrLen(tmp));\r
+       FileClose(io);\r
+\r
+       Free(tmp);\r
+\r
+       // SYS ファイルの処理\r
+       b = ReadDumpW(src_sys);\r
+       if (b == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (DumpBufW(b, dest_sys) == false)\r
+       {\r
+               FreeBuf(b);\r
+               return false;\r
+       }\r
+\r
+       FreeBuf(b);\r
+\r
+       return true;\r
+}\r
+\r
+// MAC アドレスの生成\r
+void MsGenMacAddress(UCHAR *mac)\r
+{\r
+       UCHAR hash_src[40];\r
+       UCHAR hash[20];\r
+       UINT64 now;\r
+       // 引数チェック\r
+       if (mac == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Rand(hash_src, 40);\r
+       now = SystemTime64();\r
+       Copy(hash_src, &now, sizeof(now));\r
+\r
+       Hash(hash, hash_src, sizeof(hash_src), true);\r
+\r
+       mac[0] = 0x00;\r
+       mac[1] = 0xAC;\r
+       mac[2] = hash[0];\r
+       mac[3] = hash[1];\r
+       mac[4] = hash[2];\r
+       mac[5] = hash[3];\r
+}\r
+\r
+// ドライバインストールの完了\r
+void MsFinishDriverInstall(char *instance_name, char *sen_sys)\r
+{\r
+       wchar_t src_inf[MAX_PATH];\r
+       wchar_t src_sys[MAX_PATH];\r
+       wchar_t dest_inf[MAX_PATH];\r
+       wchar_t dest_sys[MAX_PATH];\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       MsGetDriverPath(instance_name, src_inf, src_sys, dest_inf, dest_sys, sen_sys);\r
+\r
+       // ファイル削除\r
+       FileDeleteW(dest_inf);\r
+       FileDeleteW(dest_sys);\r
+}\r
+\r
+// ドライバファイルのパスの取得\r
+void MsGetDriverPath(char *instance_name, wchar_t *src_inf, wchar_t *src_sys, wchar_t *dest_inf, wchar_t *dest_sys, char *sen_sys)\r
+{\r
+       wchar_t *src_filename;\r
+       wchar_t *src_sys_filename;\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       src_filename = DRIVER_INF_FILE_NAME;\r
+       src_sys_filename = DRIVER_SYS_FILE_NAME;\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               src_filename = DRIVER_INF_FILE_NAME_9X;\r
+               src_sys_filename = DRIVER_SYS_FILE_NAME_9X;\r
+       }\r
+       else if (MsIsIA64() || MsIsX64())\r
+       {\r
+               if (MsIsX64())\r
+               {\r
+                       src_filename = DRIVER_INF_FILE_NAME_X64;\r
+                       src_sys_filename = DRIVER_SYS_FILE_NAME_X64;\r
+               }\r
+               else\r
+               {\r
+                       src_filename = DRIVER_INF_FILE_NAME_IA64;\r
+                       src_sys_filename = DRIVER_SYS_FILE_NAME_IA64;\r
+               }\r
+       }\r
+\r
+       if (src_inf != NULL)\r
+       {\r
+               UniStrCpy(src_inf, MAX_PATH, src_filename);\r
+       }\r
+\r
+       if (src_sys != NULL)\r
+       {\r
+               UniStrCpy(src_sys, MAX_PATH, src_sys_filename);\r
+       }\r
+\r
+       if (dest_inf != NULL)\r
+       {\r
+               char inf_name[MAX_PATH];\r
+               Format(inf_name, sizeof(inf_name), DRIVER_INSTALL_INF_NAME_TAG, instance_name);\r
+               UniFormat(dest_inf, MAX_PATH, L"%s\\%S", ms->MyTempDirW, inf_name);\r
+       }\r
+\r
+       if (dest_sys != NULL)\r
+       {\r
+               char sys_name[MAX_PATH];\r
+               StrCpy(sys_name, sizeof(sys_name), sen_sys);\r
+               UniFormat(dest_sys, MAX_PATH, L"%s\\%S", ms->MyTempDirW, sys_name);\r
+       }\r
+}\r
+void MsGetDriverPathA(char *instance_name, char *src_inf, char *src_sys, char *dest_inf, char *dest_sys, char *sen_sys)\r
+{\r
+       wchar_t src_inf_w[MAX_PATH];\r
+       wchar_t src_sys_w[MAX_PATH];\r
+       wchar_t dest_inf_w[MAX_PATH];\r
+       wchar_t dest_sys_w[MAX_PATH];\r
+\r
+       // 引数チェック\r
+       if (instance_name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       MsGetDriverPath(instance_name, src_inf_w, src_sys_w, dest_inf_w, dest_sys_w, sen_sys);\r
+\r
+       UniToStr(src_inf, MAX_PATH, src_inf_w);\r
+       UniToStr(src_sys, MAX_PATH, src_sys_w);\r
+       UniToStr(dest_inf, MAX_PATH, dest_inf_w);\r
+       UniToStr(dest_sys, MAX_PATH, dest_sys_w);\r
+}\r
+\r
+// 指定された名前の仮想 LAN カードがすでに登録されているかどうかを調べる\r
+bool MsIsVLanExists(char *tag_name, char *instance_name)\r
+{\r
+       char *guid;\r
+       // 引数チェック\r
+       if (instance_name == NULL || tag_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       guid = MsGetNetworkAdapterGuid(tag_name, instance_name);\r
+       if (guid == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Free(guid);\r
+       return true;\r
+}\r
+\r
+// ネットワーク設定ダイアログを表示する\r
+// ※ これはもう使っていない。うまく動かないからである。\r
+//    代わりに ShowWindowsNetworkConnectionDialog() を使うこと。\r
+bool MsShowNetworkConfiguration(HWND hWnd)\r
+{\r
+       IO *link_file = MsCreateTempFileByExt(".lnk");\r
+       char name[MAX_PATH];\r
+       SHELLEXECUTEINFO info;\r
+\r
+       // ファイル名確保\r
+       StrCpy(name, sizeof(name), link_file->Name);\r
+\r
+       // ショートカット作成\r
+       if (FileWrite(link_file, network_connection_link, sizeof(network_connection_link)) == false)\r
+       {\r
+               FileCloseAndDelete(link_file);\r
+               return false;\r
+       }\r
+\r
+       FileClose(link_file);\r
+\r
+       // ショートカットの実行\r
+       Zero(&info, sizeof(info));\r
+       info.cbSize = sizeof(info);\r
+       info.hwnd = (HWND)hWnd;\r
+       info.lpVerb = "open";\r
+       info.lpFile = name;\r
+       info.nShow = SW_SHOWDEFAULT;\r
+       info.fMask = SEE_MASK_NOCLOSEPROCESS;\r
+       if (ShellExecuteEx(&info) == false)\r
+       {\r
+               FileDelete(name);\r
+               return false;\r
+       }\r
+\r
+       // プロセス終了まで待機\r
+       WaitForSingleObject(info.hProcess, INFINITE);\r
+       CloseHandle(info.hProcess);\r
+\r
+       // ファイルの削除\r
+       FileDelete(name);\r
+\r
+       return true;\r
+}\r
+\r
+// 拡張子を元に一時ファイルを作成する\r
+IO *MsCreateTempFileByExt(char *ext)\r
+{\r
+       char *tmp = MsCreateTempFileNameByExt(ext);\r
+       IO *ret;\r
+\r
+       if (tmp == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = FileCreate(tmp);\r
+       Free(tmp);\r
+\r
+       return ret;\r
+}\r
+\r
+// 拡張子を指定するとその拡張子を持つ一時ファイルを作成する\r
+char *MsCreateTempFileNameByExt(char *ext)\r
+{\r
+       UCHAR rand[2];\r
+       char *ret = NULL;\r
+       // 引数チェック\r
+       if (ext == NULL)\r
+       {\r
+               ext = "tmp";\r
+       }\r
+       if (ext[0] == '.')\r
+       {\r
+               ext++;\r
+       }\r
+       if (StrLen(ext) == 0)\r
+       {\r
+               ext = "tmp";\r
+       }\r
+\r
+       while (true)\r
+       {\r
+               char new_filename[MAX_PATH];\r
+               char *fullpath;\r
+               char rand_str[MAX_PATH];\r
+               IO *io;\r
+               Rand(rand, sizeof(rand));\r
+\r
+               BinToStr(rand_str, sizeof(rand_str), rand, sizeof(rand));\r
+               Format(new_filename, sizeof(new_filename), "__%s.%s", rand_str, ext);\r
+\r
+               fullpath = MsCreateTempFileName(new_filename);\r
+               io = FileOpen(fullpath, false);\r
+               if (io == NULL)\r
+               {\r
+                       ret = fullpath;\r
+                       break;\r
+               }\r
+               FileClose(io);\r
+\r
+               Free(fullpath);\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// 一時ファイルを作成する\r
+IO *MsCreateTempFile(char *name)\r
+{\r
+       IO *ret;\r
+       char *tmp;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       tmp = MsCreateTempFileName(name);\r
+       if (tmp == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = FileCreate(tmp);\r
+       Free(tmp);\r
+\r
+       return ret;\r
+}\r
+\r
+// 一時ファイル名を作成する\r
+char *MsCreateTempFileName(char *name)\r
+{\r
+       char tmp[MAX_PATH];\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp), "%s\\%s", ms->MyTempDir, name);\r
+\r
+       return CopyStr(tmp);\r
+}\r
+\r
+// システムに残っているが使用されていない VPN 用一時ディレクトリを削除する\r
+void MsDeleteTempDir()\r
+{\r
+       HANDLE h;\r
+       wchar_t dir_mask[MAX_PATH];\r
+       WIN32_FIND_DATAA data_a;\r
+       WIN32_FIND_DATAW data_w;\r
+\r
+       Zero(&data_a, sizeof(data_a));\r
+       Zero(&data_w, sizeof(data_w));\r
+\r
+       UniFormat(dir_mask, sizeof(dir_mask), L"%s\\*", ms->TempDirW);\r
+\r
+       if (IsNt())\r
+       {\r
+               h = FindFirstFileW(dir_mask, &data_w);\r
+       }\r
+       else\r
+       {\r
+               char *tmp_a = CopyUniToStr(dir_mask);\r
+\r
+               h = FindFirstFileA(tmp_a, &data_a);\r
+\r
+               Free(tmp_a);\r
+       }\r
+\r
+       if (h != INVALID_HANDLE_VALUE)\r
+       {\r
+               bool b = true;\r
+\r
+               do\r
+               {\r
+                       if (IsNt() == false)\r
+                       {\r
+                               Zero(&data_w, sizeof(data_w));\r
+                               StrToUni(data_w.cFileName, sizeof(data_w.cFileName), data_a.cFileName);\r
+                               data_w.dwFileAttributes = data_a.dwFileAttributes;\r
+                               data_w.ftCreationTime = data_a.ftCreationTime;\r
+                               data_w.ftLastWriteTime = data_a.ftLastWriteTime;\r
+                               data_w.nFileSizeHigh = data_a.nFileSizeHigh;\r
+                               data_w.nFileSizeLow = data_a.nFileSizeLow;\r
+                       }\r
+\r
+                       if (UniStrCmpi(data_w.cFileName, L".") != 0 &&\r
+                               UniStrCmpi(data_w.cFileName, L"..") != 0)\r
+                       {\r
+                               if (data_w.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)\r
+                               {\r
+                                       if (UniStartWith(data_w.cFileName, L"VPN_") && UniStrLen(data_w.cFileName) == 8)\r
+                                       {\r
+                                               wchar_t lock_file_name[MAX_PATH];\r
+                                               wchar_t dir_name[MAX_PATH];\r
+                                               bool delete_now = false;\r
+                                               IO *io;\r
+\r
+                                               UniFormat(dir_name, sizeof(dir_name), L"%s\\%s",\r
+                                                       ms->TempDirW, data_w.cFileName);\r
+                                               MsGenLockFile(lock_file_name, sizeof(lock_file_name), dir_name);\r
+\r
+                                               io = FileOpenExW(lock_file_name, false, false);\r
+                                               if (io != NULL)\r
+                                               {\r
+                                                       // ロックファイルがロックされていなければ削除マーク\r
+                                                       FileClose(io);\r
+                                                       io = FileOpenW(lock_file_name, true);\r
+                                                       if (io != NULL)\r
+                                                       {\r
+                                                               delete_now = true;\r
+                                                               FileClose(io);\r
+                                                       }\r
+                                               }\r
+                                               else\r
+                                               {\r
+                                                       DIRLIST *d;\r
+\r
+                                                       // 中にあるすべてのファイルがロックされていなければ削除マーク\r
+                                                       delete_now = true;\r
+\r
+                                                       d = EnumDirW(dir_name);\r
+                                                       if (d != NULL)\r
+                                                       {\r
+                                                               UINT i;\r
+\r
+                                                               for (i = 0;i < d->NumFiles;i++)\r
+                                                               {\r
+                                                                       wchar_t full_path[MAX_PATH];\r
+\r
+                                                                       UniFormat(full_path, sizeof(full_path), L"%s\\%s", dir_name, d->File[i]->FileNameW);\r
+\r
+                                                                       io = FileOpenW(full_path, true);\r
+                                                                       if (io != NULL)\r
+                                                                       {\r
+                                                                               delete_now = true;\r
+                                                                               FileClose(io);\r
+                                                                       }\r
+                                                               }\r
+                                                               FreeDir(d);\r
+                                                       }\r
+                                               }\r
+                                               if (delete_now)\r
+                                               {\r
+                                                       MsDeleteAllFileW(dir_name);\r
+\r
+                                                       Win32DeleteDirW(dir_name);\r
+                                               }\r
+                                       }\r
+                               }\r
+                       }\r
+\r
+\r
+                       Zero(&data_w, sizeof(data_w));\r
+                       Zero(&data_a, sizeof(data_a));\r
+\r
+                       if (IsNt())\r
+                       {\r
+                               b = FindNextFileW(h, &data_w);\r
+                       }\r
+                       else\r
+                       {\r
+                               b = FindNextFileA(h, &data_a);\r
+                       }\r
+               }\r
+               while (b);\r
+\r
+               FindClose(h);\r
+       }\r
+}\r
+\r
+// 指定したディレクトリ内のファイルをすべて削除する\r
+void MsDeleteAllFile(char *dir)\r
+{\r
+       HANDLE h;\r
+       char file_mask[MAX_PATH];\r
+       WIN32_FIND_DATA data;\r
+       // 引数チェック\r
+       if (dir == NULL || IsEmptyStr(dir))\r
+       {\r
+               return;\r
+       }\r
+\r
+       Format(file_mask, sizeof(file_mask), "%s\\*.*", dir);\r
+\r
+       h = FindFirstFile(file_mask, &data);\r
+       if (h != INVALID_HANDLE_VALUE)\r
+       {\r
+               do\r
+               {\r
+                       if (StrCmpi(data.cFileName, ".") != 0 &&\r
+                               StrCmpi(data.cFileName, "..") != 0)\r
+                       {\r
+                               char fullpath[MAX_PATH];\r
+                               Format(fullpath, sizeof(fullpath), "%s\\%s", dir, data.cFileName);\r
+                               if ((data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == false)\r
+                               {\r
+                                       DeleteFile(fullpath);\r
+                               }\r
+                               else\r
+                               {\r
+                                       MsDeleteAllFile(fullpath);\r
+                                       RemoveDirectory(fullpath);\r
+                               }\r
+                       }\r
+               }\r
+               while (FindNextFile(h, &data));\r
+\r
+               FindClose(h);\r
+       }\r
+}\r
+void MsDeleteAllFileW(wchar_t *dir)\r
+{\r
+       HANDLE h;\r
+       wchar_t file_mask[MAX_PATH];\r
+       WIN32_FIND_DATAW data;\r
+       // 引数チェック\r
+       if (dir == NULL || UniIsEmptyStr(dir))\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char *dir_a = CopyUniToStr(dir);\r
+\r
+               MsDeleteAllFile(dir_a);\r
+\r
+               Free(dir_a);\r
+\r
+               return;\r
+       }\r
+\r
+       UniFormat(file_mask, sizeof(file_mask), L"%s\\*.*", dir);\r
+\r
+       h = FindFirstFileW(file_mask, &data);\r
+       if (h != INVALID_HANDLE_VALUE)\r
+       {\r
+               do\r
+               {\r
+                       if (UniStrCmpi(data.cFileName, L".") != 0 &&\r
+                               UniStrCmpi(data.cFileName, L"..") != 0)\r
+                       {\r
+                               wchar_t fullpath[MAX_PATH];\r
+\r
+                               UniFormat(fullpath, sizeof(fullpath), L"%s\\%s", dir, data.cFileName);\r
+\r
+                               if ((data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == false)\r
+                               {\r
+                                       DeleteFileW(fullpath);\r
+                               }\r
+                               else\r
+                               {\r
+                                       MsDeleteAllFileW(fullpath);\r
+                                       RemoveDirectoryW(fullpath);\r
+                               }\r
+                       }\r
+               }\r
+               while (FindNextFileW(h, &data));\r
+\r
+               FindClose(h);\r
+       }\r
+}\r
+\r
+// 一時ディレクトリを初期化する\r
+void MsInitTempDir()\r
+{\r
+       wchar_t tmp[MAX_PATH];\r
+       wchar_t tmp2[16];\r
+       UCHAR random[2];\r
+       wchar_t lockfilename[MAX_PATH];\r
+       UINT num = 0;\r
+\r
+       // 使われていない一時ディレクトリの削除\r
+       MsDeleteTempDir();\r
+\r
+       // 一時ディレクトリ名の決定\r
+       while (true)\r
+       {\r
+               random[0] = rand() % 256;\r
+               random[1] = rand() % 256;\r
+               BinToStrW(tmp2, sizeof(tmp2), random, sizeof(random));\r
+\r
+               UniFormat(tmp, sizeof(tmp), L"%s\\VPN_%s", ms->TempDirW, tmp2);\r
+\r
+               // ディレクトリの作成\r
+               if (MakeDirW(tmp))\r
+               {\r
+                       break;\r
+               }\r
+\r
+               if ((num++) >= 100)\r
+               {\r
+                       // 何度も失敗する\r
+                       char msg[MAX_SIZE];\r
+                       Format(msg, sizeof(msg),\r
+                               "Couldn't create Temporary Directory: %s\r\n\r\n"\r
+                               "Please contact your system administrator.",\r
+                               tmp);\r
+                       exit(0);\r
+               }\r
+       }\r
+\r
+       ms->MyTempDirW = CopyUniStr(tmp);\r
+       ms->MyTempDir = CopyUniToStr(tmp);\r
+\r
+       // ロックファイルの作成\r
+       MsGenLockFile(lockfilename, sizeof(lockfilename), ms->MyTempDirW);\r
+       ms->LockFile = FileCreateW(lockfilename);\r
+}\r
+\r
+// 一時ディレクトリを解放する\r
+void MsFreeTempDir()\r
+{\r
+       wchar_t lock_file_name[MAX_SIZE];\r
+\r
+       // ロックファイルの削除\r
+       MsGenLockFile(lock_file_name, sizeof(lock_file_name), ms->MyTempDirW);\r
+       FileClose(ms->LockFile);\r
+\r
+       // メモリ解放\r
+       Free(ms->MyTempDir);\r
+       Free(ms->MyTempDirW);\r
+       ms->MyTempDir = NULL;\r
+       ms->MyTempDirW = NULL;\r
+\r
+       // ディレクトリ削除\r
+       MsDeleteTempDir();\r
+}\r
+\r
+// ロックファイル名の生成\r
+void MsGenLockFile(wchar_t *name, UINT size, wchar_t *temp_dir)\r
+{\r
+       // 引数チェック\r
+       if (name == NULL || temp_dir == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       UniFormat(name, size, L"%s\\VPN_Lock.dat", temp_dir);\r
+}\r
+\r
+// ネットワーク設定の初期化\r
+void MsInitNetworkConfig(char *tag_name, char *instance_name, char *connection_tag_name)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       char *config_str;\r
+       // 引数チェック\r
+       if (tag_name == NULL || instance_name == NULL || connection_tag_name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // 文字列などの設定\r
+       Format(tmp, sizeof(tmp), connection_tag_name, instance_name);\r
+       MsSetNetworkConfig(tag_name, instance_name, tmp, true);\r
+\r
+       // インターフェイス・メトリック値の設定\r
+       config_str = MsGetNetworkAdapterGuid(tag_name, instance_name);\r
+       if (config_str != NULL)\r
+       {\r
+               Format(tmp, sizeof(tmp), "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters\\Interfaces\\%s",\r
+                       config_str);\r
+\r
+               MsRegWriteInt(REG_LOCAL_MACHINE, tmp, "InterfaceMetric", 1);\r
+               MsRegWriteInt(REG_LOCAL_MACHINE, tmp, "EnableDeadGWDetect", 0);\r
+\r
+               if (MsRegReadInt(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                       "packetix_no_optimize") == 0)\r
+               {\r
+                       MsRegWriteInt(REG_LOCAL_MACHINE,\r
+                               "SYSTEM\\CurrentControlSet\\Services\\Tcpip\\Parameters",\r
+                               "EnableDeadGWDetect",\r
+                               0);\r
+               }\r
+\r
+               Free(config_str);\r
+       }\r
+}\r
+\r
+// ネットワーク設定を行う\r
+void MsSetNetworkConfig(char *tag_name, char *instance_name, char *friendly_name, bool show_icon)\r
+{\r
+       char *key;\r
+       char *old_name;\r
+       // 引数チェック\r
+       if (tag_name == NULL || instance_name == NULL || friendly_name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       key = MsGetNetworkConfigRegKeyNameFromInstanceName(tag_name, instance_name);\r
+       if (key == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       old_name = MsRegReadStr(REG_LOCAL_MACHINE, key, "Name");\r
+       if (old_name != NULL)\r
+       {\r
+               if (MsIsVista())\r
+               {\r
+                       char arg[MAX_PATH];\r
+                       char netsh[MAX_PATH];\r
+\r
+                       Format(netsh, sizeof(netsh), "%s\\netsh.exe", MsGetSystem32Dir());\r
+\r
+                       if (StrCmp(old_name, friendly_name) != 0)\r
+                       {\r
+                               Format(arg, sizeof(arg), "interface set interface name=\"%s\" newname=\"%s\"",\r
+                                       old_name, friendly_name);\r
+\r
+                               Run(netsh, arg, true, true);\r
+                       }\r
+\r
+                       Format(arg, sizeof(arg), "netsh interface ipv4 set interface interface=\"%s\" metric=1",\r
+                               friendly_name);\r
+\r
+                       Run(netsh, arg, true, true);\r
+               }\r
+       }\r
+\r
+       if (StrCmp(old_name, friendly_name) != 0)\r
+       {\r
+               MsRegWriteStr(REG_LOCAL_MACHINE, key, "Name", friendly_name);\r
+       }\r
+\r
+       MsRegWriteInt(REG_LOCAL_MACHINE, key, "ShowIcon", show_icon ? 1 : 0);\r
+\r
+       Free(key);\r
+\r
+       Free(old_name);\r
+}\r
+\r
+// ネットワーク設定キー名をインスタンス名から取得\r
+char *MsGetNetworkConfigRegKeyNameFromInstanceName(char *tag_name, char *instance_name)\r
+{\r
+       char *guid, *ret;\r
+       // 引数チェック\r
+       if (tag_name == NULL || instance_name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       guid = MsGetNetworkAdapterGuid(tag_name, instance_name);\r
+       if (guid == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = MsGetNetworkConfigRegKeyNameFromGuid(guid);\r
+\r
+       Free(guid);\r
+\r
+       return ret;\r
+}\r
+\r
+// ネットワーク設定キー名を GUID から取得\r
+char *MsGetNetworkConfigRegKeyNameFromGuid(char *guid)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       // 引数チェック\r
+       if (guid == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp),\r
+               "SYSTEM\\CurrentControlSet\\Control\\Network\\{4D36E972-E325-11CE-BFC1-08002BE10318}\\%s\\Connection",\r
+               guid);\r
+\r
+       return CopyStr(tmp);\r
+}\r
+\r
+// MAC アドレスの設定\r
+void MsSetMacAddress(char *tag_name, char *instance_name, char *mac_address)\r
+{\r
+       TOKEN_LIST *key_list;\r
+       UINT i;\r
+       char dest_name[MAX_SIZE];\r
+       char mac_str[MAX_SIZE];\r
+       // 引数チェック\r
+       if (tag_name == NULL || instance_name == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // MAC アドレスの正規化\r
+       if (NormalizeMacAddress(mac_str, sizeof(mac_str), mac_address) == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // 目的の名前を生成\r
+       Format(dest_name, sizeof(dest_name), tag_name, instance_name);\r
+\r
+       // キーを列挙\r
+       if (MsIsNt())\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}");\r
+       }\r
+       else\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "System\\CurrentControlSet\\Services\\Class\\Net");\r
+       }\r
+       if (key_list == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < key_list->NumTokens;i++)\r
+       {\r
+               char *key_name = key_list->Token[i];\r
+               char full_key_name[MAX_SIZE];\r
+               char *driver_desc;\r
+\r
+               if (MsIsNt())\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}\\%s",\r
+                               key_name);\r
+               }\r
+               else\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "System\\CurrentControlSet\\Services\\Class\\Net\\%s",\r
+                               key_name);\r
+               }\r
+\r
+               // DriverDesc を読み込む\r
+               driver_desc = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DriverDesc");\r
+               if (driver_desc != NULL)\r
+               {\r
+                       if (StrCmpi(dest_name, driver_desc) == 0)\r
+                       {\r
+                               // MAC アドレスの書き込み\r
+                               MsRegWriteStr(REG_LOCAL_MACHINE, full_key_name, "NetworkAddress", mac_str);\r
+                               Free(driver_desc);\r
+\r
+                               // ドライバの再起動\r
+                               MsRestartVLan(instance_name);\r
+                               break;\r
+                       }\r
+                       Free(driver_desc);\r
+               }\r
+       }\r
+\r
+       FreeToken(key_list);\r
+\r
+       return;\r
+}\r
+\r
+// デバイスドライバのファイル名の取得\r
+char *MsGetDriverFileName(char *tag_name, char *instance_name)\r
+{\r
+       TOKEN_LIST *key_list;\r
+       UINT i;\r
+       char *ret = NULL;\r
+       char dest_name[MAX_SIZE];\r
+       // 引数チェック\r
+       if (tag_name == NULL || instance_name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 目的の名前を生成\r
+       Format(dest_name, sizeof(dest_name), tag_name, instance_name);\r
+\r
+       // キーを列挙\r
+       if (MsIsNt())\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}");\r
+       }\r
+       else\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "System\\CurrentControlSet\\Services\\Class\\Net");\r
+       }\r
+       if (key_list == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       for (i = 0;i < key_list->NumTokens;i++)\r
+       {\r
+               char *key_name = key_list->Token[i];\r
+               char full_key_name[MAX_SIZE];\r
+               char *driver_desc;\r
+\r
+               if (MsIsNt())\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}\\%s",\r
+                               key_name);\r
+               }\r
+               else\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "System\\CurrentControlSet\\Services\\Class\\Net\\%s",\r
+                               key_name);\r
+               }\r
+\r
+               // DriverDesc を読み込む\r
+               driver_desc = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DriverDesc");\r
+               if (driver_desc != NULL)\r
+               {\r
+                       if (StrCmpi(dest_name, driver_desc) == 0)\r
+                       {\r
+                               // ファイル名を読み込む\r
+                               ret = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DeviceVxDs");\r
+                               Free(driver_desc);\r
+                               break;\r
+                       }\r
+                       Free(driver_desc);\r
+               }\r
+       }\r
+\r
+       FreeToken(key_list);\r
+\r
+       return ret;\r
+}\r
+\r
+// デバイスドライバのバージョンの取得\r
+char *MsGetDriverVersion(char *tag_name, char *instance_name)\r
+{\r
+       TOKEN_LIST *key_list;\r
+       TOKEN_LIST *t;\r
+       UINT i;\r
+       char *ret = NULL;\r
+       char dest_name[MAX_SIZE];\r
+       // 引数チェック\r
+       if (tag_name == NULL || instance_name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 目的の名前を生成\r
+       Format(dest_name, sizeof(dest_name), tag_name, instance_name);\r
+\r
+       // キーを列挙\r
+       if (MsIsNt())\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}");\r
+       }\r
+       else\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "System\\CurrentControlSet\\Services\\Class\\Net");\r
+       }\r
+       if (key_list == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       for (i = 0;i < key_list->NumTokens;i++)\r
+       {\r
+               char *key_name = key_list->Token[i];\r
+               char full_key_name[MAX_SIZE];\r
+               char *driver_desc;\r
+\r
+               if (MsIsNt())\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}\\%s",\r
+                               key_name);\r
+               }\r
+               else\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "System\\CurrentControlSet\\Services\\Class\\Net\\%s",\r
+                               key_name);\r
+               }\r
+\r
+               // DriverDesc を読み込む\r
+               driver_desc = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DriverDesc");\r
+               if (driver_desc != NULL)\r
+               {\r
+                       if (StrCmpi(dest_name, driver_desc) == 0)\r
+                       {\r
+                               // バージョン情報を読み込む\r
+                               ret = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DriverVersion");\r
+                               if (ret == NULL)\r
+                               {\r
+                                       ret = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "SenVersion");\r
+                               }\r
+                               Free(driver_desc);\r
+                               break;\r
+                       }\r
+                       Free(driver_desc);\r
+               }\r
+       }\r
+\r
+       FreeToken(key_list);\r
+\r
+       if (ret == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       t = ParseToken(ret, ", ");\r
+       if (t->NumTokens == 2)\r
+       {\r
+               Free(ret);\r
+               ret = CopyStr(t->Token[1]);\r
+       }\r
+       FreeToken(t);\r
+\r
+       return ret;\r
+}\r
+\r
+// MAC アドレスの取得\r
+char *MsGetMacAddress(char *tag_name, char *instance_name)\r
+{\r
+       TOKEN_LIST *key_list;\r
+       UINT i;\r
+       char *ret = NULL;\r
+       char dest_name[MAX_SIZE];\r
+       // 引数チェック\r
+       if (tag_name == NULL || instance_name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 目的の名前を生成\r
+       Format(dest_name, sizeof(dest_name), tag_name, instance_name);\r
+\r
+       // キーを列挙\r
+       if (MsIsNt())\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}");\r
+       }\r
+       else\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "System\\CurrentControlSet\\Services\\Class\\Net");\r
+       }\r
+\r
+       if (key_list == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       for (i = 0;i < key_list->NumTokens;i++)\r
+       {\r
+               char *key_name = key_list->Token[i];\r
+               char full_key_name[MAX_SIZE];\r
+               char *driver_desc;\r
+\r
+               if (MsIsNt())\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}\\%s",\r
+                               key_name);\r
+               }\r
+               else\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "System\\CurrentControlSet\\Services\\Class\\Net\\%s",\r
+                               key_name);\r
+               }\r
+\r
+               // DriverDesc を読み込む\r
+               driver_desc = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DriverDesc");\r
+               if (driver_desc != NULL)\r
+               {\r
+                       if (StrCmpi(dest_name, driver_desc) == 0)\r
+                       {\r
+                               // MAC アドレスを読み込む\r
+                               ret = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "NetworkAddress");\r
+\r
+                               if (IsEmptyStr(ret) == false)\r
+                               {\r
+                                       // MAC アドレスにハイフンを入れる\r
+                                       BUF *b = StrToBin(ret);\r
+                                       if (b != NULL && b->Size == 6)\r
+                                       {\r
+                                               char tmp[MAX_SIZE];\r
+                                               MacToStr(tmp, sizeof(tmp), b->Buf);\r
+\r
+                                               Free(ret);\r
+                                               ret = CopyStr(tmp);\r
+                                       }\r
+                                       FreeBuf(b);\r
+                               }\r
+\r
+                               Free(driver_desc);\r
+                               break;\r
+                       }\r
+                       Free(driver_desc);\r
+               }\r
+       }\r
+\r
+       FreeToken(key_list);\r
+\r
+       return ret;\r
+}\r
+\r
+// 仮想 LAN カードのデバイス名が本当に存在するかどうかチェックする\r
+bool MsCheckVLanDeviceIdFromRootEnum(char *name)\r
+{\r
+       TOKEN_LIST *t;\r
+       char *root;\r
+       char *keyname;\r
+       UINT i;\r
+       bool ret;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt())\r
+       {\r
+               root = "SYSTEM\\CurrentControlSet\\Enum\\Root\\NET";\r
+               keyname = "HardwareID";\r
+       }\r
+       else\r
+       {\r
+               root = "Enum\\Root\\Net";\r
+               keyname = "CompatibleIDs";\r
+       }\r
+\r
+       t = MsRegEnumKey(REG_LOCAL_MACHINE, root);\r
+       if (t == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       ret = false;\r
+\r
+       for (i = 0;i < t->NumTokens;i++)\r
+       {\r
+               char *subname = t->Token[i];\r
+               char fullname[MAX_SIZE];\r
+               char *value;\r
+\r
+               Format(fullname, sizeof(fullname), "%s\\%s", root, subname);\r
+\r
+               value = MsRegReadStr(REG_LOCAL_MACHINE, fullname, keyname);\r
+               if (value != NULL)\r
+               {\r
+                       if (StrCmpi(value, name) == 0)\r
+                       {\r
+                               ret = true;\r
+                       }\r
+                       Free(value);\r
+               }\r
+\r
+               if (ret)\r
+               {\r
+                       break;\r
+               }\r
+       }\r
+\r
+       FreeToken(t);\r
+\r
+       return ret;\r
+}\r
+\r
+// ネットワークアダプタの GUID の取得\r
+char *MsGetNetworkAdapterGuid(char *tag_name, char *instance_name)\r
+{\r
+       TOKEN_LIST *key_list;\r
+       UINT i;\r
+       char *ret = NULL;\r
+       char dest_name[MAX_SIZE];\r
+       // 引数チェック\r
+       if (tag_name == NULL || instance_name == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 目的の名前を生成\r
+       Format(dest_name, sizeof(dest_name), tag_name, instance_name);\r
+\r
+       // キーを列挙\r
+       if (MsIsNt())\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}");\r
+       }\r
+       else\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "System\\CurrentControlSet\\Services\\Class\\Net");\r
+       }\r
+       if (key_list == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       for (i = 0;i < key_list->NumTokens;i++)\r
+       {\r
+               char *key_name = key_list->Token[i];\r
+               char full_key_name[MAX_SIZE];\r
+               char *driver_desc;\r
+               char *device_id;\r
+\r
+               if (MsIsNt())\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}\\%s",\r
+                               key_name);\r
+               }\r
+               else\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "System\\CurrentControlSet\\Services\\Class\\Net\\%s",\r
+                               key_name);\r
+               }\r
+\r
+               device_id = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "MatchingDeviceId");\r
+\r
+               if (device_id != NULL)\r
+               {\r
+                       if (MsCheckVLanDeviceIdFromRootEnum(device_id))\r
+                       {\r
+                               // DriverDesc を読み込む\r
+                               driver_desc = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DriverDesc");\r
+                               if (driver_desc != NULL)\r
+                               {\r
+                                       if (StrCmpi(dest_name, driver_desc) == 0)\r
+                                       {\r
+                                               // NetCfgInstanceId を読み込む\r
+                                               if (MsIsNt())\r
+                                               {\r
+                                                       ret = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "NetCfgInstanceId");\r
+                                               }\r
+                                               else\r
+                                               {\r
+                                                       ret = CopyStr("");\r
+                                               }\r
+                                               Free(driver_desc);\r
+                                               Free(device_id);\r
+                                               break;\r
+                                       }\r
+                                       Free(driver_desc);\r
+                               }\r
+                       }\r
+                       Free(device_id);\r
+               }\r
+       }\r
+\r
+       FreeToken(key_list);\r
+\r
+       return ret;\r
+}\r
+// ネットワーク接続名の取得\r
+wchar_t *MsGetNetworkConnectionName(char *guid)\r
+{\r
+       wchar_t *ncname = NULL;\r
+       // 引数チェック\r
+       if (guid == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // ネットワーク接続名を取得\r
+       if (IsNt() != false && GetOsInfo()->OsType >= OSTYPE_WINDOWS_2000_PROFESSIONAL)\r
+       {\r
+               char tmp[MAX_SIZE];\r
+               Format(tmp, sizeof(tmp), "SYSTEM\\CurrentControlSet\\Control\\Network\\{4D36E972-E325-11CE-BFC1-08002BE10318}\\%s\\Connection", guid);\r
+               ncname = MsRegReadStrW(REG_LOCAL_MACHINE, tmp, "Name");\r
+       }\r
+\r
+       return ncname;\r
+}\r
+\r
+// 新しい Sen のドライバファイル名を生成\r
+bool MsMakeNewSenDriverFilename(char *name, UINT size)\r
+{\r
+       TOKEN_LIST *t = MsEnumSenDriverFilenames();\r
+       UINT i;\r
+       bool ret = false;\r
+\r
+       i = 0;\r
+       while (true)\r
+       {\r
+               char tmp[MAX_PATH];\r
+               UINT n;\r
+\r
+               i++;\r
+               if (i >= 10000)\r
+               {\r
+                       break;\r
+               }\r
+\r
+               n = Rand32() % DRIVER_INSTALL_SYS_NAME_TAG_MAXID;\r
+\r
+               MsGenerateSenDriverFilenameFromInt(tmp, sizeof(tmp), n);\r
+\r
+               if (IsInToken(t, tmp) == false)\r
+               {\r
+                       StrCpy(name, size, tmp);\r
+                       ret = true;\r
+                       break;\r
+               }\r
+       }\r
+\r
+       FreeToken(t);\r
+\r
+       return ret;\r
+}\r
+\r
+// Sen のドライバファイル名を整数から生成\r
+void MsGenerateSenDriverFilenameFromInt(char *name, UINT size, UINT n)\r
+{\r
+       Format(name, size, DRIVER_INSTALL_SYS_NAME_TAG_NEW, n);\r
+}\r
+\r
+// インストールされている Sen のドライバファイル名の列挙\r
+TOKEN_LIST *MsEnumSenDriverFilenames()\r
+{\r
+       TOKEN_LIST *neos = MsEnumNetworkAdaptersSen();\r
+       LIST *o = NewListFast(NULL);\r
+       TOKEN_LIST *ret;\r
+       UINT i;\r
+\r
+       for (i = 0;i < neos->NumTokens;i++)\r
+       {\r
+               char filename[MAX_PATH];\r
+               if (MsGetSenDeiverFilename(filename, sizeof(filename), neos->Token[i]))\r
+               {\r
+                       Add(o, CopyStr(filename));\r
+               }\r
+       }\r
+\r
+       FreeToken(neos);\r
+\r
+       ret = ListToTokenList(o);\r
+       FreeStrList(o);\r
+\r
+       return ret;\r
+}\r
+\r
+// Sen のドライバファイル名を取得\r
+bool MsGetSenDeiverFilename(char *name, UINT size, char *instance_name)\r
+{\r
+       char tmp[MAX_SIZE];\r
+       char *ret;\r
+       // 引数チェック\r
+       if (name == NULL || instance_name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       Format(tmp, sizeof(tmp), "SYSTEM\\CurrentControlSet\\Services\\Sen_%s", instance_name);\r
+\r
+       ret = MsRegReadStr(REG_LOCAL_MACHINE, tmp, "ImagePath");\r
+       if (ret == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       GetFileNameFromFilePath(name, size, ret);\r
+       Free(ret);\r
+\r
+       return true;\r
+}\r
+\r
+// ネットワークアダプタの列挙 (Sen のみ)\r
+TOKEN_LIST *MsEnumNetworkAdaptersSen()\r
+{\r
+       TOKEN_LIST *key_list;\r
+       TOKEN_LIST *ret;\r
+       LIST *o;\r
+       UINT i;\r
+\r
+       // キーを列挙\r
+       if (MsIsNt())\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}");\r
+       }\r
+       else\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "System\\CurrentControlSet\\Services\\Class\\Net");\r
+       }\r
+       if (key_list == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       o = NewListFast(CompareStr);\r
+\r
+       for (i = 0;i < key_list->NumTokens;i++)\r
+       {\r
+               char *key_name = key_list->Token[i];\r
+               char full_key_name[MAX_SIZE];\r
+               char *driver_desc;\r
+               char *device_id;\r
+\r
+               if (MsIsNt())\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}\\%s",\r
+                               key_name);\r
+               }\r
+               else\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "System\\CurrentControlSet\\Services\\Class\\Net\\%s",\r
+                               key_name);\r
+               }\r
+\r
+               // DriverDesc を読み込む\r
+               driver_desc = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DriverDesc");\r
+               if (driver_desc != NULL)\r
+               {\r
+                       // 特定の名前で始まっているかどうか確認する\r
+                       device_id = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "MatchingDeviceId");\r
+\r
+                       if (device_id != NULL)\r
+                       {\r
+                               if (MsCheckVLanDeviceIdFromRootEnum(device_id))\r
+                               {\r
+                                       char *tag = "senadapter_";\r
+                                       if (StartWith(device_id, tag))\r
+                                       {\r
+                                               char tmp[MAX_SIZE];\r
+                                               StrCpy(tmp, sizeof(tmp), &device_id[StrLen(tag)]);\r
+\r
+                                               Add(o, CopyStr(tmp));\r
+                                       }\r
+                               }\r
+                               Free(device_id);\r
+                       }\r
+\r
+                       Free(driver_desc);\r
+               }\r
+       }\r
+\r
+       FreeToken(key_list);\r
+\r
+       ret = ZeroMalloc(sizeof(TOKEN_LIST));\r
+       ret->NumTokens = LIST_NUM(o);\r
+       ret->Token = ZeroMalloc(sizeof(char *) * ret->NumTokens);\r
+       for (i = 0;i < ret->NumTokens;i++)\r
+       {\r
+               ret->Token[i] = LIST_DATA(o, i);\r
+       }\r
+\r
+       ReleaseList(o);\r
+\r
+       return ret;\r
+}\r
+\r
+// ネットワークアダプタの列挙\r
+TOKEN_LIST *MsEnumNetworkAdapters(char *start_with_name, char *start_with_name_2)\r
+{\r
+       TOKEN_LIST *key_list;\r
+       TOKEN_LIST *ret;\r
+       LIST *o;\r
+       UINT i;\r
+\r
+       // キーを列挙\r
+       if (MsIsNt())\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}");\r
+       }\r
+       else\r
+       {\r
+               key_list = MsRegEnumKey(REG_LOCAL_MACHINE,\r
+                       "System\\CurrentControlSet\\Services\\Class\\Net");\r
+       }\r
+       if (key_list == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       o = NewListFast(CompareStr);\r
+\r
+       for (i = 0;i < key_list->NumTokens;i++)\r
+       {\r
+               char *key_name = key_list->Token[i];\r
+               char full_key_name[MAX_SIZE];\r
+               char *driver_desc;\r
+               char *device_id;\r
+\r
+               if (MsIsNt())\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002bE10318}\\%s",\r
+                               key_name);\r
+               }\r
+               else\r
+               {\r
+                       Format(full_key_name, sizeof(full_key_name),\r
+                               "System\\CurrentControlSet\\Services\\Class\\Net\\%s",\r
+                               key_name);\r
+               }\r
+\r
+               // DriverDesc を読み込む\r
+               driver_desc = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "DriverDesc");\r
+               if (driver_desc != NULL)\r
+               {\r
+                       // 特定の名前で始まっているかどうか確認する\r
+                       if ((IsEmptyStr(start_with_name) && IsEmptyStr(start_with_name_2)) ||\r
+                               (StartWith(driver_desc, start_with_name) || StartWith(driver_desc, start_with_name_2)))\r
+                       {\r
+                               device_id = MsRegReadStr(REG_LOCAL_MACHINE, full_key_name, "MatchingDeviceId");\r
+\r
+                               if (device_id != NULL)\r
+                               {\r
+                                       if (MsCheckVLanDeviceIdFromRootEnum(device_id))\r
+                                       {\r
+                                               char instance_name[MAX_SIZE];\r
+                                               // 名前からインスタンス名だけを抽出する\r
+                                               if (StartWith(driver_desc, start_with_name))\r
+                                               {\r
+                                                       if (StrLen(driver_desc) > (StrLen(start_with_name) + 3))\r
+                                                       {\r
+                                                               StrCpy(instance_name, sizeof(instance_name),\r
+                                                                       driver_desc + StrLen(start_with_name) + 3);\r
+                                                               Add(o, CopyStr(instance_name));\r
+                                                       }\r
+                                               }\r
+                                               else\r
+                                               {\r
+                                                       if (StrLen(driver_desc) > (StrLen(start_with_name_2) + 3))\r
+                                                       {\r
+                                                               StrCpy(instance_name, sizeof(instance_name),\r
+                                                                       driver_desc + StrLen(start_with_name_2) + 3);\r
+                                                               Add(o, CopyStr(instance_name));\r
+                                                       }\r
+                                               }\r
+                                       }\r
+                                       Free(device_id);\r
+                               }\r
+                       }\r
+\r
+                       Free(driver_desc);\r
+               }\r
+       }\r
+\r
+       FreeToken(key_list);\r
+\r
+       ret = ZeroMalloc(sizeof(TOKEN_LIST));\r
+       ret->NumTokens = LIST_NUM(o);\r
+       ret->Token = ZeroMalloc(sizeof(char *) * ret->NumTokens);\r
+       for (i = 0;i < ret->NumTokens;i++)\r
+       {\r
+               ret->Token[i] = LIST_DATA(o, i);\r
+       }\r
+\r
+       ReleaseList(o);\r
+\r
+       return ret;\r
+}\r
+\r
+// ドメインへのログオンを試行する\r
+bool MsCheckLogon(wchar_t *username, char *password)\r
+{\r
+       wchar_t password_unicode[MAX_SIZE];\r
+       HANDLE h;\r
+       // 引数チェック\r
+       if (username == NULL || password == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       StrToUni(password_unicode, sizeof(password_unicode), password);\r
+\r
+       if (GET_KETA(GetOsInfo()->OsType, 100) >= 2)\r
+       {\r
+               if (ms->nt->LogonUserW(username, NULL, password_unicode, LOGON32_LOGON_NETWORK, LOGON32_PROVIDER_DEFAULT, &h) == false)\r
+               {\r
+                       // ログオン失敗\r
+                       return false;\r
+               }\r
+       }\r
+       else\r
+       {\r
+               char username_ansi[MAX_SIZE];\r
+               UniToStr(username_ansi, sizeof(username_ansi), username);\r
+\r
+               if (ms->nt->LogonUserA(username_ansi, NULL, password, LOGON32_LOGON_NETWORK, LOGON32_PROVIDER_DEFAULT, &h) == false)\r
+               {\r
+                       // ログオン失敗\r
+                       return false;\r
+               }\r
+       }\r
+\r
+       CloseHandle(h);\r
+\r
+       return true;\r
+}\r
+\r
+// ドメインへのログオンを試行する\r
+bool MsIsPasswordEmpty(wchar_t *username)\r
+{\r
+       HANDLE h;\r
+       // 引数チェック\r
+       if (username == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (GET_KETA(GetOsInfo()->OsType, 100) >= 2)\r
+       {\r
+               if (ms->nt->LogonUserW(username, NULL, L"", LOGON32_LOGON_NETWORK, LOGON32_PROVIDER_DEFAULT, &h) == false)\r
+               {\r
+                       // ログオン失敗\r
+                       if (GetLastError() == 1327)\r
+                       {\r
+                               // パスワードが空\r
+                               return true;\r
+                       }\r
+                       else\r
+                       {\r
+                               // パスワードが間違っている\r
+                               return false;\r
+                       }\r
+               }\r
+       }\r
+       else\r
+       {\r
+               char username_ansi[MAX_SIZE];\r
+               UniToStr(username_ansi, sizeof(username_ansi), username);\r
+\r
+               if (ms->nt->LogonUserA(username_ansi, NULL, "", LOGON32_LOGON_NETWORK, LOGON32_PROVIDER_DEFAULT, &h) == false)\r
+               {\r
+                       // ログオン失敗\r
+                       if (GetLastError() == 1327)\r
+                       {\r
+                               // パスワードが空\r
+                               return true;\r
+                       }\r
+                       else\r
+                       {\r
+                               // パスワードが間違っている\r
+                               return false;\r
+                       }\r
+               }\r
+       }\r
+\r
+       CloseHandle(h);\r
+\r
+       // ログオン成功ということはパスワードが空ということになる\r
+       return false;\r
+}\r
+\r
+// シャットダウンの実行 (NT)\r
+bool MsShutdownEx(bool reboot, bool force, UINT time_limit, char *message)\r
+{\r
+       if (MsIsNt() == false)\r
+       {\r
+               return MsShutdown(reboot, force);\r
+       }\r
+\r
+       // 特権の取得\r
+       if (MsEnablePrivilege(SE_SHUTDOWN_NAME, true) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // シャットダウンの実行\r
+       if (ms->nt->InitiateSystemShutdown(NULL, message, time_limit, force, reboot) == false)\r
+       {\r
+               MsEnablePrivilege(SE_SHUTDOWN_NAME, false);\r
+               return false;\r
+       }\r
+\r
+       // 特権の解放\r
+       MsEnablePrivilege(SE_SHUTDOWN_NAME, false);\r
+\r
+       return true;\r
+}\r
+\r
+// シャットダウンの実行\r
+bool MsShutdown(bool reboot, bool force)\r
+{\r
+       UINT flag = 0;\r
+       // 特権の取得\r
+       if (MsEnablePrivilege(SE_SHUTDOWN_NAME, true) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       flag |= (reboot ? EWX_REBOOT : EWX_SHUTDOWN);\r
+       flag |= (force ? EWX_FORCE : 0);\r
+\r
+       // シャットダウンの実行\r
+       if (ExitWindowsEx(flag, 0) == false)\r
+       {\r
+               MsEnablePrivilege(SE_SHUTDOWN_NAME, false);\r
+               return false;\r
+       }\r
+\r
+       // 特権の解放\r
+       MsEnablePrivilege(SE_SHUTDOWN_NAME, false);\r
+\r
+       return true;\r
+}\r
+\r
+// 特権を有効または無効にする\r
+bool MsEnablePrivilege(char *name, bool enable)\r
+{\r
+       HANDLE hToken;\r
+       NT_API *nt = ms->nt;\r
+       LUID luid;\r
+       TOKEN_PRIVILEGES *tp;\r
+       bool ret;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (MsIsNt() == false)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       // プロセストークンを開く\r
+       if (nt->OpenProcessToken(ms->hCurrentProcess, TOKEN_ADJUST_PRIVILEGES, &hToken) == false)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // ローカル一意識別子を取得する\r
+       if (nt->LookupPrivilegeValue(NULL, name, &luid) == FALSE)\r
+       {\r
+               CloseHandle(hToken);\r
+               return false;\r
+       }\r
+\r
+       // 特権を有効 / 無効にするための構造体を作成する\r
+       tp = ZeroMalloc(sizeof(TOKEN_PRIVILEGES));\r
+       tp->PrivilegeCount = 1;\r
+       tp->Privileges[0].Attributes = enable ? SE_PRIVILEGE_ENABLED : 0;\r
+       Copy(&tp->Privileges[0].Luid, &luid, sizeof(LUID));\r
+\r
+       // 特権を操作する\r
+       ret = nt->AdjustTokenPrivileges(hToken, false, tp, sizeof(TOKEN_PRIVILEGES), 0, 0);\r
+\r
+       Free(tp);\r
+       CloseHandle(hToken);\r
+\r
+       return ret;\r
+}\r
+\r
+// 現在の OS が NT 系かどうか取得\r
+bool MsIsNt()\r
+{\r
+       if (ms == NULL)\r
+       {\r
+               OSVERSIONINFO os;\r
+               Zero(&os, sizeof(os));\r
+               os.dwOSVersionInfoSize = sizeof(os);\r
+               GetVersionEx(&os);\r
+               if (os.dwPlatformId == VER_PLATFORM_WIN32_NT)\r
+               {\r
+                       return true;\r
+               }\r
+               else\r
+               {\r
+                       return false;\r
+               }\r
+       }\r
+\r
+       return ms->IsNt;\r
+}\r
+\r
+// 現在のユーザーが Admin かどうか取得\r
+bool MsIsAdmin()\r
+{\r
+       return ms->IsAdmin;\r
+}\r
+\r
+// NT 系関数のロード\r
+NT_API *MsLoadNtApiFunctions()\r
+{\r
+       NT_API *nt = ZeroMalloc(sizeof(NT_API));\r
+       OSVERSIONINFO info;\r
+\r
+       Zero(&info, sizeof(info));\r
+       info.dwOSVersionInfoSize = sizeof(info);\r
+       GetVersionEx(&info);\r
+\r
+       nt->hKernel32 = LoadLibrary("kernel32.dll");\r
+       if (nt->hKernel32 == NULL)\r
+       {\r
+               Free(nt);\r
+               return NULL;\r
+       }\r
+\r
+       nt->hAdvapi32 = LoadLibrary("advapi32.dll");\r
+       if (nt->hAdvapi32 == NULL)\r
+       {\r
+               Free(nt);\r
+               return NULL;\r
+       }\r
+\r
+       nt->hShell32 = LoadLibrary("shell32.dll");\r
+       if (nt->hShell32 == NULL)\r
+       {\r
+               FreeLibrary(nt->hAdvapi32);\r
+               Free(nt);\r
+               return NULL;\r
+       }\r
+\r
+       nt->hPsApi = LoadLibrary("psapi.dll");\r
+\r
+       if (info.dwMajorVersion >= 5)\r
+       {\r
+               nt->hNewDev = LoadLibrary("newdev.dll");\r
+               if (nt->hNewDev == NULL)\r
+               {\r
+                       FreeLibrary(nt->hShell32);\r
+                       FreeLibrary(nt->hAdvapi32);\r
+                       Free(nt);\r
+                       return NULL;\r
+               }\r
+\r
+               nt->hSetupApi = LoadLibrary("setupapi.dll");\r
+       }\r
+\r
+       nt->hSecur32 = LoadLibrary("secur32.dll");\r
+\r
+       nt->hUser32 = LoadLibrary("user32.dll");\r
+\r
+       nt->hDbgHelp = LoadLibrary("dbghelp.dll");\r
+\r
+       // 関数の読み込み\r
+       nt->IsWow64Process =\r
+               (BOOL (__stdcall *)(HANDLE,BOOL *))\r
+               GetProcAddress(nt->hKernel32, "IsWow64Process");\r
+\r
+       nt->GetFileInformationByHandle =\r
+               (BOOL (__stdcall *)(HANDLE,LPBY_HANDLE_FILE_INFORMATION))\r
+               GetProcAddress(nt->hKernel32, "GetFileInformationByHandle");\r
+\r
+       nt->GetProcessHeap =\r
+               (HANDLE (__stdcall *)())\r
+               GetProcAddress(nt->hKernel32, "GetProcessHeap");\r
+\r
+       nt->SetProcessShutdownParameters =\r
+               (BOOL (__stdcall *)(DWORD,DWORD))\r
+               GetProcAddress(nt->hKernel32, "SetProcessShutdownParameters");\r
+\r
+       nt->GetNativeSystemInfo =\r
+               (void (__stdcall *)(SYSTEM_INFO *))\r
+               GetProcAddress(nt->hKernel32, "GetNativeSystemInfo");\r
+\r
+       nt->AdjustTokenPrivileges =\r
+               (BOOL (__stdcall *)(HANDLE,BOOL,PTOKEN_PRIVILEGES,DWORD,PTOKEN_PRIVILEGES,PDWORD))\r
+               GetProcAddress(nt->hAdvapi32, "AdjustTokenPrivileges");\r
+\r
+       nt->LookupPrivilegeValue =\r
+               (BOOL (__stdcall *)(char *,char *,PLUID))\r
+               GetProcAddress(nt->hAdvapi32, "LookupPrivilegeValueA");\r
+\r
+       nt->OpenProcessToken =\r
+               (BOOL (__stdcall *)(HANDLE,DWORD,PHANDLE))\r
+               GetProcAddress(nt->hAdvapi32, "OpenProcessToken");\r
+\r
+       nt->InitiateSystemShutdown =\r
+               (BOOL (__stdcall *)(LPTSTR,LPTSTR,DWORD,BOOL,BOOL))\r
+               GetProcAddress(nt->hAdvapi32, "InitiateSystemShutdownA");\r
+\r
+       nt->LogonUserW =\r
+               (BOOL (__stdcall *)(wchar_t *,wchar_t *,wchar_t *,DWORD,DWORD,HANDLE *))\r
+               GetProcAddress(nt->hAdvapi32, "LogonUserW");\r
+\r
+       nt->LogonUserA =\r
+               (BOOL (__stdcall *)(char *,char *,char *,DWORD,DWORD,HANDLE * ))\r
+               GetProcAddress(nt->hAdvapi32, "LogonUserA");\r
+\r
+       nt->DuplicateTokenEx =\r
+               (BOOL (__stdcall *)(HANDLE,DWORD,SECURITY_ATTRIBUTES *,SECURITY_IMPERSONATION_LEVEL,TOKEN_TYPE,HANDLE *))\r
+               GetProcAddress(nt->hAdvapi32, "DuplicateTokenEx");\r
+\r
+       nt->ConvertStringSidToSidA =\r
+               (BOOL (__stdcall *)(LPCSTR,PSID *))\r
+               GetProcAddress(nt->hAdvapi32, "ConvertStringSidToSidA");\r
+\r
+       nt->GetTokenInformation =\r
+               (BOOL (__stdcall *)(HANDLE,TOKEN_INFORMATION_CLASS,void *,DWORD,PDWORD))\r
+               GetProcAddress(nt->hAdvapi32, "GetTokenInformation");\r
+\r
+       nt->SetTokenInformation =\r
+               (BOOL (__stdcall *)(HANDLE,TOKEN_INFORMATION_CLASS,void *,DWORD))\r
+               GetProcAddress(nt->hAdvapi32, "SetTokenInformation");\r
+\r
+       nt->CreateProcessAsUserA =\r
+               (BOOL (__stdcall *)(HANDLE,LPCSTR,LPSTR,LPSECURITY_ATTRIBUTES,LPSECURITY_ATTRIBUTES,BOOL,DWORD,void *,LPCSTR,LPSTARTUPINFOA,LPPROCESS_INFORMATION))\r
+               GetProcAddress(nt->hAdvapi32, "CreateProcessAsUserA");\r
+\r
+       nt->CreateProcessAsUserW =\r
+               (BOOL (__stdcall *)(HANDLE,LPCWSTR,LPWSTR,LPSECURITY_ATTRIBUTES,LPSECURITY_ATTRIBUTES,BOOL,DWORD,void *,LPCWSTR,LPSTARTUPINFOW,LPPROCESS_INFORMATION))\r
+               GetProcAddress(nt->hAdvapi32, "CreateProcessAsUserW");\r
+\r
+       nt->LookupAccountSidA =\r
+               (BOOL (__stdcall *)(LPCSTR,PSID,LPSTR,LPDWORD,LPSTR,LPDWORD,PSID_NAME_USE))\r
+               GetProcAddress(nt->hAdvapi32, "LookupAccountSidA");\r
+\r
+       nt->LookupAccountNameA =\r
+               (BOOL (__stdcall *)(LPCSTR,LPCSTR,PSID,LPDWORD,LPSTR,LPDWORD,PSID_NAME_USE))\r
+               GetProcAddress(nt->hAdvapi32, "LookupAccountNameA");\r
+\r
+       if (info.dwMajorVersion >= 5)\r
+       {\r
+               nt->UpdateDriverForPlugAndPlayDevicesW =\r
+                       (BOOL (__stdcall *)(HWND,wchar_t *,wchar_t *,UINT,BOOL *))\r
+                       GetProcAddress(nt->hNewDev, "UpdateDriverForPlugAndPlayDevicesW");\r
+\r
+               nt->CM_Get_Device_ID_ExA =\r
+                       (UINT (__stdcall *)(DWORD,char *,UINT,UINT,HANDLE))\r
+                       GetProcAddress(nt->hSetupApi, "CM_Get_Device_ID_ExA");\r
+\r
+               nt->CM_Get_DevNode_Status_Ex =\r
+                       (UINT (__stdcall *)(UINT *,UINT *,DWORD,UINT,HANDLE))\r
+                       GetProcAddress(nt->hSetupApi, "CM_Get_DevNode_Status_Ex");\r
+       }\r
+\r
+       nt->hWtsApi32 = LoadLibrary("wtsapi32.dll");\r
+       if (nt->hWtsApi32 != NULL)\r
+       {\r
+               // ターミナルサービス関係の API\r
+               nt->WTSQuerySessionInformation =\r
+                       (UINT (__stdcall *)(HANDLE,DWORD,WTS_INFO_CLASS,wchar_t *,DWORD *))\r
+                       GetProcAddress(nt->hWtsApi32, "WTSQuerySessionInformationW");\r
+               nt->WTSFreeMemory =\r
+                       (void (__stdcall *)(void *))\r
+                       GetProcAddress(nt->hWtsApi32, "WTSFreeMemory");\r
+               nt->WTSDisconnectSession =\r
+                       (BOOL (__stdcall *)(HANDLE,DWORD,BOOL))\r
+                       GetProcAddress(nt->hWtsApi32, "WTSDisconnectSession");\r
+               nt->WTSEnumerateSessionsA =\r
+                       (BOOL (__stdcall *)(HANDLE,DWORD,DWORD,PWTS_SESSION_INFOA *,DWORD *))\r
+                       GetProcAddress(nt->hWtsApi32, "WTSEnumerateSessionsA");\r
+       }\r
+\r
+       // サービス系 API\r
+       nt->OpenSCManager =\r
+               (SC_HANDLE (__stdcall *)(LPCTSTR,LPCTSTR,DWORD))\r
+               GetProcAddress(nt->hAdvapi32, "OpenSCManagerA");\r
+       nt->CreateServiceA =\r
+               (SC_HANDLE (__stdcall *)(SC_HANDLE,LPCTSTR,LPCTSTR,DWORD,DWORD,DWORD,DWORD,LPCTSTR,LPCTSTR,LPDWORD,LPCTSTR,LPCTSTR,LPCTSTR))\r
+               GetProcAddress(nt->hAdvapi32, "CreateServiceA");\r
+       nt->CreateServiceW =\r
+               (SC_HANDLE (__stdcall *)(SC_HANDLE,LPCWSTR,LPCWSTR,DWORD,DWORD,DWORD,DWORD,LPCWSTR,LPCWSTR,LPDWORD,LPCWSTR,LPCWSTR,LPCWSTR))\r
+               GetProcAddress(nt->hAdvapi32, "CreateServiceW");\r
+       nt->ChangeServiceConfig2 =\r
+               (BOOL (__stdcall *)(SC_HANDLE,DWORD,LPVOID))\r
+               GetProcAddress(nt->hAdvapi32, "ChangeServiceConfig2W");\r
+       nt->CloseServiceHandle =\r
+               (BOOL (__stdcall *)(SC_HANDLE))\r
+               GetProcAddress(nt->hAdvapi32, "CloseServiceHandle");\r
+       nt->OpenService =\r
+               (SC_HANDLE (__stdcall *)(SC_HANDLE,LPCTSTR,DWORD))\r
+               GetProcAddress(nt->hAdvapi32, "OpenServiceA");\r
+       nt->QueryServiceStatus =\r
+               (BOOL (__stdcall *)(SC_HANDLE,LPSERVICE_STATUS))\r
+               GetProcAddress(nt->hAdvapi32, "QueryServiceStatus");\r
+       nt->StartService =\r
+               (BOOL (__stdcall *)(SC_HANDLE,DWORD,LPCTSTR))\r
+               GetProcAddress(nt->hAdvapi32, "StartServiceA");\r
+       nt->ControlService =\r
+               (BOOL (__stdcall *)(SC_HANDLE,DWORD,LPSERVICE_STATUS))\r
+               GetProcAddress(nt->hAdvapi32, "ControlService");\r
+       nt->SetServiceStatus =\r
+               (BOOL (__stdcall *)(SERVICE_STATUS_HANDLE,LPSERVICE_STATUS))\r
+               GetProcAddress(nt->hAdvapi32, "SetServiceStatus");\r
+       nt->RegisterServiceCtrlHandler =\r
+               (SERVICE_STATUS_HANDLE (__stdcall *)(LPCTSTR,LPHANDLER_FUNCTION))\r
+               GetProcAddress(nt->hAdvapi32, "RegisterServiceCtrlHandlerW");\r
+       nt->StartServiceCtrlDispatcher =\r
+               (BOOL (__stdcall *)(const LPSERVICE_TABLE_ENTRY))\r
+               GetProcAddress(nt->hAdvapi32, "StartServiceCtrlDispatcherW");\r
+       nt->DeleteService =\r
+               (BOOL (__stdcall *)(SC_HANDLE))\r
+               GetProcAddress(nt->hAdvapi32, "DeleteService");\r
+       nt->RegisterEventSourceW =\r
+               (HANDLE (__stdcall *)(LPCWSTR,LPCWSTR))\r
+               GetProcAddress(nt->hAdvapi32, "RegisterEventSourceW");\r
+       nt->ReportEventW =\r
+               (BOOL (__stdcall *)(HANDLE,WORD,WORD,DWORD,PSID,WORD,DWORD,LPCWSTR *,LPVOID))\r
+               GetProcAddress(nt->hAdvapi32, "ReportEventW");\r
+       nt->DeregisterEventSource =\r
+               (BOOL (__stdcall *)(HANDLE))\r
+               GetProcAddress(nt->hAdvapi32, "DeregisterEventSource");\r
+       nt->Wow64DisableWow64FsRedirection =\r
+               (BOOL (__stdcall *)(void **))\r
+               GetProcAddress(nt->hKernel32, "Wow64DisableWow64FsRedirection");\r
+       nt->Wow64EnableWow64FsRedirection =\r
+               (BOOLEAN (__stdcall *)(BOOLEAN))\r
+               GetProcAddress(nt->hKernel32, "Wow64EnableWow64FsRedirection");\r
+       nt->Wow64RevertWow64FsRedirection =\r
+               (BOOL (__stdcall *)(void *))\r
+               GetProcAddress(nt->hKernel32, "Wow64RevertWow64FsRedirection");\r
+\r
+       if (nt->hPsApi != NULL)\r
+       {\r
+               // プロセス系 API\r
+               nt->EnumProcesses =\r
+                       (BOOL (__stdcall *)(DWORD *,DWORD,DWORD *))\r
+                       GetProcAddress(nt->hPsApi, "EnumProcesses");\r
+\r
+               nt->EnumProcessModules =\r
+                       (BOOL (__stdcall *)(HANDLE,HMODULE * ,DWORD,DWORD *))\r
+                       GetProcAddress(nt->hPsApi, "EnumProcessModules");\r
+\r
+               nt->GetModuleFileNameExA =\r
+                       (DWORD (__stdcall *)(HANDLE,HMODULE,LPSTR,DWORD))\r
+                       GetProcAddress(nt->hPsApi, "GetModuleFileNameExA");\r
+\r
+               nt->GetModuleFileNameExW =\r
+                       (DWORD (__stdcall *)(HANDLE,HMODULE,LPWSTR,DWORD))\r
+                       GetProcAddress(nt->hPsApi, "GetModuleFileNameExW");\r
+       }\r
+\r
+       // レジストリ系 API\r
+       nt->RegDeleteKeyExA =\r
+               (LONG (__stdcall *)(HKEY,LPCTSTR,REGSAM,DWORD))\r
+               GetProcAddress(nt->hAdvapi32, "RegDeleteKeyExA");\r
+\r
+       // セキュリティ系 API\r
+       if (nt->hSecur32 != NULL)\r
+       {\r
+               nt->GetUserNameExA =\r
+                       (BOOL (__stdcall *)(EXTENDED_NAME_FORMAT,LPSTR,PULONG))\r
+                       GetProcAddress(nt->hSecur32, "GetUserNameExA");\r
+\r
+               nt->GetUserNameExW =\r
+                       (BOOL (__stdcall *)(EXTENDED_NAME_FORMAT,LPWSTR,PULONG))\r
+                       GetProcAddress(nt->hSecur32, "GetUserNameExW");\r
+       }\r
+\r
+       // デスクトップ系 API\r
+       if (nt->hUser32 != NULL)\r
+       {\r
+               nt->SwitchDesktop =\r
+                       (BOOL (__stdcall *)(HDESK))\r
+                       GetProcAddress(nt->hUser32, "SwitchDesktop");\r
+               nt->OpenDesktopA =\r
+                       (HDESK (__stdcall *)(LPTSTR,DWORD,BOOL,ACCESS_MASK))\r
+                       GetProcAddress(nt->hUser32, "OpenDesktopA");\r
+               nt->CloseDesktop =\r
+                       (BOOL (__stdcall *)(HDESK))\r
+                       GetProcAddress(nt->hUser32, "CloseDesktop");\r
+       }\r
+\r
+       // デバッグ系 API\r
+       if (nt->hDbgHelp != NULL)\r
+       {\r
+               nt->MiniDumpWriteDump =\r
+                       (BOOL (__stdcall *)(HANDLE,DWORD,HANDLE,MINIDUMP_TYPE,PMINIDUMP_EXCEPTION_INFORMATION,PMINIDUMP_USER_STREAM_INFORMATION,PMINIDUMP_CALLBACK_INFORMATION))\r
+                       GetProcAddress(nt->hDbgHelp, "MiniDumpWriteDump");\r
+       }\r
+\r
+       return nt;\r
+}\r
+\r
+// NT 系関数の解放\r
+void MsFreeNtApiFunctions(NT_API *nt)\r
+{\r
+       // 引数チェック\r
+       if (nt == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (nt->hSecur32 != NULL)\r
+       {\r
+               FreeLibrary(nt->hSecur32);\r
+       }\r
+\r
+       if (nt->hNewDev != NULL)\r
+       {\r
+               FreeLibrary(nt->hSetupApi);\r
+               FreeLibrary(nt->hNewDev);\r
+       }\r
+\r
+       FreeLibrary(nt->hAdvapi32);\r
+\r
+       FreeLibrary(nt->hShell32);\r
+\r
+       if (nt->hWtsApi32 != NULL)\r
+       {\r
+               FreeLibrary(nt->hWtsApi32);\r
+       }\r
+\r
+       if (nt->hPsApi != NULL)\r
+       {\r
+               FreeLibrary(nt->hPsApi);\r
+       }\r
+\r
+       if (nt->hUser32 != NULL)\r
+       {\r
+               FreeLibrary(nt->hUser32);\r
+       }\r
+\r
+       if (nt->hDbgHelp != NULL)\r
+       {\r
+               FreeLibrary(nt->hDbgHelp);\r
+       }\r
+\r
+       FreeLibrary(nt->hKernel32);\r
+\r
+       Free(nt);\r
+}\r
+\r
+// 64 bit アプリケーションのために 32 bit レジストリキーへのアクセスを強制するアクセスマスクを生成する\r
+DWORD MsRegAccessMaskFor64Bit(bool force32bit)\r
+{\r
+       return MsRegAccessMaskFor64BitEx(force32bit, false);\r
+}\r
+DWORD MsRegAccessMaskFor64BitEx(bool force32bit, bool force64bit)\r
+{\r
+       if (MsIs64BitWindows() == false)\r
+       {\r
+               return 0;\r
+       }\r
+       if (force32bit)\r
+       {\r
+               return KEY_WOW64_32KEY;\r
+       }\r
+       if (force64bit)\r
+       {\r
+               return KEY_WOW64_64KEY;\r
+       }\r
+\r
+       return 0;\r
+}\r
+\r
+// 値の削除\r
+bool MsRegDeleteValue(UINT root, char *keyname, char *valuename)\r
+{\r
+       return MsRegDeleteValueEx(root, keyname, valuename, false);\r
+}\r
+bool MsRegDeleteValueEx(UINT root, char *keyname, char *valuename, bool force32bit)\r
+{\r
+       return MsRegDeleteValueEx2(root, keyname, valuename, force32bit, false);\r
+}\r
+bool MsRegDeleteValueEx2(UINT root, char *keyname, char *valuename, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       bool ret;\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_ALL_ACCESS | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (RegDeleteValue(h, valuename) != ERROR_SUCCESS)\r
+       {\r
+               ret = false;\r
+       }\r
+       else\r
+       {\r
+               ret = true;\r
+       }\r
+\r
+       RegCloseKey(h);\r
+\r
+       return ret;\r
+}\r
+\r
+// キーの削除\r
+bool MsRegDeleteKey(UINT root, char *keyname)\r
+{\r
+       return MsRegDeleteKeyEx(root, keyname, false);\r
+}\r
+bool MsRegDeleteKeyEx(UINT root, char *keyname, bool force32bit)\r
+{\r
+       return MsRegDeleteKeyEx2(root, keyname, force32bit, false);\r
+}\r
+bool MsRegDeleteKeyEx2(UINT root, char *keyname, bool force32bit, bool force64bit)\r
+{\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (MsIsNt() && ms->nt->RegDeleteKeyExA != NULL)\r
+       {\r
+               if (ms->nt->RegDeleteKeyExA(MsGetRootKeyFromInt(root), keyname, MsRegAccessMaskFor64BitEx(force32bit, force64bit), 0) != ERROR_SUCCESS)\r
+               {\r
+                       return false;\r
+               }\r
+       }\r
+       else\r
+       {\r
+               if (RegDeleteKey(MsGetRootKeyFromInt(root), keyname) != ERROR_SUCCESS)\r
+               {\r
+                       return false;\r
+               }\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 値の列挙\r
+TOKEN_LIST *MsRegEnumValue(UINT root, char *keyname)\r
+{\r
+       return MsRegEnumValueEx(root, keyname, false);\r
+}\r
+TOKEN_LIST *MsRegEnumValueEx(UINT root, char *keyname, bool force32bit)\r
+{\r
+       return MsRegEnumValueEx2(root, keyname, force32bit, false);\r
+}\r
+TOKEN_LIST *MsRegEnumValueEx2(UINT root, char *keyname, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       UINT i;\r
+       TOKEN_LIST *t;\r
+       LIST *o;\r
+\r
+       if (keyname == NULL)\r
+       {\r
+               h = MsGetRootKeyFromInt(root);\r
+       }\r
+       else\r
+       {\r
+               if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_READ | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+               {\r
+                       return NULL;\r
+               }\r
+       }\r
+\r
+       o = NewListFast(CompareStr);\r
+\r
+       for (i = 0;;i++)\r
+       {\r
+               char tmp[MAX_SIZE];\r
+               UINT ret;\r
+               UINT size = sizeof(tmp);\r
+\r
+               Zero(tmp, sizeof(tmp));\r
+               ret = RegEnumValue(h, i, tmp, &size, NULL, NULL, NULL, NULL);\r
+               if (ret == ERROR_NO_MORE_ITEMS)\r
+               {\r
+                       break;\r
+               }\r
+               else if (ret != ERROR_SUCCESS)\r
+               {\r
+                       break;\r
+               }\r
+\r
+               Add(o, CopyStr(tmp));\r
+       }\r
+\r
+       Sort(o);\r
+\r
+       t = ZeroMalloc(sizeof(TOKEN_LIST));\r
+       t->NumTokens = LIST_NUM(o);\r
+       t->Token = ZeroMalloc(sizeof(char *) * t->NumTokens);\r
+       for (i = 0;i < t->NumTokens;i++)\r
+       {\r
+               t->Token[i] = LIST_DATA(o, i);\r
+       }\r
+\r
+       ReleaseList(o);\r
+\r
+       if (keyname != NULL)\r
+       {\r
+               RegCloseKey(h);\r
+       }\r
+\r
+       return t;\r
+}\r
+\r
+// キーの列挙\r
+TOKEN_LIST *MsRegEnumKey(UINT root, char *keyname)\r
+{\r
+       return MsRegEnumKeyEx(root, keyname, false);\r
+}\r
+TOKEN_LIST *MsRegEnumKeyEx(UINT root, char *keyname, bool force32bit)\r
+{\r
+       return MsRegEnumKeyEx2(root, keyname, force32bit, false);\r
+}\r
+TOKEN_LIST *MsRegEnumKeyEx2(UINT root, char *keyname, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       UINT i;\r
+       TOKEN_LIST *t;\r
+       LIST *o;\r
+\r
+       if (keyname == NULL)\r
+       {\r
+               h = MsGetRootKeyFromInt(root);\r
+       }\r
+       else\r
+       {\r
+               if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_READ | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+               {\r
+                       return NULL;\r
+               }\r
+       }\r
+\r
+       o = NewListFast(CompareStr);\r
+\r
+       for (i = 0;;i++)\r
+       {\r
+               char tmp[MAX_SIZE];\r
+               UINT ret;\r
+               UINT size = sizeof(tmp);\r
+               FILETIME ft;\r
+\r
+               Zero(tmp, sizeof(tmp));\r
+               ret = RegEnumKeyEx(h, i, tmp, &size, NULL, NULL, NULL, &ft);\r
+               if (ret == ERROR_NO_MORE_ITEMS)\r
+               {\r
+                       break;\r
+               }\r
+               else if (ret != ERROR_SUCCESS)\r
+               {\r
+                       break;\r
+               }\r
+\r
+               Add(o, CopyStr(tmp));\r
+       }\r
+\r
+       Sort(o);\r
+\r
+       t = ZeroMalloc(sizeof(TOKEN_LIST));\r
+       t->NumTokens = LIST_NUM(o);\r
+       t->Token = ZeroMalloc(sizeof(char *) * t->NumTokens);\r
+       for (i = 0;i < t->NumTokens;i++)\r
+       {\r
+               t->Token[i] = LIST_DATA(o, i);\r
+       }\r
+\r
+       ReleaseList(o);\r
+\r
+       if (keyname != NULL)\r
+       {\r
+               RegCloseKey(h);\r
+       }\r
+\r
+       return t;\r
+}\r
+\r
+// バイナリデータを設定する\r
+bool MsRegWriteBin(UINT root, char *keyname, char *valuename, void *data, UINT size)\r
+{\r
+       return MsRegWriteBinEx(root, keyname, valuename, data, size, false);\r
+}\r
+bool MsRegWriteBinEx(UINT root, char *keyname, char *valuename, void *data, UINT size, bool force32bit)\r
+{\r
+       return MsRegWriteBinEx2(root, keyname, valuename, data, size, force32bit, false);\r
+}\r
+bool MsRegWriteBinEx2(UINT root, char *keyname, char *valuename, void *data, UINT size, bool force32bit, bool force64bit)\r
+{\r
+       // 引数チェック\r
+       if (keyname == NULL || (size != 0 && data == NULL))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return MsRegWriteValueEx2(root, keyname, valuename, REG_BINARY, data, size, force32bit, force64bit);\r
+}\r
+\r
+// 整数値を設定する\r
+bool MsRegWriteInt(UINT root, char *keyname, char *valuename, UINT value)\r
+{\r
+       return MsRegWriteIntEx(root, keyname, valuename, value, false);\r
+}\r
+bool MsRegWriteIntEx(UINT root, char *keyname, char *valuename, UINT value, bool force32bit)\r
+{\r
+       return MsRegWriteIntEx2(root, keyname, valuename, value, force32bit, false);\r
+}\r
+bool MsRegWriteIntEx2(UINT root, char *keyname, char *valuename, UINT value, bool force32bit, bool force64bit)\r
+{\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // エンディアン補正\r
+       if (IsBigEndian())\r
+       {\r
+               value = Swap32(value);\r
+       }\r
+\r
+       return MsRegWriteValueEx2(root, keyname, valuename, REG_DWORD_LITTLE_ENDIAN, &value, sizeof(UINT), force32bit, force64bit);\r
+}\r
+\r
+// 文字列を設定する\r
+bool MsRegWriteStrExpand(UINT root, char *keyname, char *valuename, char *str)\r
+{\r
+       return MsRegWriteStrExpandEx(root, keyname, valuename, str, false);\r
+}\r
+bool MsRegWriteStrExpandEx(UINT root, char *keyname, char *valuename, char *str, bool force32bit)\r
+{\r
+       return MsRegWriteStrExpandEx2(root, keyname, valuename, str, force32bit, false);\r
+}\r
+bool MsRegWriteStrExpandEx2(UINT root, char *keyname, char *valuename, char *str, bool force32bit, bool force64bit)\r
+{\r
+       // 引数チェック\r
+       if (keyname == NULL || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return MsRegWriteValueEx2(root, keyname, valuename, REG_EXPAND_SZ, str, StrSize(str), force32bit, force64bit);\r
+}\r
+bool MsRegWriteStrExpandW(UINT root, char *keyname, char *valuename, wchar_t *str)\r
+{\r
+       return MsRegWriteStrExpandExW(root, keyname, valuename, str, false);\r
+}\r
+bool MsRegWriteStrExpandExW(UINT root, char *keyname, char *valuename, wchar_t *str, bool force32bit)\r
+{\r
+       return MsRegWriteStrExpandEx2W(root, keyname, valuename, str, force32bit, false);\r
+}\r
+bool MsRegWriteStrExpandEx2W(UINT root, char *keyname, char *valuename, wchar_t *str, bool force32bit, bool force64bit)\r
+{\r
+       // 引数チェック\r
+       if (keyname == NULL || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return MsRegWriteValueEx2W(root, keyname, valuename, REG_EXPAND_SZ, str, UniStrSize(str), force32bit, force64bit);\r
+}\r
+\r
+bool MsRegWriteStr(UINT root, char *keyname, char *valuename, char *str)\r
+{\r
+       return MsRegWriteStrEx(root, keyname, valuename, str, false);\r
+}\r
+bool MsRegWriteStrEx(UINT root, char *keyname, char *valuename, char *str, bool force32bit)\r
+{\r
+       return MsRegWriteStrEx2(root, keyname, valuename, str, force32bit, false);\r
+}\r
+bool MsRegWriteStrEx2(UINT root, char *keyname, char *valuename, char *str, bool force32bit, bool force64bit)\r
+{\r
+       // 引数チェック\r
+       if (keyname == NULL || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return MsRegWriteValueEx2(root, keyname, valuename, REG_SZ, str, StrSize(str), force32bit, force64bit);\r
+}\r
+bool MsRegWriteStrW(UINT root, char *keyname, char *valuename, wchar_t *str)\r
+{\r
+       return MsRegWriteStrExW(root, keyname, valuename, str, false);\r
+}\r
+bool MsRegWriteStrExW(UINT root, char *keyname, char *valuename, wchar_t *str, bool force32bit)\r
+{\r
+       return MsRegWriteStrEx2W(root, keyname, valuename, str, force32bit, false);\r
+}\r
+bool MsRegWriteStrEx2W(UINT root, char *keyname, char *valuename, wchar_t *str, bool force32bit, bool force64bit)\r
+{\r
+       // 引数チェック\r
+       if (keyname == NULL || str == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return MsRegWriteValueEx2W(root, keyname, valuename, REG_SZ, str, UniStrSize(str), force32bit, force64bit);\r
+}\r
+\r
+// 値を設定する\r
+bool MsRegWriteValue(UINT root, char *keyname, char *valuename, UINT type, void *data, UINT size)\r
+{\r
+       return MsRegWriteValueEx(root, keyname, valuename, type, data, size, false);\r
+}\r
+bool MsRegWriteValueEx(UINT root, char *keyname, char *valuename, UINT type, void *data, UINT size, bool force32bit)\r
+{\r
+       return MsRegWriteValueEx2(root, keyname, valuename, type, data, size, force32bit, false);\r
+}\r
+bool MsRegWriteValueEx2(UINT root, char *keyname, char *valuename, UINT type, void *data, UINT size, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       // 引数チェック\r
+       if (keyname == NULL || (size != 0 && data == NULL))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // キーを作成する\r
+       MsRegNewKeyEx2(root, keyname, force32bit, force64bit);\r
+\r
+       // キーを開く\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_ALL_ACCESS | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 値を書き込む\r
+       if (RegSetValueEx(h, valuename, 0, type, data, size) != ERROR_SUCCESS)\r
+       {\r
+               RegCloseKey(h);\r
+               return false;\r
+       }\r
+\r
+       // キーを閉じる\r
+       RegCloseKey(h);\r
+\r
+       return true;\r
+}\r
+bool MsRegWriteValueW(UINT root, char *keyname, char *valuename, UINT type, void *data, UINT size)\r
+{\r
+       return MsRegWriteValueExW(root, keyname, valuename, type, data, size, false);\r
+}\r
+bool MsRegWriteValueExW(UINT root, char *keyname, char *valuename, UINT type, void *data, UINT size, bool force32bit)\r
+{\r
+       return MsRegWriteValueEx2W(root, keyname, valuename, type, data, size, force32bit, false);\r
+}\r
+bool MsRegWriteValueEx2W(UINT root, char *keyname, char *valuename, UINT type, void *data, UINT size, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       wchar_t valuename_w[MAX_SIZE];\r
+       // 引数チェック\r
+       if (keyname == NULL || (size != 0 && data == NULL))\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               UINT size_a;\r
+               void *data_a;\r
+               bool ret;\r
+\r
+               if (type == REG_SZ || type == REG_MULTI_SZ || type == REG_EXPAND_SZ)\r
+               {\r
+                       data_a = CopyUniToStr(data);\r
+                       size_a = StrSize(data_a);\r
+               }\r
+               else\r
+               {\r
+                       data_a = Clone(data, size);\r
+                       size_a = size;\r
+               }\r
+\r
+               ret = MsRegWriteValueEx2(root, keyname, valuename, type, data_a, size_a, force32bit, force64bit);\r
+\r
+               Free(data_a);\r
+\r
+               return ret;\r
+       }\r
+\r
+       StrToUni(valuename_w, sizeof(valuename_w), valuename);\r
+\r
+       // キーを作成する\r
+       MsRegNewKeyEx2(root, keyname, force32bit, force64bit);\r
+\r
+       // キーを開く\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_ALL_ACCESS | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 値を書き込む\r
+       if (RegSetValueExW(h, valuename_w, 0, type, data, size) != ERROR_SUCCESS)\r
+       {\r
+               RegCloseKey(h);\r
+               return false;\r
+       }\r
+\r
+       // キーを閉じる\r
+       RegCloseKey(h);\r
+\r
+       return true;\r
+}\r
+\r
+// バイナリデータを取得する\r
+BUF *MsRegReadBin(UINT root, char *keyname, char *valuename)\r
+{\r
+       return MsRegReadBinEx(root, keyname, valuename, false);\r
+}\r
+BUF *MsRegReadBinEx(UINT root, char *keyname, char *valuename, bool force32bit)\r
+{\r
+       return MsRegReadBinEx2(root, keyname, valuename, force32bit, false);\r
+}\r
+BUF *MsRegReadBinEx2(UINT root, char *keyname, char *valuename, bool force32bit, bool force64bit)\r
+{\r
+       char *ret;\r
+       UINT type, size;\r
+       BUF *b;\r
+       // 引数チェック\r
+       if (keyname == NULL || valuename == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       // 値を読み込む\r
+       if (MsRegReadValueEx2(root, keyname, valuename, &ret, &type, &size, force32bit, force64bit) == false)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       b = NewBuf();\r
+\r
+       WriteBuf(b, ret, size);\r
+       SeekBuf(b, 0, 0);\r
+\r
+       Free(ret);\r
+\r
+       return b;\r
+}\r
+\r
+// 整数値を取得する\r
+UINT MsRegReadInt(UINT root, char *keyname, char *valuename)\r
+{\r
+       return MsRegReadIntEx(root, keyname, valuename, false);\r
+}\r
+UINT MsRegReadIntEx(UINT root, char *keyname, char *valuename, bool force32bit)\r
+{\r
+       return MsRegReadIntEx2(root, keyname, valuename, force32bit, false);\r
+}\r
+UINT MsRegReadIntEx2(UINT root, char *keyname, char *valuename, bool force32bit, bool force64bit)\r
+{\r
+       char *ret;\r
+       UINT type, size;\r
+       UINT value;\r
+       // 引数チェック\r
+       if (keyname == NULL || valuename == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       // 値を読み込む\r
+       if (MsRegReadValueEx2(root, keyname, valuename, &ret, &type, &size, force32bit, force64bit) == false)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       // 種類をチェックする\r
+       if (type != REG_DWORD_LITTLE_ENDIAN && type != REG_DWORD_BIG_ENDIAN)\r
+       {\r
+               // DWORD 以外である\r
+               Free(ret);\r
+               return 0;\r
+       }\r
+\r
+       // サイズをチェックする\r
+       if (size != sizeof(UINT))\r
+       {\r
+               Free(ret);\r
+               return 0;\r
+       }\r
+\r
+       Copy(&value, ret, sizeof(UINT));\r
+\r
+       Free(ret);\r
+\r
+       // エンディアン変換\r
+       if (IsLittleEndian())\r
+       {\r
+#ifdef REG_DWORD_BIG_ENDIAN\r
+               if (type == REG_DWORD_BIG_ENDIAN)\r
+               {\r
+                       value = Swap32(value);\r
+               }\r
+#endif // REG_DWORD_BIG_ENDIAN\r
+       }\r
+       else\r
+       {\r
+#ifdef REG_DWORD_LITTLE_ENDIAN_FLAG\r
+               if (type == REG_DWORD_LITTLE_ENDIAN_FLAG)\r
+               {\r
+                       value = Swap32(value);\r
+               }\r
+#endif // REG_DWORD_LITTLE_ENDIAN_FLAG\r
+       }\r
+\r
+       return value;\r
+}\r
+\r
+// 文字列リストを取得する\r
+LIST *MsRegReadStrList(UINT root, char *keyname, char *valuename)\r
+{\r
+       return MsRegReadStrListEx(root, keyname, valuename, false);\r
+}\r
+LIST *MsRegReadStrListEx(UINT root, char *keyname, char *valuename, bool force32bit)\r
+{\r
+       return MsRegReadStrListEx2(root, keyname, valuename, force32bit, false);\r
+}\r
+LIST *MsRegReadStrListEx2(UINT root, char *keyname, char *valuename, bool force32bit, bool force64bit)\r
+{\r
+       LIST *o;\r
+       char *ret;\r
+       UINT type, size;\r
+       // 引数チェック\r
+       if (keyname == NULL || valuename == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 値を読み込む\r
+       if (MsRegReadValueEx2(root, keyname, valuename, &ret, &type, &size, force32bit, force64bit) == false)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 種類をチェックする\r
+       if (type != REG_MULTI_SZ)\r
+       {\r
+               // 文字列リスト以外である\r
+               Free(ret);\r
+               return NULL;\r
+       }\r
+\r
+       if (size < 2)\r
+       {\r
+               // サイズ不正\r
+               Free(ret);\r
+               return NULL;\r
+       }\r
+\r
+       if (ret[size - 1] != 0)\r
+       {\r
+               // データ不正\r
+               Free(ret);\r
+               return NULL;\r
+       }\r
+\r
+       // リスト作成\r
+       o = StrToStrList(ret, size);\r
+\r
+       Free(ret);\r
+\r
+       return o;\r
+}\r
+\r
+// 文字列を取得する\r
+char *MsRegReadStr(UINT root, char *keyname, char *valuename)\r
+{\r
+       return MsRegReadStrEx(root, keyname, valuename, false);\r
+}\r
+char *MsRegReadStrEx(UINT root, char *keyname, char *valuename, bool force32bit)\r
+{\r
+       return MsRegReadStrEx2(root, keyname, valuename, force32bit, false);\r
+}\r
+char *MsRegReadStrEx2(UINT root, char *keyname, char *valuename, bool force32bit, bool force64bit)\r
+{\r
+       char *ret;\r
+       UINT type, size;\r
+       // 引数チェック\r
+       if (keyname == NULL || valuename == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 値を読み込む\r
+       if (MsRegReadValueEx2(root, keyname, valuename, &ret, &type, &size, force32bit, force64bit) == false)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 種類をチェックする\r
+       if (type != REG_SZ && type != REG_EXPAND_SZ && type != REG_MULTI_SZ)\r
+       {\r
+               // 文字列以外である\r
+               Free(ret);\r
+\r
+               if (type == REG_MULTI_SZ)\r
+               {\r
+                       // 文字列リストである\r
+                       LIST *o = MsRegReadStrList(root, keyname, valuename);\r
+                       if (o != NULL)\r
+                       {\r
+                               if (LIST_NUM(o) >= 1)\r
+                               {\r
+                                       ret = CopyStr(LIST_DATA(o, 0));\r
+                                       FreeStrList(o);\r
+                                       return ret;\r
+                               }\r
+                       }\r
+               }\r
+               return NULL;\r
+       }\r
+\r
+       if (size == 0)\r
+       {\r
+               // サイズ不正\r
+               Free(ret);\r
+\r
+               return CopyStr("");\r
+       }\r
+\r
+       if (ret[size - 1] != 0)\r
+       {\r
+               // データ不正\r
+               Free(ret);\r
+               return NULL;\r
+       }\r
+\r
+       return ret;\r
+}\r
+wchar_t *MsRegReadStrW(UINT root, char *keyname, char *valuename)\r
+{\r
+       return MsRegReadStrExW(root, keyname, valuename, false);\r
+}\r
+wchar_t *MsRegReadStrExW(UINT root, char *keyname, char *valuename, bool force32bit)\r
+{\r
+       return MsRegReadStrEx2W(root, keyname, valuename, force32bit, false);\r
+}\r
+wchar_t *MsRegReadStrEx2W(UINT root, char *keyname, char *valuename, bool force32bit, bool force64bit)\r
+{\r
+       wchar_t *ret;\r
+       UINT type, size;\r
+       // 引数チェック\r
+       if (keyname == NULL || valuename == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 値を読み込む\r
+       if (MsRegReadValueEx2W(root, keyname, valuename, &ret, &type, &size, force32bit, force64bit) == false)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       // 種類をチェックする\r
+       if (type != REG_SZ && type != REG_EXPAND_SZ)\r
+       {\r
+               // 文字列以外である\r
+               Free(ret);\r
+\r
+               return NULL;\r
+       }\r
+\r
+       if (ret[size / sizeof(wchar_t) - 1] != 0)\r
+       {\r
+               // データ不正\r
+               Free(ret);\r
+               return NULL;\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// 値を読み込む\r
+bool MsRegReadValue(UINT root, char *keyname, char *valuename, void **data, UINT *type, UINT *size)\r
+{\r
+       return MsRegReadValueEx(root, keyname, valuename, data, type, size, false);\r
+}\r
+bool MsRegReadValueEx(UINT root, char *keyname, char *valuename, void **data, UINT *type, UINT *size, bool force32bit)\r
+{\r
+       return MsRegReadValueEx2(root, keyname, valuename, data, type, size, force32bit, false);\r
+}\r
+bool MsRegReadValueEx2(UINT root, char *keyname, char *valuename, void **data, UINT *type, UINT *size, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       UINT ret;\r
+       // 引数チェック\r
+       if (keyname == NULL || data == NULL || type == NULL || size == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       *type = 0;\r
+       *size = 0;\r
+\r
+       // キーを開く\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_READ | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 値を開く\r
+       *data = ZeroMalloc(*size);\r
+       ret = RegQueryValueEx(h, valuename, 0, type, *data, size);\r
+\r
+       if (ret == ERROR_SUCCESS)\r
+       {\r
+               // 読み取り完了\r
+               RegCloseKey(h);\r
+               return true;\r
+       }\r
+\r
+       if (ret != ERROR_MORE_DATA)\r
+       {\r
+               // 変なエラーが発生した\r
+               Free(*data);\r
+               *data = NULL;\r
+               RegCloseKey(h);\r
+               return false;\r
+       }\r
+\r
+       // メモリを再確保してデータを取得\r
+       *data = ReAlloc(*data, *size);\r
+       ret = RegQueryValueEx(h, valuename, 0, type, *data, size);\r
+       if (ret != ERROR_SUCCESS)\r
+       {\r
+               // エラー発生\r
+               Free(*data);\r
+               *data = NULL;\r
+               RegCloseKey(h);\r
+       }\r
+\r
+       RegCloseKey(h);\r
+\r
+       return true;\r
+}\r
+bool MsRegReadValueW(UINT root, char *keyname, char *valuename, void **data, UINT *type, UINT *size)\r
+{\r
+       return MsRegReadValueExW(root, keyname, valuename, data, type, size, false);\r
+}\r
+bool MsRegReadValueExW(UINT root, char *keyname, char *valuename, void **data, UINT *type, UINT *size, bool force32bit)\r
+{\r
+       return MsRegReadValueEx2W(root, keyname, valuename, data, type, size, force32bit, false);\r
+}\r
+bool MsRegReadValueEx2W(UINT root, char *keyname, char *valuename, void **data, UINT *type, UINT *size, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       UINT ret;\r
+       wchar_t valuename_w[MAX_SIZE];\r
+       // 引数チェック\r
+       if (keyname == NULL || data == NULL || type == NULL || size == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       *type = 0;\r
+       *size = 0;\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               bool ret;\r
+               void *data_a = NULL;\r
+               UINT type_a = 0, size_a = 0;\r
+\r
+               ret = MsRegReadValueEx2(root, keyname, valuename, &data_a, &type_a, &size_a, force32bit, force64bit);\r
+\r
+               if (ret != false)\r
+               {\r
+                       if (type_a == REG_SZ || type_a == REG_MULTI_SZ || type_a == REG_EXPAND_SZ)\r
+                       {\r
+                               *data = CopyStrToUni(data_a);\r
+                               Free(data_a);\r
+\r
+                               size_a = UniStrSize(*data);\r
+                       }\r
+                       else\r
+                       {\r
+                               *data = data_a;\r
+                       }\r
+\r
+                       *type = type_a;\r
+                       *size = size_a;\r
+               }\r
+\r
+               return ret;\r
+       }\r
+\r
+       StrToUni(valuename_w, sizeof(valuename_w), valuename);\r
+\r
+       // キーを開く\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_READ | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 値を開く\r
+       *data = ZeroMalloc(*size);\r
+       ret = RegQueryValueExW(h, valuename_w, 0, type, *data, size);\r
+\r
+       if (ret == ERROR_SUCCESS)\r
+       {\r
+               // 読み取り完了\r
+               RegCloseKey(h);\r
+               return true;\r
+       }\r
+\r
+       if (ret != ERROR_MORE_DATA)\r
+       {\r
+               // 変なエラーが発生した\r
+               Free(*data);\r
+               *data = NULL;\r
+               RegCloseKey(h);\r
+               return false;\r
+       }\r
+\r
+       // メモリを再確保してデータを取得\r
+       *data = ReAlloc(*data, *size);\r
+       ret = RegQueryValueExW(h, valuename_w, 0, type, *data, size);\r
+       if (ret != ERROR_SUCCESS)\r
+       {\r
+               // エラー発生\r
+               Free(*data);\r
+               *data = NULL;\r
+               RegCloseKey(h);\r
+       }\r
+\r
+       RegCloseKey(h);\r
+\r
+       return true;\r
+}\r
+\r
+// 値の種類とサイズを取得する\r
+bool MsRegGetValueTypeAndSize(UINT root, char *keyname, char *valuename, UINT *type, UINT *size)\r
+{\r
+       return MsRegGetValueTypeAndSizeEx(root, keyname, valuename, type, size, false);\r
+}\r
+bool MsRegGetValueTypeAndSizeEx(UINT root, char *keyname, char *valuename, UINT *type, UINT *size, bool force32bit)\r
+{\r
+       return MsRegGetValueTypeAndSizeEx2(root, keyname, valuename, type, size, force32bit, false);\r
+}\r
+bool MsRegGetValueTypeAndSizeEx2(UINT root, char *keyname, char *valuename, UINT *type, UINT *size, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       UINT ret;\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (type != NULL)\r
+       {\r
+               *type = 0;\r
+       }\r
+       if (size != NULL)\r
+       {\r
+               *size = 0;\r
+       }\r
+\r
+       // キーを開く\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_READ | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 値を開く\r
+       ret = RegQueryValueEx(h, valuename, 0, type, NULL, size);\r
+\r
+       if (ret == ERROR_SUCCESS || ret == ERROR_MORE_DATA)\r
+       {\r
+               RegCloseKey(h);\r
+               return true;\r
+       }\r
+\r
+       RegCloseKey(h);\r
+\r
+       return false;\r
+}\r
+bool MsRegGetValueTypeAndSizeW(UINT root, char *keyname, char *valuename, UINT *type, UINT *size)\r
+{\r
+       return MsRegGetValueTypeAndSizeExW(root, keyname, valuename, type, size, false);\r
+}\r
+bool MsRegGetValueTypeAndSizeExW(UINT root, char *keyname, char *valuename, UINT *type, UINT *size, bool force32bit)\r
+{\r
+       return MsRegGetValueTypeAndSizeEx2W(root, keyname, valuename, type, size, force32bit, false);\r
+}\r
+bool MsRegGetValueTypeAndSizeEx2W(UINT root, char *keyname, char *valuename, UINT *type, UINT *size, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       UINT ret;\r
+       wchar_t valuename_w[MAX_SIZE];\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (type != NULL)\r
+       {\r
+               *type = 0;\r
+       }\r
+       if (size != NULL)\r
+       {\r
+               *size = 0;\r
+       }\r
+       if (IsNt() == false)\r
+       {\r
+               UINT type_a = 0;\r
+               UINT size_a = 0;\r
+\r
+               bool ret = MsRegGetValueTypeAndSizeEx2(root, keyname, valuename, &type_a, &size_a, force32bit, force64bit);\r
+\r
+               if (type_a == REG_SZ || type_a == REG_MULTI_SZ || type_a == REG_EXPAND_SZ)\r
+               {\r
+                       size_a = size_a * sizeof(wchar_t);\r
+               }\r
+\r
+               if (type != NULL)\r
+               {\r
+                       *type = type_a;\r
+               }\r
+\r
+               if (size != NULL)\r
+               {\r
+                       *size = size_a;\r
+               }\r
+\r
+               return ret;\r
+       }\r
+\r
+       StrToUni(valuename_w, sizeof(valuename_w), valuename);\r
+\r
+       // キーを開く\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_READ | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 値を開く\r
+       ret = RegQueryValueExW(h, valuename_w, 0, type, NULL, size);\r
+\r
+       if (ret == ERROR_SUCCESS || ret == ERROR_MORE_DATA)\r
+       {\r
+               RegCloseKey(h);\r
+               return true;\r
+       }\r
+\r
+       RegCloseKey(h);\r
+\r
+       return false;\r
+}\r
+\r
+// 指定された値がレジストリに存在するかどうか確認する\r
+bool MsRegIsValue(UINT root, char *keyname, char *valuename)\r
+{\r
+       return MsRegIsValueEx(root, keyname, valuename, false);\r
+}\r
+bool MsRegIsValueEx(UINT root, char *keyname, char *valuename, bool force32bit)\r
+{\r
+       return MsRegIsValueEx2(root, keyname, valuename, force32bit, false);\r
+}\r
+bool MsRegIsValueEx2(UINT root, char *keyname, char *valuename, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       UINT type, size;\r
+       UINT ret;\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // キーを開く\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), keyname, 0, KEY_READ | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 値を開く\r
+       size = 0;\r
+       ret = RegQueryValueEx(h, valuename, 0, &type, NULL, &size);\r
+\r
+       if (ret == ERROR_SUCCESS || ret == ERROR_MORE_DATA)\r
+       {\r
+               RegCloseKey(h);\r
+               return true;\r
+       }\r
+\r
+       RegCloseKey(h);\r
+\r
+       return false;\r
+}\r
+\r
+// レジストリにキーを作成する\r
+bool MsRegNewKey(UINT root, char *keyname)\r
+{\r
+       return MsRegNewKeyEx(root, keyname, false);\r
+}\r
+bool MsRegNewKeyEx(UINT root, char *keyname, bool force32bit)\r
+{\r
+       return MsRegNewKeyEx2(root, keyname, force32bit, false);\r
+}\r
+bool MsRegNewKeyEx2(UINT root, char *keyname, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       // 引数チェック\r
+       if (keyname == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // キーが存在するかどうか確認する\r
+       if (MsRegIsKeyEx2(root, keyname, force32bit, force64bit))\r
+       {\r
+               // すでに存在している\r
+               return true;\r
+       }\r
+\r
+       // キーを作成する\r
+       if (RegCreateKeyEx(MsGetRootKeyFromInt(root), keyname, 0, NULL, REG_OPTION_NON_VOLATILE,\r
+               KEY_ALL_ACCESS | MsRegAccessMaskFor64BitEx(force32bit, force64bit), NULL, &h, NULL) != ERROR_SUCCESS)\r
+       {\r
+               // 失敗\r
+               return false;\r
+       }\r
+\r
+       RegCloseKey(h);\r
+\r
+       return true;\r
+}\r
+\r
+// 指定されたキーがレジストリに存在するかどうか確認する\r
+bool MsRegIsKey(UINT root, char *name)\r
+{\r
+       return MsRegIsKeyEx(root, name, false);\r
+}\r
+bool MsRegIsKeyEx(UINT root, char *name, bool force32bit)\r
+{\r
+       return MsRegIsKeyEx2(root, name, force32bit, false);\r
+}\r
+bool MsRegIsKeyEx2(UINT root, char *name, bool force32bit, bool force64bit)\r
+{\r
+       HKEY h;\r
+       // 引数チェック\r
+       if (name == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (RegOpenKeyEx(MsGetRootKeyFromInt(root), name, 0, KEY_READ | MsRegAccessMaskFor64BitEx(force32bit, force64bit), &h) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       RegCloseKey(h);\r
+\r
+       return true;\r
+}\r
+\r
+// ルートキーハンドルの取得\r
+HKEY MsGetRootKeyFromInt(UINT root)\r
+{\r
+       switch (root)\r
+       {\r
+       case REG_CLASSES_ROOT:\r
+               return HKEY_CLASSES_ROOT;\r
+\r
+       case REG_LOCAL_MACHINE:\r
+               return HKEY_LOCAL_MACHINE;\r
+\r
+       case REG_CURRENT_USER:\r
+               return HKEY_CURRENT_USER;\r
+\r
+       case REG_USERS:\r
+               return HKEY_USERS;\r
+       }\r
+\r
+       return NULL;\r
+}\r
+\r
+// コマンドライン文字列から実行ファイル名の部分をカットする (Unicode 版)\r
+wchar_t *MsCutExeNameFromUniCommandLine(wchar_t *str)\r
+{\r
+       // 引数チェック\r
+       if (str == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (str[0] != L'\"')\r
+       {\r
+               UINT i = UniSearchStrEx(str, L" ", 0, true);\r
+               if (i == INFINITE)\r
+               {\r
+                       return str + UniStrLen(str);\r
+               }\r
+               else\r
+               {\r
+                       return str + i + 1;\r
+               }\r
+       }\r
+       else\r
+       {\r
+               str++;\r
+               while (true)\r
+               {\r
+                       if ((*str) == 0)\r
+                       {\r
+                               return str + UniStrLen(str);\r
+                       }\r
+                       if ((*str) == L'\"')\r
+                       {\r
+                               break;\r
+                       }\r
+                       str++;\r
+               }\r
+\r
+               while (true)\r
+               {\r
+                       if ((*str) == 0)\r
+                       {\r
+                               return str + UniStrLen(str);\r
+                       }\r
+                       if ((*str) == L' ')\r
+                       {\r
+                               return str + 1;\r
+                       }\r
+                       str++;\r
+               }\r
+       }\r
+}\r
+\r
+// コマンドライン文字列から実行ファイル名の部分をカットする\r
+char *MsCutExeNameFromCommandLine(char *str)\r
+{\r
+       // 引数チェック\r
+       if (str == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       if (str[0] != '\"')\r
+       {\r
+               UINT i = SearchStrEx(str, " ", 0, true);\r
+               if (i == INFINITE)\r
+               {\r
+                       return str + StrLen(str);\r
+               }\r
+               else\r
+               {\r
+                       return str + i + 1;\r
+               }\r
+       }\r
+       else\r
+       {\r
+               str++;\r
+               while (true)\r
+               {\r
+                       if ((*str) == 0)\r
+                       {\r
+                               return str + StrLen(str);\r
+                       }\r
+                       if ((*str) == '\"')\r
+                       {\r
+                               break;\r
+                       }\r
+                       str++;\r
+               }\r
+\r
+               while (true)\r
+               {\r
+                       if ((*str) == 0)\r
+                       {\r
+                               return str + StrLen(str);\r
+                       }\r
+                       if ((*str) == ' ')\r
+                       {\r
+                               return str + 1;\r
+                       }\r
+                       str++;\r
+               }\r
+       }\r
+}\r
+\r
+// プロセスハンドルの取得\r
+void *MsGetCurrentProcess()\r
+{\r
+       return ms->hCurrentProcess;\r
+}\r
+\r
+// プロセス ID の取得\r
+UINT MsGetCurrentProcessId()\r
+{\r
+       return ms->CurrentProcessId;\r
+}\r
+\r
+// EXE ファイル名の取得\r
+char *MsGetExeFileName()\r
+{\r
+       return ms == NULL ? "Unknown" : ms->ExeFileName;\r
+}\r
+\r
+// EXE ファイルが置いてあるディレクトリ名の取得\r
+char *MsGetExeDirName()\r
+{\r
+       return ms->ExeFileDir;\r
+}\r
+wchar_t *MsGetExeDirNameW()\r
+{\r
+       return ms->ExeFileDirW;\r
+}\r
+\r
+// 特殊なディレクトリ名の取得\r
+char *MsGetSpecialDir(int id)\r
+{\r
+       LPITEMIDLIST t = NULL;\r
+       char tmp[MAX_PATH];\r
+\r
+       if (SHGetSpecialFolderLocation(NULL, id, &t) != S_OK)\r
+       {\r
+               return CopyStr(ms->ExeFileDir);\r
+       }\r
+\r
+       if (SHGetPathFromIDList(t, tmp) == false)\r
+       {\r
+               return CopyStr(ms->ExeFileDir);\r
+       }\r
+\r
+       Win32NukuEn(tmp, sizeof(tmp), tmp);\r
+\r
+       return CopyStr(tmp);\r
+}\r
+wchar_t *MsGetSpecialDirW(int id)\r
+{\r
+       LPITEMIDLIST t = NULL;\r
+       wchar_t tmp[MAX_PATH];\r
+\r
+       if (IsNt() == false)\r
+       {\r
+               char *tmp = MsGetSpecialDir(id);\r
+               wchar_t *ret = CopyStrToUni(tmp);\r
+\r
+               Free(tmp);\r
+\r
+               return ret;\r
+       }\r
+\r
+       if (SHGetSpecialFolderLocation(NULL, id, &t) != S_OK)\r
+       {\r
+               return UniCopyStr(ms->ExeFileDirW);\r
+       }\r
+\r
+       if (SHGetPathFromIDListW(t, tmp) == false)\r
+       {\r
+               return UniCopyStr(ms->ExeFileDirW);\r
+       }\r
+\r
+       Win32NukuEnW(tmp, sizeof(tmp), tmp);\r
+\r
+       return UniCopyStr(tmp);\r
+}\r
+\r
+// 特殊なディレクトリをすべて取得する\r
+void MsGetSpecialDirs()\r
+{\r
+       char tmp[MAX_PATH];\r
+\r
+       // System32\r
+       GetSystemDirectory(tmp, sizeof(tmp));\r
+       Win32NukuEn(tmp, sizeof(tmp), tmp);\r
+       ms->System32Dir = CopyStr(tmp);\r
+       ms->System32DirW = CopyStrToUni(tmp);\r
+\r
+       // Windows ディレクトリは System32 ディレクトリの 1 つ上にある\r
+       Win32GetDirFromPath(tmp, sizeof(tmp), tmp);\r
+       Win32NukuEn(tmp, sizeof(tmp), tmp);\r
+       ms->WindowsDir = CopyStr(tmp);\r
+       ms->WindowsDirW = CopyStrToUni(tmp);\r
+\r
+       // Windows ディレクトリの下の Temp ディレクトリ\r
+       Format(tmp, sizeof(tmp), "%s\\Temp", ms->WindowsDir);\r
+       ms->WinTempDir = CopyStr(tmp);\r
+       ms->WinTempDirW = CopyStrToUni(tmp);\r
+       MsUniMakeDirEx(ms->WinTempDirW);\r
+\r
+       // システムドライブ\r
+       tmp[2] = 0;\r
+       ms->WindowsDrive = CopyStr(tmp);\r
+       ms->WindowsDriveW = CopyStrToUni(tmp);\r
+\r
+       // Temp\r
+       GetTempPath(MAX_PATH, tmp);\r
+       Win32NukuEn(tmp, sizeof(tmp), tmp);\r
+       ms->TempDir = CopyStr(tmp);\r
+\r
+       // Temp (Unicode) の取得\r
+       if (IsNt())\r
+       {\r
+               wchar_t tmp_w[MAX_PATH];\r
+\r
+               GetTempPathW(MAX_PATH, tmp_w);\r
+               Win32NukuEnW(tmp_w, sizeof(tmp_w), tmp_w);\r
+\r
+               ms->TempDirW = CopyUniStr(tmp_w);\r
+       }\r
+       else\r
+       {\r
+               ms->TempDirW = CopyStrToUni(tmp);\r
+       }\r
+       MakeDirExW(ms->TempDirW);\r
+       MakeDirEx(ms->TempDir);\r
+\r
+       // Program Files\r
+       ms->ProgramFilesDir = MsGetSpecialDir(CSIDL_PROGRAM_FILES);\r
+       if (StrCmpi(ms->ProgramFilesDir, ms->ExeFileDir) == 0)\r
+       {\r
+               char tmp[MAX_PATH];\r
+               Format(tmp, sizeof(tmp), "%s\\Program Files", ms->WindowsDrive);\r
+\r
+               Free(ms->ProgramFilesDir);\r
+               ms->ProgramFilesDir = CopyStr(tmp);\r
+       }\r
+\r
+       ms->ProgramFilesDirW = MsGetSpecialDirW(CSIDL_PROGRAM_FILES);\r
+       if (UniStrCmpi(ms->ProgramFilesDirW, ms->ExeFileDirW) == 0)\r
+       {\r
+               wchar_t tmp[MAX_PATH];\r
+               UniFormat(tmp, sizeof(tmp), L"%s\\Program Files", ms->WindowsDriveW);\r
+\r
+               Free(ms->ProgramFilesDirW);\r
+               ms->ProgramFilesDirW = UniCopyStr(tmp);\r
+       }\r
+\r
+       if (MsIsNt())\r
+       {\r
+               // 共通のスタートメニュー\r
+               ms->CommonStartMenuDir = MsGetSpecialDir(CSIDL_COMMON_STARTMENU);\r
+               ms->CommonStartMenuDirW = MsGetSpecialDirW(CSIDL_COMMON_STARTMENU);\r
+\r
+               // 共通のプログラム\r
+               ms->CommonProgramsDir = MsGetSpecialDir(CSIDL_COMMON_PROGRAMS);\r
+               ms->CommonProgramsDirW = MsGetSpecialDirW(CSIDL_COMMON_PROGRAMS);\r
+\r
+               // 共通のスタートアップ\r
+               ms->CommonStartupDir = MsGetSpecialDir(CSIDL_COMMON_STARTUP);\r
+               ms->CommonStartupDirW = MsGetSpecialDirW(CSIDL_COMMON_STARTUP);\r
+\r
+               // 共通のアプリケーションデータ\r
+               ms->CommonAppDataDir = MsGetSpecialDir(CSIDL_COMMON_APPDATA);\r
+               ms->CommonAppDataDirW = MsGetSpecialDirW(CSIDL_COMMON_APPDATA);\r
+\r
+               // 共通のデスクトップ\r
+               ms->CommonDesktopDir = MsGetSpecialDir(CSIDL_COMMON_DESKTOPDIRECTORY);\r
+               ms->CommonDesktopDirW = MsGetSpecialDirW(CSIDL_COMMON_DESKTOPDIRECTORY);\r
+\r
+               // Local Settings\r
+               ms->LocalAppDataDir = MsGetSpecialDir(CSIDL_LOCAL_APPDATA);\r
+               ms->LocalAppDataDirW = MsGetSpecialDirW(CSIDL_LOCAL_APPDATA);\r
+       }\r
+       else\r
+       {\r
+               // 個別のスタートメニュー\r
+               ms->PersonalStartMenuDir = MsGetSpecialDir(CSIDL_STARTMENU);\r
+               ms->CommonStartMenuDir = CopyStr(ms->PersonalStartMenuDir);\r
+               ms->PersonalStartMenuDirW = MsGetSpecialDirW(CSIDL_STARTMENU);\r
+               ms->CommonStartMenuDirW = CopyUniStr(ms->PersonalStartMenuDirW);\r
+\r
+               // 個別のプログラム\r
+               ms->PersonalProgramsDir = MsGetSpecialDir(CSIDL_PROGRAMS);\r
+               ms->CommonProgramsDir = CopyStr(ms->PersonalProgramsDir);\r
+               ms->PersonalProgramsDirW = MsGetSpecialDirW(CSIDL_PROGRAMS);\r
+               ms->CommonProgramsDirW = CopyUniStr(ms->PersonalProgramsDirW);\r
+\r
+               // 個別のスタートアップ\r
+               ms->PersonalStartupDir = MsGetSpecialDir(CSIDL_STARTUP);\r
+               ms->CommonStartupDir = CopyStr(ms->PersonalStartupDir);\r
+               ms->PersonalStartupDirW = MsGetSpecialDirW(CSIDL_STARTUP);\r
+               ms->CommonStartupDirW = CopyUniStr(ms->PersonalStartupDirW);\r
+\r
+               // 個別のアプリケーションデータ\r
+               ms->PersonalAppDataDir = MsGetSpecialDir(CSIDL_APPDATA);\r
+               ms->CommonAppDataDir = CopyStr(ms->PersonalAppDataDir);\r
+               ms->PersonalAppDataDirW = MsGetSpecialDirW(CSIDL_APPDATA);\r
+               ms->CommonAppDataDirW = CopyUniStr(ms->PersonalAppDataDirW);\r
+\r
+               // 個別のデスクトップ\r
+               ms->PersonalDesktopDir = MsGetSpecialDir(CSIDL_DESKTOP);\r
+               ms->CommonDesktopDir = CopyStr(ms->PersonalDesktopDir);\r
+               ms->PersonalDesktopDirW = MsGetSpecialDirW(CSIDL_DESKTOP);\r
+               ms->CommonDesktopDirW = CopyUniStr(ms->PersonalDesktopDirW);\r
+\r
+               // Local Settings\r
+               ms->LocalAppDataDir = CopyStr(ms->PersonalAppDataDir);\r
+               ms->LocalAppDataDirW = CopyUniStr(ms->PersonalAppDataDirW);\r
+       }\r
+}\r
+\r
+// 現在のユーザーが Administrators かどうかチェックする\r
+bool MsCheckIsAdmin()\r
+{\r
+       UCHAR test_bit[32];\r
+       UCHAR tmp[32];\r
+       char *name = "Vpn_Check_Admin_Key";\r
+       DWORD type;\r
+       DWORD size;\r
+       Rand(test_bit, sizeof(test_bit));\r
+\r
+       if (RegSetValueEx(HKEY_LOCAL_MACHINE, name, 0, REG_BINARY, test_bit, sizeof(test_bit)) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       size = sizeof(tmp);\r
+       if (RegQueryValueEx(HKEY_LOCAL_MACHINE, name, 0, &type, tmp, &size) != ERROR_SUCCESS)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       RegDeleteValue(HKEY_LOCAL_MACHINE, name);\r
+\r
+       if (Cmp(test_bit, tmp, 32) != 0)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// ライブラリの初期化\r
+void MsInit()\r
+{\r
+       char *str_ansi;\r
+       wchar_t *str_unicode;\r
+       OSVERSIONINFO os;\r
+       char tmp[MAX_SIZE];\r
+       UINT size;\r
+       if (ms != NULL)\r
+       {\r
+               // すでに初期化されている\r
+               return;\r
+       }\r
+\r
+       ms = ZeroMalloc(sizeof(MS));\r
+\r
+       // インスタンスハンドルの取得\r
+       ms->hInst = GetModuleHandleA(NULL);\r
+\r
+       // KERNEL32.DLL の取得\r
+       ms->hKernel32 = LoadLibrary("kernel32.dll");\r
+\r
+       // OS からコマンドライン文字列を取得する\r
+       str_ansi = CopyStr(GetCommandLineA());\r
+       Trim(str_ansi);\r
+       str_unicode = UniCopyStr(GetCommandLineW());\r
+       UniTrim(str_unicode);\r
+\r
+       SetCommandLineStr(MsCutExeNameFromCommandLine(str_ansi));\r
+       SetCommandLineUniStr(MsCutExeNameFromUniCommandLine(str_unicode));\r
+\r
+       Free(str_unicode);\r
+       Free(str_ansi);\r
+\r
+       // OS のバージョンを取得する\r
+       Zero(&os, sizeof(os));\r
+       os.dwOSVersionInfoSize = sizeof(os);\r
+       GetVersionEx(&os);\r
+\r
+       if (os.dwPlatformId == VER_PLATFORM_WIN32_NT)\r
+       {\r
+               // NT 系\r
+               ms->IsNt = true;\r
+\r
+               ms->nt = MsLoadNtApiFunctions();\r
+\r
+               if (ms->nt == NULL)\r
+               {\r
+                       ms->IsNt = false;\r
+                       ms->IsAdmin = true;\r
+               }\r
+               else\r
+               {\r
+                       // Administrators 判定\r
+                       ms->IsAdmin = MsCheckIsAdmin();\r
+               }\r
+       }\r
+       else\r
+       {\r
+               // 9x 系: 常に Administrators を偽装\r
+               ms->IsAdmin = true;\r
+       }\r
+\r
+       // 現在のプロセスに関する情報を取得する\r
+       ms->hCurrentProcess = GetCurrentProcess();\r
+       ms->CurrentProcessId = GetCurrentProcessId();\r
+\r
+       // EXE ファイル名を取得\r
+       GetModuleFileName(NULL, tmp, sizeof(tmp));\r
+       ms->ExeFileName = CopyStr(tmp);\r
+       Win32GetDirFromPath(tmp, sizeof(tmp), tmp);\r
+       ms->ExeFileDir = CopyStr(tmp);\r
+\r
+       // EXE ファイル名 (Unicode) を取得\r
+       if (IsNt())\r
+       {\r
+               wchar_t tmp_w[MAX_PATH];\r
+\r
+               GetModuleFileNameW(NULL, tmp_w, sizeof(tmp_w));\r
+               ms->ExeFileNameW = CopyUniStr(tmp_w);\r
+\r
+               Win32GetDirFromPathW(tmp_w, sizeof(tmp_w), tmp_w);\r
+               ms->ExeFileDirW = CopyUniStr(tmp_w);\r
+       }\r
+       else\r
+       {\r
+               ms->ExeFileNameW = CopyStrToUni(ms->ExeFileName);\r
+               ms->ExeFileDirW = CopyStrToUni(ms->ExeFileDir);\r
+       }\r
+\r
+       // 特殊なディレクトリを取得\r
+       MsGetSpecialDirs();\r
+\r
+       // 一時ディレクトリの初期化\r
+       MsInitTempDir();\r
+\r
+       // ユーザー名の取得\r
+       size = sizeof(tmp);\r
+       GetUserName(tmp, &size);\r
+       ms->UserName = CopyStr(tmp);\r
+\r
+       // ユーザー名の取得 (Unicode)\r
+       if (IsNt())\r
+       {\r
+               wchar_t tmp_w[MAX_PATH];\r
+\r
+               size = sizeof(tmp_w);\r
+\r
+               GetUserNameW(tmp_w, &size);\r
+               ms->UserNameW = CopyUniStr(tmp_w);\r
+       }\r
+       else\r
+       {\r
+               ms->UserNameW = CopyStrToUni(ms->UserName);\r
+       }\r
+\r
+       // フルユーザー名の取得\r
+       if (ms->nt != NULL && ms->nt->GetUserNameExA != NULL)\r
+       {\r
+               wchar_t tmp_w[MAX_PATH];\r
+\r
+               size = sizeof(tmp);\r
+               if (ms->nt->GetUserNameExA(NameSamCompatible, tmp, &size))\r
+               {\r
+                       ms->UserNameEx = CopyStr(tmp);\r
+               }\r
+\r
+               size = sizeof(tmp_w);\r
+               if (ms->nt->GetUserNameExW(NameSamCompatible, tmp_w, &size))\r
+               {\r
+                       ms->UserNameExW = CopyUniStr(tmp_w);\r
+               }\r
+       }\r
+\r
+       if (ms->UserNameEx == NULL)\r
+       {\r
+               ms->UserNameEx = CopyStr(ms->UserName);\r
+       }\r
+       if (ms->UserNameExW == NULL)\r
+       {\r
+               ms->UserNameExW = CopyUniStr(ms->UserNameW);\r
+       }\r
+\r
+       ms_critical_section = ZeroMalloc(sizeof(CRITICAL_SECTION));\r
+       InitializeCriticalSection(ms_critical_section);\r
+\r
+       // アダプタリストの初期化\r
+       MsInitAdapterListModule();\r
+\r
+       // minidump ベースファイル名の初期化\r
+       if (true)\r
+       {\r
+               wchar_t tmp[MAX_PATH];\r
+               if (MsIsAdmin())\r
+               {\r
+                       CombinePathW(tmp, sizeof(tmp), ms->ExeFileDirW, L"vpn_debug\\dump");\r
+               }\r
+               else\r
+               {\r
+                       CombinePathW(tmp, sizeof(tmp), ms->TempDirW, L"vpn_debug\\dump");\r
+               }\r
+               ms->MinidumpBaseFileNameW = CopyUniStr(tmp);\r
+       }\r
+\r
+       MsSetEnableMinidump(true);\r
+\r
+       if (MsIsNt())\r
+       {\r
+               if (ms->nt->MiniDumpWriteDump != NULL)\r
+               {\r
+                       SetUnhandledExceptionFilter(MsExceptionHandler);\r
+               }\r
+       }\r
+}\r
+\r
+// minidump を作成するかどうか選択する\r
+void MsSetEnableMinidump(bool enabled)\r
+{\r
+       ms->MiniDumpEnabled = enabled;\r
+}\r
+\r
+// minidump を出力する\r
+void MsWriteMinidump(wchar_t *filename, void *ex)\r
+{\r
+       wchar_t tmp[MAX_PATH];\r
+       wchar_t dir[MAX_PATH];\r
+       HANDLE h;\r
+       MINIDUMP_EXCEPTION_INFORMATION info;\r
+       struct _EXCEPTION_POINTERS *exp = (struct _EXCEPTION_POINTERS *)ex;\r
+\r
+       if (filename != NULL)\r
+       {\r
+               UniStrCpy(tmp, sizeof(tmp), filename);\r
+       }\r
+       else\r
+       {\r
+               SYSTEMTIME tm;\r
+\r
+               Zero(&tm, sizeof(tm));\r
+               GetLocalTime(&tm);\r
+\r
+               UniFormat(tmp, sizeof(tmp), L"%s_%04u%02u%02u_%02u%02u%02u.dmp",\r
+                       ms->MinidumpBaseFileNameW,\r
+                       tm.wYear, tm.wMonth, tm.wDay, tm.wHour, tm.wMinute, tm.wSecond);\r
+       }\r
+\r
+       GetDirNameFromFilePathW(dir, sizeof(dir), tmp);\r
+\r
+       CreateDirectoryW(dir, NULL);\r
+\r
+       Zero(&info, sizeof(info));\r
+\r
+       if (exp != NULL)\r
+       {\r
+               info.ThreadId = GetCurrentThreadId();\r
+               info.ExceptionPointers = exp;\r
+               info.ClientPointers = true;\r
+       }\r
+\r
+       h = CreateFileW(tmp, GENERIC_READ | GENERIC_WRITE,\r
+               FILE_SHARE_READ, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,\r
+               NULL);\r
+       if (h != INVALID_HANDLE_VALUE)\r
+       {\r
+               ms->nt->MiniDumpWriteDump(ms->hCurrentProcess, ms->CurrentProcessId,\r
+                       h,\r
+                       MiniDumpNormal | MiniDumpWithFullMemory | MiniDumpWithDataSegs |\r
+                       MiniDumpWithHandleData\r
+                       ,\r
+                       info.ThreadId == 0 ? NULL : &info, NULL, NULL);\r
+\r
+               FlushFileBuffers(h);\r
+               CloseHandle(h);\r
+       }\r
+}\r
+\r
+// 例外ハンドラ\r
+LONG CALLBACK MsExceptionHandler(struct _EXCEPTION_POINTERS *ExceptionInfo)\r
+{\r
+       if (ms->MiniDumpEnabled)\r
+       {\r
+               MsWriteMinidump(NULL, ExceptionInfo);\r
+       }\r
+\r
+       return EXCEPTION_CONTINUE_SEARCH;\r
+}\r
+\r
+// ライブラリの解放\r
+void MsFree()\r
+{\r
+       if (ms == NULL)\r
+       {\r
+               // 初期化されていない\r
+               return;\r
+       }\r
+\r
+       // アダプタリストの解放\r
+       MsFreeAdapterListModule();\r
+\r
+       // 一時ディレクトリの解放\r
+       MsFreeTempDir();\r
+\r
+       if (ms->IsNt)\r
+       {\r
+               // NT 系 API の解放\r
+               MsFreeNtApiFunctions(ms->nt);\r
+       }\r
+\r
+       // メモリ解放\r
+       // ANSI\r
+       Free(ms->WindowsDir);\r
+       Free(ms->System32Dir);\r
+       Free(ms->TempDir);\r
+       Free(ms->WinTempDir);\r
+       Free(ms->WindowsDrive);\r
+       Free(ms->ProgramFilesDir);\r
+       Free(ms->CommonStartMenuDir);\r
+       Free(ms->CommonProgramsDir);\r
+       Free(ms->CommonStartupDir);\r
+       Free(ms->CommonAppDataDir);\r
+       Free(ms->CommonDesktopDir);\r
+       Free(ms->PersonalStartMenuDir);\r
+       Free(ms->PersonalProgramsDir);\r
+       Free(ms->PersonalStartupDir);\r
+       Free(ms->PersonalAppDataDir);\r
+       Free(ms->PersonalDesktopDir);\r
+       Free(ms->MyDocumentsDir);\r
+       Free(ms->ExeFileDir);\r
+       Free(ms->ExeFileName);\r
+       Free(ms->UserName);\r
+       Free(ms->UserNameEx);\r
+       Free(ms->LocalAppDataDir);\r
+       // Unicode\r
+       Free(ms->WindowsDirW);\r
+       Free(ms->System32DirW);\r
+       Free(ms->TempDirW);\r
+       Free(ms->WinTempDirW);\r
+       Free(ms->WindowsDriveW);\r
+       Free(ms->ProgramFilesDirW);\r
+       Free(ms->CommonStartMenuDirW);\r
+       Free(ms->CommonProgramsDirW);\r
+       Free(ms->CommonStartupDirW);\r
+       Free(ms->CommonAppDataDirW);\r
+       Free(ms->CommonDesktopDirW);\r
+       Free(ms->PersonalStartMenuDirW);\r
+       Free(ms->PersonalProgramsDirW);\r
+       Free(ms->PersonalStartupDirW);\r
+       Free(ms->PersonalAppDataDirW);\r
+       Free(ms->PersonalDesktopDirW);\r
+       Free(ms->MyDocumentsDirW);\r
+       Free(ms->ExeFileDirW);\r
+       Free(ms->ExeFileNameW);\r
+       Free(ms->UserNameW);\r
+       Free(ms->UserNameExW);\r
+       Free(ms->LocalAppDataDirW);\r
+       Free(ms->MinidumpBaseFileNameW);\r
+       Free(ms);\r
+       ms = NULL;\r
+\r
+       Free(ms_critical_section);\r
+       ms_critical_section = NULL;\r
+}\r
+\r
+// ディレクトリ取得関係\r
+char *MsGetCommonAppDataDir()\r
+{\r
+       return ms->CommonAppDataDir;\r
+}\r
+char *MsGetLocalAppDataDir()\r
+{\r
+       return ms->LocalAppDataDir;\r
+}\r
+char *MsGetWindowsDir()\r
+{\r
+       return ms->WindowsDir;\r
+}\r
+wchar_t *MsGetWindowsDirW()\r
+{\r
+       return ms->WindowsDirW;\r
+}\r
+char *MsGetSystem32Dir()\r
+{\r
+       return ms->System32Dir;\r
+}\r
+char *MsGetTempDir()\r
+{\r
+       return ms->TempDir;\r
+}\r
+char *MsGetWindowsDrive()\r
+{\r
+       return ms->WindowsDrive;\r
+}\r
+char *MsGetProgramFilesDir()\r
+{\r
+       return ms->ProgramFilesDir;\r
+}\r
+char *MsGetCommonStartMenuDir()\r
+{\r
+       return ms->CommonStartMenuDir;\r
+}\r
+char *MsGetCommonProgramsDir()\r
+{\r
+       return ms->CommonProgramsDir;\r
+}\r
+char *MsGetCommonStartupDir()\r
+{\r
+       return ms->CommonStartupDir;\r
+}\r
+char *MsGetCommonDesktopDir()\r
+{\r
+       return ms->CommonDesktopDir;\r
+}\r
+char *MsGetPersonalStartMenuDir()\r
+{\r
+       if (ms->PersonalStartMenuDir == NULL)\r
+       {\r
+               ms->PersonalStartMenuDir = MsGetSpecialDir(CSIDL_STARTMENU);\r
+       }\r
+       return ms->PersonalStartMenuDir;\r
+}\r
+char *MsGetPersonalProgramsDir()\r
+{\r
+       if (ms->PersonalProgramsDir == NULL)\r
+       {\r
+               ms->PersonalProgramsDir = MsGetSpecialDir(CSIDL_PROGRAMS);\r
+       }\r
+       return ms->PersonalProgramsDir;\r
+}\r
+char *MsGetPersonalStartupDir()\r
+{\r
+       if (ms->PersonalStartupDir == NULL)\r
+       {\r
+               ms->PersonalStartupDir = MsGetSpecialDir(CSIDL_STARTUP);\r
+       }\r
+       return ms->PersonalStartupDir;\r
+}\r
+char *MsGetPersonalAppDataDir()\r
+{\r
+       if (ms->PersonalAppDataDir == NULL)\r
+       {\r
+               ms->PersonalAppDataDir = MsGetSpecialDir(CSIDL_APPDATA);\r
+       }\r
+       return ms->PersonalAppDataDir;\r
+}\r
+char *MsGetPersonalDesktopDir()\r
+{\r
+       if (ms->PersonalDesktopDir == NULL)\r
+       {\r
+               ms->PersonalDesktopDir = MsGetSpecialDir(CSIDL_DESKTOP);\r
+       }\r
+       return ms->PersonalDesktopDir;\r
+}\r
+char *MsGetMyDocumentsDir()\r
+{\r
+       if (ms->MyDocumentsDir == NULL)\r
+       {\r
+               ms->MyDocumentsDir = MsGetSpecialDir(CSIDL_PERSONAL);\r
+       }\r
+       return ms->MyDocumentsDir;\r
+}\r
+char *MsGetMyTempDir()\r
+{\r
+       return ms->MyTempDir;\r
+}\r
+char *MsGetUserName()\r
+{\r
+       return ms->UserName;\r
+}\r
+char *MsGetUserNameEx()\r
+{\r
+       return ms->UserNameEx;\r
+}\r
+char *MsGetWinTempDir()\r
+{\r
+       return ms->WinTempDir;\r
+}\r
+\r
+wchar_t *MsGetExeFileNameW()\r
+{\r
+       return ms == NULL ? L"Unknown" : ms->ExeFileNameW;\r
+}\r
+wchar_t *MsGetExeFileDirW()\r
+{\r
+       return ms->ExeFileDirW;\r
+}\r
+wchar_t *MsGetWindowDirW()\r
+{\r
+       return ms->WindowsDirW;\r
+}\r
+wchar_t *MsGetSystem32DirW()\r
+{\r
+       return ms->System32DirW;\r
+}\r
+wchar_t *MsGetTempDirW()\r
+{\r
+       return ms->TempDirW;\r
+}\r
+wchar_t *MsGetWindowsDriveW()\r
+{\r
+       return ms->WindowsDriveW;\r
+}\r
+wchar_t *MsGetProgramFilesDirW()\r
+{\r
+       return ms->ProgramFilesDirW;\r
+}\r
+wchar_t *MsGetCommonStartMenuDirW()\r
+{\r
+       return ms->CommonStartMenuDirW;\r
+}\r
+wchar_t *MsGetCommonProgramsDirW()\r
+{\r
+       return ms->CommonProgramsDirW;\r
+}\r
+wchar_t *MsGetCommonStartupDirW()\r
+{\r
+       return ms->CommonStartupDirW;\r
+}\r
+wchar_t *MsGetCommonAppDataDirW()\r
+{\r
+       return ms->CommonAppDataDirW;\r
+}\r
+wchar_t *MsGetCommonDesktopDirW()\r
+{\r
+       return ms->CommonDesktopDirW;\r
+}\r
+wchar_t *MsGetPersonalStartMenuDirW()\r
+{\r
+       if (ms->PersonalStartMenuDirW == NULL)\r
+       {\r
+               ms->PersonalStartMenuDirW = MsGetSpecialDirW(CSIDL_STARTMENU);\r
+       }\r
+\r
+       return ms->PersonalStartMenuDirW;\r
+}\r
+wchar_t *MsGetPersonalProgramsDirW()\r
+{\r
+       if (ms->PersonalProgramsDirW == NULL)\r
+       {\r
+               ms->PersonalProgramsDirW = MsGetSpecialDirW(CSIDL_PROGRAMS);\r
+       }\r
+\r
+       return ms->PersonalProgramsDirW;\r
+}\r
+wchar_t *MsGetPersonalStartupDirW()\r
+{\r
+       if (ms->PersonalStartupDirW == NULL)\r
+       {\r
+               ms->PersonalStartupDirW = MsGetSpecialDirW(CSIDL_STARTUP);\r
+       }\r
+\r
+       return ms->PersonalStartupDirW;\r
+}\r
+wchar_t *MsGetPersonalAppDataDirW()\r
+{\r
+       if (ms->PersonalAppDataDirW == NULL)\r
+       {\r
+               ms->PersonalAppDataDirW = MsGetSpecialDirW(CSIDL_APPDATA);\r
+       }\r
+\r
+       return ms->PersonalAppDataDirW;\r
+}\r
+wchar_t *MsGetPersonalDesktopDirW()\r
+{\r
+       if (ms->PersonalDesktopDirW == NULL)\r
+       {\r
+               ms->PersonalDesktopDirW = MsGetSpecialDirW(CSIDL_DESKTOP);\r
+       }\r
+\r
+       return ms->PersonalDesktopDirW;\r
+}\r
+wchar_t *MsGetMyDocumentsDirW()\r
+{\r
+       if (ms->MyDocumentsDirW == NULL)\r
+       {\r
+               ms->MyDocumentsDirW = MsGetSpecialDirW(CSIDL_PERSONAL);\r
+       }\r
+\r
+       return ms->MyDocumentsDirW;\r
+}\r
+wchar_t *MsGetLocalAppDataDirW()\r
+{\r
+       return ms->LocalAppDataDirW;\r
+}\r
+wchar_t *MsGetMyTempDirW()\r
+{\r
+       return ms->MyTempDirW;\r
+}\r
+wchar_t *MsGetUserNameW()\r
+{\r
+       return ms->UserNameW;\r
+}\r
+wchar_t *MsGetUserNameExW()\r
+{\r
+       return ms->UserNameExW;\r
+}\r
+wchar_t *MsGetWinTempDirW()\r
+{\r
+       return ms->WinTempDirW;\r
+}\r
+\r
+\r
+#endif // WIN32\r
+\r