* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Mayaqua / Secure.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Secure.c b/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Secure.c
new file mode 100644 (file)
index 0000000..91800a2
--- /dev/null
@@ -0,0 +1,2188 @@
+// 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
+// Secure.c\r
+// セキュリティトークン管理モジュール\r
+\r
+#define        SECURE_C\r
+#define        ENCRYPT_C\r
+\r
+#ifdef WIN32\r
+#include <windows.h>\r
+#endif // WIN32\r
+\r
+#include <stdio.h>\r
+#include <stdlib.h>\r
+#include <string.h>\r
+#include <wchar.h>\r
+#include <stdarg.h>\r
+#include <time.h>\r
+#include <errno.h>\r
+#include <openssl/ssl.h>\r
+#include <openssl/err.h>\r
+#include <openssl/rand.h>\r
+#include <openssl/engine.h>\r
+#include <openssl/bio.h>\r
+#include <openssl/x509.h>\r
+#include <openssl/pkcs7.h>\r
+#include <openssl/pkcs12.h>\r
+#include <openssl/rc4.h>\r
+#include <openssl/md5.h>\r
+#include <openssl/sha.h>\r
+#include <Mayaqua/Mayaqua.h>\r
+#include <Mayaqua/cryptoki.h>\r
+\r
+\r
+#define        MAX_OBJ                         1024            // ハードウェア内の最大オブジェクト数 (想定)\r
+\r
+#define        A_SIZE(a, i)            (a[(i)].ulValueLen)\r
+#define        A_SET(a, i, value, size)        (a[i].pValue = value;a[i].ulValueLen = size;)\r
+\r
+#ifdef OS_WIN32\r
+// Win32 用コード\r
+\r
+// Win32 用 DLL 読み込み\r
+HINSTANCE Win32SecureLoadLibraryEx(char *dllname, DWORD flags)\r
+{\r
+       char tmp1[MAX_PATH];\r
+       char tmp2[MAX_PATH];\r
+       char tmp3[MAX_PATH];\r
+       HINSTANCE h;\r
+       // 引数チェック\r
+       if (dllname == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       Format(tmp1, sizeof(tmp1), "%s\\%s", MsGetSystem32Dir(), dllname);\r
+       Format(tmp2, sizeof(tmp2), "%s\\JPKI\\%s", MsGetProgramFilesDir(), dllname);\r
+       Format(tmp3, sizeof(tmp3), "%s\\LGWAN\\%s", MsGetProgramFilesDir(), dllname);\r
+\r
+       h = LoadLibraryEx(dllname, NULL, flags);\r
+       if (h != NULL)\r
+       {\r
+               return h;\r
+       }\r
+\r
+       h = LoadLibraryEx(tmp1, NULL, flags);\r
+       if (h != NULL)\r
+       {\r
+               return h;\r
+       }\r
+\r
+       h = LoadLibraryEx(tmp2, NULL, flags);\r
+       if (h != NULL)\r
+       {\r
+               return h;\r
+       }\r
+\r
+       h = LoadLibraryEx(tmp3, NULL, flags);\r
+       if (h != NULL)\r
+       {\r
+               return h;\r
+       }\r
+\r
+       return NULL;\r
+}\r
+\r
+// 指定したデバイスがインストールされているか調査\r
+bool Win32IsDeviceSupported(SECURE_DEVICE *dev)\r
+{\r
+       HINSTANCE hInst;\r
+       // 引数チェック\r
+       if (dev == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // DLL が読み込み可能かチェック\r
+       hInst = Win32SecureLoadLibraryEx(dev->ModuleName, DONT_RESOLVE_DLL_REFERENCES);\r
+       if (hInst == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       FreeLibrary(hInst);\r
+\r
+       return true;\r
+}\r
+\r
+// デバイスモジュールの読み込み\r
+bool Win32LoadSecModule(SECURE *sec)\r
+{\r
+       SEC_DATA_WIN32 *w;\r
+       HINSTANCE hInst;\r
+       CK_FUNCTION_LIST_PTR api = NULL;\r
+       CK_RV (*get_function_list)(CK_FUNCTION_LIST_PTR_PTR);\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (sec->Dev->Id == 9)\r
+       {\r
+               char username[MAX_SIZE];\r
+               DWORD size;\r
+               // 住基ネットのデバイスドライバでは、Software\JPKI レジストリキーの内容を\r
+               // SYSTEM の HKLU でも持っていなければならないので、もし持っていない場合は\r
+               // 別のユーザーの値からコピーする\r
+//             if (MsRegIsValue(REG_CURRENT_USER, "Software\\JPKI", "Name") == false ||\r
+//                     MsRegIsValue(REG_CURRENT_USER, "Software\\JPKI", "RWType") == false)\r
+               size = sizeof(username);\r
+               GetUserName(username, &size);\r
+               if (StrCmpi(username, "System") == 0)\r
+               {\r
+                       TOKEN_LIST *t = MsRegEnumKey(REG_USERS, NULL);\r
+\r
+                       if (t != NULL)\r
+                       {\r
+                               UINT i;\r
+\r
+                               for (i = 0;i < t->NumTokens;i++)\r
+                               {\r
+                                       char tmp[MAX_PATH];\r
+\r
+                                       if (StrCmpi(t->Token[i], ".DEFAULT") != 0 && StrCmpi(t->Token[i], "S-1-5-18") != 0)\r
+                                       {\r
+                                               Format(tmp, sizeof(tmp), "%s\\Software\\JPKI", t->Token[i]);\r
+\r
+                                               if (MsRegIsValue(REG_USERS, tmp, "Name") && MsRegIsValue(REG_USERS, tmp, "RWType"))\r
+                                               {\r
+                                                       char *name = MsRegReadStr(REG_USERS, tmp, "Name");\r
+                                                       char *port = MsRegReadStr(REG_USERS, tmp, "Port");\r
+                                                       UINT type = MsRegReadInt(REG_USERS, tmp, "RWType");\r
+\r
+                                                       MsRegWriteStr(REG_CURRENT_USER, "Software\\JPKI", "Name", name);\r
+                                                       MsRegWriteStr(REG_CURRENT_USER, "Software\\JPKI", "Port", port);\r
+                                                       MsRegWriteInt(REG_CURRENT_USER, "Software\\JPKI", "RWType", type);\r
+\r
+                                                       Free(name);\r
+                                                       Free(port);\r
+                                                       break;\r
+                                               }\r
+                                       }\r
+                               }\r
+\r
+                               FreeToken(t);\r
+                       }\r
+               }\r
+       }\r
+\r
+       // ライブラリのロード\r
+       hInst = Win32SecureLoadLibraryEx(sec->Dev->ModuleName, 0);\r
+       if (hInst == NULL)\r
+       {\r
+               // 失敗\r
+               return false;\r
+       }\r
+\r
+       // API の取得\r
+       get_function_list = (CK_RV (*)(CK_FUNCTION_LIST_PTR_PTR))\r
+               GetProcAddress(hInst, "C_GetFunctionList");\r
+\r
+       if (get_function_list == NULL)\r
+       {\r
+               // 失敗\r
+               FreeLibrary(hInst);\r
+               return false;\r
+       }\r
+\r
+       get_function_list(&api);\r
+       if (api == NULL)\r
+       {\r
+               // 失敗\r
+               FreeLibrary(hInst);\r
+               return false;\r
+       }\r
+\r
+       sec->Data = ZeroMalloc(sizeof(SEC_DATA_WIN32));\r
+       w = sec->Data;\r
+\r
+       w->hInst = hInst;\r
+       sec->Api = api;\r
+\r
+       return true;\r
+}\r
+\r
+// デバイスモジュールのアンロード\r
+void Win32FreeSecModule(SECURE *sec)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (sec->Data == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // アンロード\r
+       FreeLibrary(sec->Data->hInst);\r
+       Free(sec->Data);\r
+\r
+       sec->Data = NULL;\r
+}\r
+\r
+#endif // OS_WIN32\r
+\r
+\r
+// 指定されたデバイスが JPKI かどうか\r
+bool IsJPKI(bool id)\r
+{\r
+       if (id == 9 || id == 13)\r
+       {\r
+               return true;\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// セキュアデバイスの秘密鍵を名前を指定して署名\r
+bool SignSec(SECURE *sec, char *name, void *dst, void *src, UINT size)\r
+{\r
+       SEC_OBJ *obj;\r
+       UINT ret;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (name == NULL || dst == NULL || src == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+\r
+       obj = FindSecObject(sec, name, SEC_K);\r
+       if (obj == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       ret = SignSecByObject(sec, obj, dst, src, size);\r
+\r
+       FreeSecObject(obj);\r
+\r
+       return ret;\r
+}\r
+\r
+// セキュアデバイスの秘密鍵で署名\r
+bool SignSecByObject(SECURE *sec, SEC_OBJ *obj, void *dst, void *src, UINT size)\r
+{\r
+       CK_MECHANISM mechanism = {CKM_RSA_PKCS, NULL, 0};\r
+       UINT ret;\r
+       UCHAR hash[SIGN_HASH_SIZE];\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (obj == NULL || dst == NULL || src == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+       if (sec->LoginFlag == false && obj->Private)\r
+       {\r
+               sec->Error = SEC_ERROR_NOT_LOGIN;\r
+               return false;\r
+       }\r
+       if (obj->Type != SEC_K)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+\r
+       // ハッシュ\r
+       HashForSign(hash, sizeof(hash), src, size);\r
+\r
+       // 署名初期化\r
+       ret = sec->Api->C_SignInit(sec->SessionId, &mechanism, obj->Object);\r
+       if (ret != CKR_OK)\r
+       {\r
+               // 失敗\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               Debug("C_SignInit Error: 0x%x\n", ret);\r
+               return false;\r
+       }\r
+\r
+       // 署名実行\r
+       size = 128;\r
+       ret = sec->Api->C_Sign(sec->SessionId, hash, sizeof(hash), dst, &size);\r
+       if (ret != CKR_OK || size != 128)\r
+       {\r
+               // 失敗\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               Debug("C_Sign Error: 0x%x\n", ret);\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// PIN コードの変更\r
+bool ChangePin(SECURE *sec, char *old_pin, char *new_pin)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL || old_pin == NULL || new_pin == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+       if (sec->LoginFlag == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NOT_LOGIN;\r
+               return false;\r
+       }\r
+       if (sec->IsReadOnly)\r
+       {\r
+               sec->Error = SEC_ERROR_OPEN_SESSION;\r
+               return false;\r
+       }\r
+\r
+       // PIN 変更\r
+       if (sec->Api->C_SetPIN(sec->SessionId, old_pin, StrLen(old_pin),\r
+               new_pin, StrLen(new_pin)) != CKR_OK)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       return true;\r
+}\r
+\r
+// 秘密鍵オブジェクトの書き込み\r
+bool WriteSecKey(SECURE *sec, bool private_obj, char *name, K *k)\r
+{\r
+       UINT key_type = CKK_RSA;\r
+       CK_BBOOL b_true = true, b_false = false, b_private_obj = private_obj;\r
+       UINT obj_class = CKO_PRIVATE_KEY;\r
+       UINT object;\r
+       UINT ret;\r
+       BUF *b;\r
+       RSA *rsa;\r
+       UCHAR modules[MAX_SIZE], pub[MAX_SIZE], pri[MAX_SIZE], prime1[MAX_SIZE], prime2[MAX_SIZE];\r
+       CK_ATTRIBUTE a[] =\r
+       {\r
+               {CKA_MODULUS,                   modules,                0},             // 0\r
+               {CKA_PUBLIC_EXPONENT,   pub,                    0},             // 1\r
+               {CKA_PRIVATE_EXPONENT,  pri,                    0},             // 2\r
+               {CKA_PRIME_1,                   prime1,                 0},             // 3\r
+               {CKA_PRIME_2,                   prime2,                 0},             // 4\r
+               {CKA_CLASS,                             &obj_class,             sizeof(obj_class)},\r
+               {CKA_TOKEN,                             &b_true,                sizeof(b_true)},\r
+               {CKA_PRIVATE,                   &b_private_obj, sizeof(b_private_obj)},\r
+               {CKA_LABEL,                             name,                   StrLen(name)},\r
+               {CKA_KEY_TYPE,                  &key_type,              sizeof(key_type)},\r
+               {CKA_DERIVE,                    &b_false,               sizeof(b_false)},\r
+               {CKA_SUBJECT,                   name,                   StrLen(name)},\r
+               {CKA_SENSITIVE,                 &b_true,                sizeof(b_true)},\r
+               {CKA_DECRYPT,                   &b_true,                sizeof(b_true)},\r
+               {CKA_SIGN,                              &b_true,                sizeof(b_true)},\r
+               {CKA_SIGN_RECOVER,              &b_false,               sizeof(b_false)},\r
+               {CKA_EXTRACTABLE,               &b_false,               sizeof(b_false)},\r
+               {CKA_MODIFIABLE,                &b_false,               sizeof(b_false)},\r
+       };\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (name == NULL || k == NULL || k->private_key == false)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+       if (sec->LoginFlag == false && private_obj)\r
+       {\r
+               sec->Error = SEC_ERROR_NOT_LOGIN;\r
+               return false;\r
+       }\r
+\r
+       // 数値データ生成\r
+       rsa = k->pkey->pkey.rsa;\r
+       if (rsa == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       b = BigNumToBuf(rsa->n);\r
+       ReadBuf(b, modules, sizeof(modules));\r
+       A_SIZE(a, 0) = b->Size;\r
+       FreeBuf(b);\r
+\r
+       b = BigNumToBuf(rsa->e);\r
+       ReadBuf(b, pub, sizeof(pub));\r
+       A_SIZE(a, 1) = b->Size;\r
+       FreeBuf(b);\r
+\r
+       b = BigNumToBuf(rsa->d);\r
+       ReadBuf(b, pri, sizeof(pri));\r
+       A_SIZE(a, 2) = b->Size;\r
+       FreeBuf(b);\r
+\r
+       b = BigNumToBuf(rsa->p);\r
+       ReadBuf(b, prime1, sizeof(prime1));\r
+       A_SIZE(a, 3) = b->Size;\r
+       FreeBuf(b);\r
+\r
+       b = BigNumToBuf(rsa->q);\r
+       ReadBuf(b, prime2, sizeof(prime2));\r
+       A_SIZE(a, 4) = b->Size;\r
+       FreeBuf(b);\r
+\r
+       // 古い鍵があれば削除\r
+       if (CheckSecObject(sec, name, SEC_K))\r
+       {\r
+               DeleteSecKey(sec, name);\r
+       }\r
+\r
+       // 作成\r
+       if ((ret = sec->Api->C_CreateObject(sec->SessionId, a, sizeof(a) / sizeof(a[0]), &object)) != CKR_OK)\r
+       {\r
+               // 失敗\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               Debug("ret: 0x%x\n", ret);\r
+               return false;\r
+       }\r
+\r
+       // キャッシュ消去\r
+       EraseEnumSecObjectCache(sec);\r
+\r
+       return true;\r
+}\r
+\r
+// 証明書オブジェクトを名前を指定して読み込み\r
+X *ReadSecCert(SECURE *sec, char *name)\r
+{\r
+       SEC_OBJ *obj;\r
+       X *x;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+\r
+       // 検索\r
+       obj = FindSecObject(sec, name, SEC_X);\r
+       if (obj == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       // 取得\r
+       x = ReadSecCertFromObject(sec, obj);\r
+\r
+       FreeSecObject(obj);\r
+\r
+       return x;\r
+}\r
+\r
+// 証明書オブジェクトの読み込み\r
+X *ReadSecCertFromObject(SECURE *sec, SEC_OBJ *obj)\r
+{\r
+       UINT size;\r
+       X *x;\r
+       UCHAR value[4096];\r
+       BUF *b;\r
+       CK_ATTRIBUTE get[] =\r
+       {\r
+               {CKA_VALUE,             value,          sizeof(value)},\r
+       };\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+       if (sec->LoginFlag == false && obj->Private)\r
+       {\r
+               sec->Error = SEC_ERROR_NOT_LOGIN;\r
+               return false;\r
+       }\r
+       if (obj->Type != SEC_X)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+\r
+       // 取得\r
+       if (sec->Api->C_GetAttributeValue(\r
+               sec->SessionId, obj->Object, get, sizeof(get) / sizeof(get[0])) != CKR_OK)\r
+       {\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               return 0;\r
+       }\r
+\r
+       size = A_SIZE(get, 0);\r
+\r
+       // 変換\r
+       b = NewBuf();\r
+       WriteBuf(b, value, size);\r
+       SeekBuf(b, 0, 0);\r
+\r
+       x = BufToX(b, false);\r
+       if (x == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_INVALID_CERT;\r
+       }\r
+\r
+       FreeBuf(b);\r
+\r
+       return x;\r
+}\r
+\r
+// 証明書オブジェクトの書き込み\r
+bool WriteSecCert(SECURE *sec, bool private_obj, char *name, X *x)\r
+{\r
+       UINT obj_class = CKO_CERTIFICATE;\r
+       CK_BBOOL b_true = true, b_false = false, b_private_obj = private_obj;\r
+       UINT cert_type = CKC_X_509;\r
+       CK_DATE start_date, end_date;\r
+       UCHAR subject[MAX_SIZE];\r
+       UCHAR issuer[MAX_SIZE];\r
+       wchar_t w_subject[MAX_SIZE];\r
+       wchar_t w_issuer[MAX_SIZE];\r
+       UCHAR serial_number[MAX_SIZE];\r
+       UCHAR value[4096];\r
+       UINT ret;\r
+       BUF *b;\r
+       UINT object;\r
+       CK_ATTRIBUTE a[] =\r
+       {\r
+               {CKA_SUBJECT,                   subject,                0},                     // 0\r
+               {CKA_ISSUER,                    issuer,                 0},                     // 1\r
+               {CKA_SERIAL_NUMBER,             serial_number,  0},                     // 2\r
+               {CKA_VALUE,                             value,                  0},                     // 3\r
+               {CKA_CLASS,                             &obj_class,             sizeof(obj_class)},\r
+               {CKA_TOKEN,                             &b_true,                sizeof(b_true)},\r
+               {CKA_PRIVATE,                   &b_private_obj, sizeof(b_private_obj)},\r
+               {CKA_LABEL,                             name,                   StrLen(name)},\r
+               {CKA_CERTIFICATE_TYPE,  &cert_type,             sizeof(cert_type)},\r
+#if    0               // 失敗するトークンがあるのでこれは使わない\r
+               {CKA_START_DATE,                &start_date,    sizeof(start_date)},\r
+               {CKA_END_DATE,                  &end_date,              sizeof(end_date)},\r
+#endif\r
+       };\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (name == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+       if (sec->LoginFlag == false && private_obj)\r
+       {\r
+               sec->Error = SEC_ERROR_NOT_LOGIN;\r
+               return false;\r
+       }\r
+\r
+       // 証明書をバッファにコピー\r
+       b = XToBuf(x, false);\r
+       if (b == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_INVALID_CERT;\r
+               return false;\r
+       }\r
+       if (b->Size > sizeof(value))\r
+       {\r
+               // サイズが大きすぎる\r
+               FreeBuf(b);\r
+               sec->Error = SEC_ERROR_DATA_TOO_BIG;\r
+               return false;\r
+       }\r
+       Copy(value, b->Buf, b->Size);\r
+       A_SIZE(a, 3) = b->Size;\r
+       FreeBuf(b);\r
+\r
+       // Subject と Issuer を UTF-8 にエンコードして格納\r
+       GetPrintNameFromName(w_subject, sizeof(w_subject), x->subject_name);\r
+       UniToUtf8(subject, sizeof(subject), w_subject);\r
+       A_SIZE(a, 0) = StrLen(subject);\r
+       if (x->root_cert == false)\r
+       {\r
+               GetPrintNameFromName(w_issuer, sizeof(w_issuer), x->issuer_name);\r
+               UniToUtf8(issuer, sizeof(issuer), w_issuer);\r
+               A_SIZE(a, 1) = StrLen(issuer);\r
+       }\r
+\r
+       // シリアル番号をコピー\r
+       Copy(serial_number, x->serial->data, MIN(x->serial->size, sizeof(serial_number)));\r
+       A_SIZE(a, 2) = MIN(x->serial->size, sizeof(serial_number));\r
+\r
+       // 有効期限情報\r
+       UINT64ToCkDate(&start_date, SystemToLocal64(x->notBefore));\r
+       UINT64ToCkDate(&end_date, SystemToLocal64(x->notAfter));\r
+\r
+       // 同一の名前のオブジェクトがあれば削除\r
+       if (CheckSecObject(sec, name, SEC_X))\r
+       {\r
+               DeleteSecCert(sec, name);\r
+       }\r
+\r
+       // 作成\r
+       if ((ret = sec->Api->C_CreateObject(sec->SessionId, a, sizeof(a) / sizeof(a[0]), &object)) != CKR_OK)\r
+       {\r
+               // 失敗\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               Debug("Error: 0x%02x\n", ret);\r
+               return false;\r
+       }\r
+\r
+       // キャッシュ消去\r
+       EraseEnumSecObjectCache(sec);\r
+\r
+       return true;\r
+}\r
+\r
+// 秘密鍵オブジェクトの削除\r
+bool DeleteSecKey(SECURE *sec, char *name)\r
+{\r
+       return DeleteSecObjectByName(sec, name, SEC_K);\r
+}\r
+\r
+// 証明書オブジェクトの削除\r
+bool DeleteSecCert(SECURE *sec, char *name)\r
+{\r
+       return DeleteSecObjectByName(sec, name, SEC_X);\r
+}\r
+\r
+// CK_DATE を 64 bit 時刻に変換\r
+UINT64 CkDateToUINT64(struct CK_DATE *ck_date)\r
+{\r
+       SYSTEMTIME st;\r
+       char year[32], month[32], day[32];\r
+       // 引数チェック\r
+       if (ck_date == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+\r
+       Zero(year, sizeof(year));\r
+       Zero(month, sizeof(month));\r
+       Zero(day, sizeof(day));\r
+\r
+       Copy(year, ck_date->year, 4);\r
+       Copy(month, ck_date->month, 2);\r
+       Copy(day, ck_date->day, 2);\r
+\r
+       st.wYear = ToInt(year);\r
+       st.wMonth = ToInt(month);\r
+       st.wDay = ToInt(day);\r
+\r
+       return SystemToUINT64(&st);\r
+}\r
+\r
+// 64 bit 時刻を CK_DATE に変換\r
+void UINT64ToCkDate(void *p_ck_date, UINT64 time64)\r
+{\r
+       SYSTEMTIME st;\r
+       char year[32], month[32], day[32];\r
+       struct CK_DATE *ck_date = (CK_DATE *)p_ck_date;\r
+       // 引数チェック\r
+       if (ck_date == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       UINT64ToSystem(&st, time64);\r
+\r
+       Format(year, sizeof(year), "%04u", st.wYear);\r
+       Format(month, sizeof(month), "%04u", st.wMonth);\r
+       Format(day, sizeof(day), "%04u", st.wDay);\r
+\r
+       Zero(ck_date, sizeof(CK_DATE));\r
+\r
+       Copy(ck_date->year, year, 4);\r
+       Copy(ck_date->month, month, 2);\r
+       Copy(ck_date->day, day, 2);\r
+}\r
+\r
+// オブジェクトを名前で指定して削除\r
+bool DeleteSecObjectByName(SECURE *sec, char *name, UINT type)\r
+{\r
+       bool ret;\r
+       SEC_OBJ *obj;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (name == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+\r
+       // オブジェクト取得\r
+       obj = FindSecObject(sec, name, type);\r
+       if (obj == NULL)\r
+       {\r
+               // 失敗\r
+               return false;\r
+       }\r
+\r
+       // オブジェクト削除\r
+       ret = DeleteSecObject(sec, obj);\r
+\r
+       // メモリ解放\r
+       FreeSecObject(obj);\r
+\r
+       return ret;\r
+}\r
+\r
+// データの削除\r
+bool DeleteSecData(SECURE *sec, char *name)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (name == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+\r
+       return DeleteSecObjectByName(sec, name, SEC_DATA);\r
+}\r
+\r
+// セキュアオブジェクトの削除\r
+bool DeleteSecObject(SECURE *sec, SEC_OBJ *obj)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (obj == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+       if (sec->LoginFlag == false && obj->Private)\r
+       {\r
+               sec->Error = SEC_ERROR_NOT_LOGIN;\r
+               return false;\r
+       }\r
+\r
+       // オブジェクト消去\r
+       if (sec->Api->C_DestroyObject(sec->SessionId, obj->Object) != CKR_OK)\r
+       {\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               return false;\r
+       }\r
+\r
+       // キャッシュ消去\r
+       DeleteSecObjFromEnumCache(sec, obj->Name, obj->Type);\r
+\r
+       return true;\r
+}\r
+\r
+// キャッシュから指定した名前のオブジェクトを削除する\r
+void DeleteSecObjFromEnumCache(SECURE *sec, char *name, UINT type)\r
+{\r
+       UINT i;\r
+       // 引数チェック\r
+       if (sec == NULL || name == NULL || sec->EnumCache == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(sec->EnumCache);i++)\r
+       {\r
+               SEC_OBJ *obj = LIST_DATA(sec->EnumCache, i);\r
+\r
+               if (StrCmpi(obj->Name, name) == 0)\r
+               {\r
+                       if (obj->Type == type)\r
+                       {\r
+                               Delete(sec->EnumCache, obj);\r
+                               FreeSecObject(obj);\r
+                               break;\r
+                       }\r
+               }\r
+       }\r
+}\r
+\r
+// セキュアオブジェクトを名前で検索して読み込む\r
+int ReadSecData(SECURE *sec, char *name, void *data, UINT size)\r
+{\r
+       UINT ret = 0;\r
+       SEC_OBJ *obj;\r
+       // 引数チェック\r
+       if (sec == NULL || name == NULL || data == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return 0;\r
+       }\r
+\r
+       // 読み込み\r
+       obj = FindSecObject(sec, name, SEC_DATA);\r
+       if (obj == NULL)\r
+       {\r
+               // 見つからない\r
+               return 0;\r
+       }\r
+\r
+       // 読み込む\r
+       ret = ReadSecDataFromObject(sec, obj, data, size);\r
+\r
+       FreeSecObject(obj);\r
+\r
+       return ret;\r
+}\r
+\r
+// キャッシュ消去\r
+void EraseEnumSecObjectCache(SECURE *sec)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL || sec->EnumCache == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       FreeEnumSecObject(sec->EnumCache);\r
+       sec->EnumCache = NULL;\r
+}\r
+\r
+// セキュアオブジェクトの存在をチェックする\r
+bool CheckSecObject(SECURE *sec, char *name, UINT type)\r
+{\r
+       SEC_OBJ *obj;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (name == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return 0;\r
+       }\r
+\r
+       obj = FindSecObject(sec, name, type);\r
+\r
+       if (obj == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       else\r
+       {\r
+               FreeSecObject(obj);\r
+               return true;\r
+       }\r
+}\r
+\r
+// セキュアオブジェクト構造体のクローンの作成\r
+SEC_OBJ *CloneSecObject(SEC_OBJ *obj)\r
+{\r
+       SEC_OBJ *ret;\r
+       // 引数チェック\r
+       if (obj == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = ZeroMalloc(sizeof(SEC_OBJ));\r
+       ret->Name = CopyStr(obj->Name);\r
+       ret->Object = obj->Object;\r
+       ret->Private = obj->Private;\r
+       ret->Type = obj->Type;\r
+\r
+       return ret;\r
+}\r
+\r
+// セキュアオブジェクトを名前で検索して取得する\r
+SEC_OBJ *FindSecObject(SECURE *sec, char *name, UINT type)\r
+{\r
+       LIST *o;\r
+       UINT i;\r
+       SEC_OBJ *ret = NULL;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       if (name == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return NULL;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return 0;\r
+       }\r
+\r
+       // 列挙\r
+       o = EnumSecObject(sec);\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               SEC_OBJ *obj = LIST_DATA(o, i);\r
+\r
+               if (obj->Type == type || type == INFINITE)\r
+               {\r
+                       if (StrCmpi(obj->Name, name) == 0)\r
+                       {\r
+                               ret = CloneSecObject(obj);\r
+                               break;\r
+                       }\r
+               }\r
+       }\r
+       FreeEnumSecObject(o);\r
+\r
+       if (ret == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_OBJ_NOT_FOUND;\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// セキュアオブジェクトの読み込み\r
+int ReadSecDataFromObject(SECURE *sec, SEC_OBJ *obj, void *data, UINT size)\r
+{\r
+       UCHAR buf[MAX_SEC_DATA_SIZE];\r
+       UINT i;\r
+       CK_ATTRIBUTE get[] =\r
+       {\r
+               {CKA_VALUE,      buf,   sizeof(buf)},\r
+       };\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       if (obj == NULL || data == NULL || size == 0)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return 0;\r
+       }\r
+       if (obj->Type != SEC_DATA)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return 0;\r
+       }\r
+       if (sec->LoginFlag == false && obj->Private)\r
+       {\r
+               sec->Error = SEC_ERROR_NOT_LOGIN;\r
+               return 0;\r
+       }\r
+\r
+       // 取得\r
+       if (sec->Api->C_GetAttributeValue(\r
+               sec->SessionId, obj->Object, get, sizeof(get) / sizeof(get[0])) != CKR_OK)\r
+       {\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               return 0;\r
+       }\r
+\r
+       // 結果の返却\r
+       i = get[0].ulValueLen;\r
+       if (i > MAX_SEC_DATA_SIZE || i > size)\r
+       {\r
+               // データが大きすぎる\r
+               sec->Error = SEC_ERROR_DATA_TOO_BIG;\r
+               return 0;\r
+       }\r
+\r
+       // メモリコピー\r
+       Copy(data, buf, i);\r
+\r
+       return i;\r
+}\r
+\r
+// セキュアオブジェクトの列挙結果の解放\r
+void FreeEnumSecObject(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
+               SEC_OBJ *obj = LIST_DATA(o, i);\r
+\r
+               FreeSecObject(obj);\r
+       }\r
+\r
+       ReleaseList(o);\r
+}\r
+\r
+// セキュアオブジェクトの解放\r
+void FreeSecObject(SEC_OBJ *obj)\r
+{\r
+       // 引数チェック\r
+       if (obj == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Free(obj->Name);\r
+       Free(obj);\r
+}\r
+\r
+// セキュアオブジェクト列挙結果のクローン\r
+LIST *CloneEnumSecObject(LIST *o)\r
+{\r
+       LIST *ret;\r
+       UINT i;\r
+       // 引数チェック\r
+       if (o == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       ret = NewListFast(NULL);\r
+       for (i = 0;i < LIST_NUM(o);i++)\r
+       {\r
+               SEC_OBJ *obj = LIST_DATA(o, i);\r
+\r
+               Add(ret, CloneSecObject(obj));\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// セキュアオブジェクトの列挙\r
+LIST *EnumSecObject(SECURE *sec)\r
+{\r
+       CK_BBOOL b_true = true, b_false = false;\r
+       UINT objects[MAX_OBJ];\r
+       UINT i;\r
+       UINT ret;\r
+       LIST *o;\r
+       CK_ATTRIBUTE dummy[1];\r
+       CK_ATTRIBUTE a[] =\r
+       {\r
+               {CKA_TOKEN,             &b_true,                sizeof(b_true)},\r
+       };\r
+       UINT num_objects = MAX_OBJ;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return NULL;\r
+       }\r
+\r
+       Zero(dummy, sizeof(dummy));\r
+\r
+       // キャッシュがあればキャッシュを返す\r
+       if (sec->EnumCache != NULL)\r
+       {\r
+               return CloneEnumSecObject(sec->EnumCache);\r
+       }\r
+\r
+       // 列挙\r
+//     if (sec->Dev->Id != 2 && sec->Dev->Id != 14)\r
+//     {\r
+               // 通常のトークン\r
+               ret = sec->Api->C_FindObjectsInit(sec->SessionId, a, sizeof(a) / sizeof(a[0]));\r
+//     }\r
+//     else\r
+//     {\r
+               // ePass と SafeSign\r
+//             ret = sec->Api->C_FindObjectsInit(sec->SessionId, dummy, 0);\r
+//     }\r
+\r
+       if (ret != CKR_OK)\r
+       {\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               return NULL;\r
+       }\r
+       if (sec->Api->C_FindObjects(sec->SessionId, objects, sizeof(objects) / sizeof(objects[0]), &num_objects) != CKR_OK)\r
+       {\r
+               sec->Api->C_FindObjectsFinal(sec->SessionId);\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               return NULL;\r
+       }\r
+       sec->Api->C_FindObjectsFinal(sec->SessionId);\r
+\r
+       o = NewListFast(NULL);\r
+\r
+       for (i = 0;i < num_objects;i++)\r
+       {\r
+               char label[MAX_SIZE];\r
+               UINT obj_class = 0;\r
+               bool priv = false;\r
+               CK_ATTRIBUTE get[] =\r
+               {\r
+                       {CKA_LABEL, label, sizeof(label) - 1},\r
+                       {CKA_CLASS, &obj_class, sizeof(obj_class)},\r
+                       {CKA_PRIVATE, &priv, sizeof(priv)},\r
+               };\r
+\r
+               Zero(label, sizeof(label));\r
+\r
+               if (sec->Api->C_GetAttributeValue(sec->SessionId, objects[i],\r
+                       get, sizeof(get) / sizeof(get[0])) == CKR_OK)\r
+               {\r
+                       UINT type = INFINITE;\r
+\r
+                       switch (obj_class)\r
+                       {\r
+                       case CKO_DATA:\r
+                               // データ\r
+                               type = SEC_DATA;\r
+                               break;\r
+\r
+                       case CKO_CERTIFICATE:\r
+                               // 証明書\r
+                               type = SEC_X;\r
+                               break;\r
+\r
+                       case CKO_PUBLIC_KEY:\r
+                               // 公開鍵\r
+                               type = SEC_P;\r
+                               break;\r
+\r
+                       case CKO_PRIVATE_KEY:\r
+                               // 秘密鍵\r
+                               type = SEC_K;\r
+                               break;\r
+                       }\r
+\r
+                       if (type != INFINITE)\r
+                       {\r
+                               SEC_OBJ *obj = ZeroMalloc(sizeof(SEC_OBJ));\r
+\r
+                               obj->Type = type;\r
+                               obj->Object = objects[i];\r
+                               obj->Private = (priv == false) ? false : true;\r
+                               EnSafeStr(label, '?');\r
+                               TruncateCharFromStr(label, '?');\r
+                               obj->Name = CopyStr(label);\r
+\r
+                               Add(o, obj);\r
+                       }\r
+               }\r
+       }\r
+\r
+       // キャッシュ作成\r
+       sec->EnumCache = CloneEnumSecObject(o);\r
+\r
+       return o;\r
+}\r
+\r
+// データを書き込む\r
+bool WriteSecData(SECURE *sec, bool private_obj, char *name, void *data, UINT size)\r
+{\r
+       UINT object_class = CKO_DATA;\r
+       CK_BBOOL b_true = true, b_false = false, b_private_obj = private_obj;\r
+       UINT object;\r
+       CK_ATTRIBUTE a[] =\r
+       {\r
+               {CKA_TOKEN,             &b_true,                sizeof(b_true)},\r
+               {CKA_CLASS,             &object_class,  sizeof(object_class)},\r
+               {CKA_PRIVATE,   &b_private_obj, sizeof(b_private_obj)},\r
+               {CKA_LABEL,             name,                   StrLen(name)},\r
+               {CKA_VALUE,             data,                   size},\r
+       };\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+       }\r
+       if (private_obj && sec->LoginFlag == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NOT_LOGIN;\r
+               return false;\r
+       }\r
+       if (name == NULL || data == NULL || size == 0)\r
+       {\r
+               sec->Error = SEC_ERROR_BAD_PARAMETER;\r
+               return false;\r
+       }\r
+       if (size > MAX_SEC_DATA_SIZE)\r
+       {\r
+               sec->Error = SEC_ERROR_DATA_TOO_BIG;\r
+               return false;\r
+       }\r
+\r
+       // 同名のオブジェクトがあれば削除\r
+       if (CheckSecObject(sec, name, SEC_DATA))\r
+       {\r
+               DeleteSecData(sec, name);\r
+       }\r
+\r
+       // オブジェクト作成\r
+       if (sec->Api->C_CreateObject(sec->SessionId, a, sizeof(a) / sizeof(a[0]), &object) != CKR_OK)\r
+       {\r
+               sec->Error = SEC_ERROR_HARDWARE_ERROR;\r
+               return false;\r
+       }\r
+\r
+       // キャッシュ消去\r
+       EraseEnumSecObjectCache(sec);\r
+\r
+       return true;\r
+}\r
+\r
+// キャッシュに新規作成したオブジェクトの情報を追加する\r
+void AddSecObjToEnumCache(SECURE *sec, char *name, UINT type, bool private_obj, UINT object)\r
+{\r
+       SEC_OBJ *obj;\r
+       // 引数チェック\r
+       if (sec == NULL || name == NULL || sec->EnumCache == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       obj = ZeroMalloc(sizeof(SEC_OBJ));\r
+       obj->Name = CopyStr(name);\r
+       obj->Object = object;\r
+       obj->Private = private_obj;\r
+       obj->Type = type;\r
+\r
+       Add(sec->EnumCache, obj);\r
+}\r
+\r
+// トークン情報を表示\r
+void PrintSecInfo(SECURE *sec)\r
+{\r
+       SEC_INFO *s;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       s = sec->Info;\r
+       if (s == NULL)\r
+       {\r
+               Print("No Token Info.\n");\r
+               return;\r
+       }\r
+\r
+       Print(\r
+               "               Label: %S\n"\r
+               "      ManufacturerId: %S\n"\r
+               "               Model: %S\n"\r
+               "        SerialNumber: %S\n"\r
+               "          MaxSession: %u\n"\r
+               "        MaxRWSession: %u\n"\r
+               "           MinPinLen: %u\n"\r
+               "           MaxPinLen: %u\n"\r
+               "   TotalPublicMemory: %u\n"\r
+               "    FreePublicMemory: %u\n"\r
+               "  TotalPrivateMemory: %u\n"\r
+               "   FreePrivateMemory: %u\n"\r
+               "     HardwareVersion: %s\n"\r
+               "     FirmwareVersion: %s\n",\r
+               s->Label, s->ManufacturerId, s->Model, s->SerialNumber,\r
+               s->MaxSession, s->MaxRWSession, s->MinPinLen, s->MaxPinLen,\r
+               s->TotalPublicMemory, s->FreePublicMemory, s->TotalPrivateMemory,\r
+               s->FreePrivateMemory, s->HardwareVersion, s->FirmwareVersion\r
+               );\r
+}\r
+\r
+// トークン情報を取得\r
+void GetSecInfo(SECURE *sec)\r
+{\r
+       CK_TOKEN_INFO token_info;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (sec->Info != NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // 取得\r
+       Zero(&token_info, sizeof(token_info));\r
+       if (sec->Api->C_GetTokenInfo(sec->SlotIdList[sec->SessionSlotNumber], &token_info) != CKR_OK)\r
+       {\r
+               // 失敗\r
+               return;\r
+       }\r
+\r
+       sec->Info = TokenInfoToSecInfo(&token_info);\r
+}\r
+\r
+// トークン情報を解放\r
+void FreeSecInfo(SECURE *sec)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (sec->Info == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       FreeSecInfoMemory(sec->Info);\r
+       sec->Info = NULL;\r
+}\r
+\r
+// トークン情報を SEC_INFO に変換\r
+SEC_INFO *TokenInfoToSecInfo(void *p_t)\r
+{\r
+       SEC_INFO *s;\r
+       char buf[MAX_SIZE];\r
+       CK_TOKEN_INFO *t = (CK_TOKEN_INFO *)p_t;\r
+       // 引数チェック\r
+       if (t == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       s = ZeroMalloc(sizeof(SEC_INFO));\r
+\r
+       // Label\r
+       Zero(buf, sizeof(buf));\r
+       Copy(buf, t->label, sizeof(t->label));\r
+       s->Label = ZeroMalloc(CalcUtf8ToUni(buf, 0));\r
+       Utf8ToUni(s->Label, 0, buf, 0);\r
+\r
+       // ManufacturerId\r
+       Zero(buf, sizeof(buf));\r
+       Copy(buf, t->manufacturerID, sizeof(t->manufacturerID));\r
+       s->ManufacturerId = ZeroMalloc(CalcUtf8ToUni(buf, 0));\r
+       Utf8ToUni(s->ManufacturerId, 0, buf, 0);\r
+\r
+       // Model\r
+       Zero(buf, sizeof(buf));\r
+       Copy(buf, t->model, sizeof(t->model));\r
+       s->Model = ZeroMalloc(CalcUtf8ToUni(buf, 0));\r
+       Utf8ToUni(s->Model, 0, buf, 0);\r
+\r
+       // SerialNumber\r
+       Zero(buf, sizeof(buf));\r
+       Copy(buf, t->serialNumber, sizeof(t->serialNumber));\r
+       s->SerialNumber = ZeroMalloc(CalcUtf8ToUni(buf, 0));\r
+       Utf8ToUni(s->SerialNumber, 0, buf, 0);\r
+\r
+       // 数値\r
+       s->MaxSession = t->ulMaxSessionCount;\r
+       s->MaxRWSession = t->ulMaxRwSessionCount;\r
+       s->MinPinLen = t->ulMinPinLen;\r
+       s->MaxPinLen = t->ulMaxPinLen;\r
+       s->TotalPublicMemory = t->ulTotalPublicMemory;\r
+       s->FreePublicMemory = t->ulFreePublicMemory;\r
+       s->TotalPrivateMemory = t->ulTotalPrivateMemory;\r
+       s->FreePrivateMemory = t->ulFreePrivateMemory;\r
+\r
+       // ハードウェアバージョン\r
+       Format(buf, sizeof(buf), "%u.%02u", t->hardwareVersion.major, t->hardwareVersion.minor);\r
+       s->HardwareVersion = CopyStr(buf);\r
+\r
+       // ファームウェアバージョン\r
+       Format(buf, sizeof(buf), "%u.%02u", t->firmwareVersion.major, t->firmwareVersion.minor);\r
+       s->FirmwareVersion = CopyStr(buf);\r
+\r
+       return s;\r
+}\r
+\r
+// SEC_INFO のメモリを解放\r
+void FreeSecInfoMemory(SEC_INFO *s)\r
+{\r
+       // 引数チェック\r
+       if (s == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Free(s->Label);\r
+       Free(s->ManufacturerId);\r
+       Free(s->Model);\r
+       Free(s->SerialNumber);\r
+       Free(s->HardwareVersion);\r
+       Free(s->FirmwareVersion);\r
+       Free(s);\r
+}\r
+\r
+// ログアウトする\r
+void LogoutSec(SECURE *sec)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (sec->LoginFlag == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // ログアウト\r
+       sec->Api->C_Logout(sec->SessionId);\r
+\r
+       // キャッシュ消去\r
+       EraseEnumSecObjectCache(sec);\r
+\r
+       sec->LoginFlag = false;\r
+}\r
+\r
+// ログインする\r
+bool LoginSec(SECURE *sec, char *pin)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_SESSION;\r
+               return false;\r
+\r
+       }\r
+       if (sec->LoginFlag)\r
+       {\r
+               sec->Error = SEC_ERROR_ALREADY_LOGIN;\r
+               return false;\r
+       }\r
+       if (pin == NULL)\r
+       {\r
+               sec->Error = SEC_ERROR_NO_PIN_STR;\r
+               return false;\r
+       }\r
+\r
+       // ログイン\r
+       if (sec->Api->C_Login(sec->SessionId, CKU_USER, pin, StrLen(pin)) != CKR_OK)\r
+       {\r
+               // ログイン失敗\r
+               sec->Error = SEC_ERROR_BAD_PIN_CODE;\r
+               return false;\r
+       }\r
+\r
+       // キャッシュ消去\r
+       EraseEnumSecObjectCache(sec);\r
+\r
+       sec->LoginFlag = true;\r
+\r
+       return true;\r
+}\r
+\r
+// セッションを閉じる\r
+void CloseSecSession(SECURE *sec)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+       if (sec->SessionCreated == false)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // セッションを閉じる\r
+       sec->Api->C_CloseSession(sec->SessionId);\r
+\r
+       sec->SessionCreated = false;\r
+       sec->SessionId = 0;\r
+       sec->SessionSlotNumber = 0;\r
+\r
+       FreeSecInfo(sec);\r
+\r
+       // キャッシュ消去\r
+       EraseEnumSecObjectCache(sec);\r
+}\r
+\r
+// セッションを開く\r
+bool OpenSecSession(SECURE *sec, UINT slot_number)\r
+{\r
+       UINT err = 0;\r
+       UINT session;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+       if (sec->SessionCreated)\r
+       {\r
+               // すでに作成されている\r
+               sec->Error = SEC_ERROR_SESSION_EXISTS;\r
+               return false;\r
+       }\r
+       if (slot_number >= sec->NumSlot)\r
+       {\r
+               // スロット番号不正\r
+               sec->Error = SEC_ERROR_INVALID_SLOT_NUMBER;\r
+               return false;\r
+       }\r
+\r
+       // セッション作成\r
+       if ((err = sec->Api->C_OpenSession(sec->SlotIdList[slot_number],\r
+               CKF_RW_SESSION | CKF_SERIAL_SESSION, NULL, NULL, &session)) != CKR_OK)\r
+       {\r
+               // 読み書きモードでのセッション初期化に失敗した\r
+               // 読み取り専用モードかな?\r
+               if ((err = sec->Api->C_OpenSession(sec->SlotIdList[slot_number],\r
+                       CKF_SERIAL_SESSION, NULL, NULL, &session)) != CKR_OK)\r
+               {\r
+                       // 作成失敗\r
+                       sec->Error = SEC_ERROR_OPEN_SESSION;\r
+                       return false;\r
+               }\r
+               else\r
+               {\r
+                       sec->IsReadOnly = true;\r
+               }\r
+       }\r
+\r
+       sec->SessionCreated = true;\r
+       sec->SessionId = session;\r
+       sec->SessionSlotNumber = slot_number;\r
+\r
+       // トークン情報を取得\r
+       GetSecInfo(sec);\r
+\r
+       return true;\r
+}\r
+\r
+// セキュアデバイスを閉じる\r
+void CloseSec(SECURE *sec)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       // ログアウトする\r
+       LogoutSec(sec);\r
+\r
+       // セッションを閉じる\r
+       CloseSecSession(sec);\r
+\r
+       // トークン情報を解放\r
+       FreeSecInfo(sec);\r
+\r
+       // スロットリストメモリの解放\r
+       if (sec->SlotIdList != NULL)\r
+       {\r
+               Free(sec->SlotIdList);\r
+               sec->SlotIdList = NULL;\r
+       }\r
+\r
+       // モジュールのアンロード\r
+       FreeSecModule(sec);\r
+\r
+       // メモリ解放\r
+       DeleteLock(sec->lock);\r
+       Free(sec);\r
+}\r
+\r
+// セキュアデバイスを開く\r
+SECURE *OpenSec(UINT id)\r
+{\r
+       SECURE_DEVICE *dev = GetSecureDevice(id);\r
+       SECURE *sec;\r
+       UINT err;\r
+\r
+       if (dev == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       sec = ZeroMalloc(sizeof(SECURE));\r
+\r
+       sec->lock = NewLock();\r
+       sec->Error = SEC_ERROR_NOERROR;\r
+       sec->Dev = dev;\r
+\r
+       // ePass かどうか取得する\r
+       if (SearchStrEx(dev->DeviceName, "epass", 0, false) != INFINITE)\r
+       {\r
+               sec->IsEPass1000 = true;\r
+       }\r
+\r
+       // モジュールのロード\r
+       if (LoadSecModule(sec) == false)\r
+       {\r
+               CloseSec(sec);\r
+               return NULL;\r
+       }\r
+\r
+       // スロット一覧の取得\r
+       sec->NumSlot = 0;\r
+       if ((err = sec->Api->C_GetSlotList(true, NULL, &sec->NumSlot)) != CKR_OK || sec->NumSlot == 0)\r
+       {\r
+               // 失敗\r
+               FreeSecModule(sec);\r
+               CloseSec(sec);\r
+               return NULL;\r
+       }\r
+\r
+       sec->SlotIdList = (UINT *)ZeroMalloc(sizeof(UINT *) * sec->NumSlot);\r
+\r
+       if (sec->Api->C_GetSlotList(TRUE, sec->SlotIdList, &sec->NumSlot) != CKR_OK)\r
+       {\r
+               // 失敗\r
+               Free(sec->SlotIdList);\r
+               sec->SlotIdList = NULL;\r
+               FreeSecModule(sec);\r
+               CloseSec(sec);\r
+               return NULL;\r
+       }\r
+\r
+       return sec;\r
+}\r
+\r
+// セキュアデバイスのモジュールをロードする\r
+bool LoadSecModule(SECURE *sec)\r
+{\r
+       bool ret = false;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+#ifdef OS_WIN32\r
+       ret = Win32LoadSecModule(sec);\r
+#endif // OS_WIN32\r
+\r
+       // 初期化\r
+       if (sec->Api->C_Initialize(NULL) != CKR_OK)\r
+       {\r
+               // 初期化失敗\r
+               FreeSecModule(sec);\r
+               return false;\r
+       }\r
+\r
+       sec->Initialized = true;\r
+\r
+       return ret;\r
+}\r
+\r
+// セキュアデバイスのモジュールをアンロードする\r
+void FreeSecModule(SECURE *sec)\r
+{\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       if (sec->Initialized)\r
+       {\r
+               // 初期化済みなので解放する\r
+               sec->Api->C_Finalize(NULL);\r
+               sec->Initialized = false;\r
+       }\r
+\r
+#ifdef OS_WIN32\r
+       Win32FreeSecModule(sec);\r
+#endif // OS_WIN32\r
+\r
+}\r
+\r
+\r
+// セキュアデバイスを取得する\r
+SECURE_DEVICE *GetSecureDevice(UINT id)\r
+{\r
+       UINT i;\r
+\r
+       if (id == 0)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(SecureDeviceList);i++)\r
+       {\r
+               SECURE_DEVICE *dev = LIST_DATA(SecureDeviceList, i);\r
+\r
+               if (dev->Id == id)\r
+               {\r
+                       return dev;\r
+               }\r
+       }\r
+\r
+       return NULL;\r
+}\r
+\r
+// セキュアデバイスの ID を確認する\r
+bool CheckSecureDeviceId(UINT id)\r
+{\r
+       UINT i;\r
+\r
+       for (i = 0;i < LIST_NUM(SecureDeviceList);i++)\r
+       {\r
+               SECURE_DEVICE *dev = LIST_DATA(SecureDeviceList, i);\r
+\r
+               if (dev->Id == id)\r
+               {\r
+                       return true;\r
+               }\r
+       }\r
+\r
+       return false;\r
+}\r
+\r
+// サポートされているデバイスリストを取得する\r
+LIST *GetSecureDeviceList()\r
+{\r
+       return GetSupportedDeviceList();\r
+}\r
+\r
+// サポートされているデバイスリストを取得する\r
+LIST *GetSupportedDeviceList()\r
+{\r
+       // 参照カウントの増加\r
+       AddRef(SecureDeviceList->ref);\r
+\r
+       return SecureDeviceList;\r
+}\r
+\r
+// 指定したデバイスがインストールされていて利用可能かどうか調べる\r
+bool IsDeviceSupported(SECURE_DEVICE *dev)\r
+{\r
+       bool b = false;\r
+#ifdef OS_WIN32\r
+       b = Win32IsDeviceSupported(dev);\r
+#endif // OS_WIN32\r
+       return b;\r
+}\r
+\r
+// セキュアデバイスリストの初期化\r
+void InitSecureDeviceList()\r
+{\r
+       UINT i, num_supported_list;\r
+       SecureDeviceList = NewList(NULL);\r
+\r
+       num_supported_list = sizeof(SupportedList) / sizeof(SECURE_DEVICE);\r
+       for (i = 0; i < num_supported_list;i++)\r
+       {\r
+               SECURE_DEVICE *dev = &SupportedList[i];\r
+\r
+               // サポートチェック\r
+               if (IsDeviceSupported(dev))\r
+               {\r
+                       // サポートされているのでリストに追加\r
+                       Add(SecureDeviceList, dev);\r
+               }\r
+       }\r
+}\r
+\r
+// テストメイン処理\r
+void TestSecMain(SECURE *sec)\r
+{\r
+       char *test_str = "SoftEther UT-VPN";\r
+       K *public_key, *private_key;\r
+       // 引数チェック\r
+       if (sec == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Print("test_str: \"%s\"\n", test_str);\r
+\r
+       Print("Writing Data...\n");\r
+       if (WriteSecData(sec, true, "test_str", test_str, StrLen(test_str)) == false)\r
+       {\r
+               Print("WriteSecData() Failed.\n");\r
+       }\r
+       else\r
+       {\r
+               char data[MAX_SIZE];\r
+               Zero(data, sizeof(data));\r
+               Print("Reading Data...\n");\r
+               if (ReadSecData(sec, "test_str", data, sizeof(data)) == false)\r
+               {\r
+                       Print("ReadSecData() Failed.\n");\r
+               }\r
+               else\r
+               {\r
+                       Print("test_str: \"%s\"\n", data);\r
+               }\r
+               Print("Deleting Data...\n");\r
+               DeleteSecData(sec, "test_str");\r
+       }\r
+\r
+       Print("Generating Key...\n");\r
+       if (RsaGen(&private_key, &public_key, 1024) == false)\r
+       {\r
+               Print("RsaGen() Failed.\n");\r
+       }\r
+       else\r
+       {\r
+               X *cert;\r
+               NAME *name;\r
+               X_SERIAL *serial;\r
+               UINT num = 0x11220000;\r
+\r
+               Print("Creating Cert...\n");\r
+               serial = NewXSerial(&num, sizeof(UINT));\r
+               name = NewName(L"Test", L"Test", L"Test", L"JP", L"Test", L"Test");\r
+               cert = NewRootX(public_key, private_key, name, 365, NULL);\r
+               FreeXSerial(serial);\r
+               if (cert == NULL)\r
+               {\r
+                       Print("NewRootX() Failed.\n");\r
+               }\r
+               else\r
+               {\r
+                       Print("Writing Cert...\n");\r
+                       DeleteSecData(sec, "test_cer");\r
+                       if (WriteSecCert(sec, true, "test_cer", cert) == false)\r
+                       {\r
+                               Print("WriteSecCert() Failed.\n");\r
+                       }\r
+                       else\r
+                       {\r
+                               X *x;\r
+                               Print("Reading Cert...\n");\r
+                               x = ReadSecCert(sec, "test_cer");\r
+                               if (x == NULL)\r
+                               {\r
+                                       Print("ReadSecCert() Failed.\n");\r
+                               }\r
+                               else\r
+                               {\r
+                                       Print("Checking two Certs... ");\r
+                                       if (CompareX(x, cert) == false)\r
+                                       {\r
+                                               Print("[FAILED]\n");\r
+                                       }\r
+                                       else\r
+                                       {\r
+                                               Print("Ok.\n");\r
+                                       }\r
+                                       FreeX(x);\r
+                               }\r
+                               if (cert != NULL)\r
+                               {\r
+                                       X *x;\r
+                                       XToFile(cert, "cert_tmp.cer", true);\r
+                                       x = FileToX("cert_tmp.cer");\r
+                                       if (CompareX(x, cert) == false)\r
+                                       {\r
+                                               Print("[FAILED]\n");\r
+                                       }\r
+                                       else\r
+                                       {\r
+                                               Print("Ok.\n");\r
+                                               Print("Writing Private Key...\n");\r
+                                               DeleteSecKey(sec, "test_key");\r
+                                               if (WriteSecKey(sec, true, "test_key", private_key) == false)\r
+                                               {\r
+                                                       Print("WriteSecKey() Failed.\n");\r
+                                               }\r
+                                               else\r
+                                               {\r
+                                                       UCHAR sign_cpu[128];\r
+                                                       UCHAR sign_sec[128];\r
+                                                       K *pub = GetKFromX(cert);\r
+                                                       Print("Ok.\n");\r
+                                                       Print("Signing Data by CPU...\n");\r
+                                                       if (RsaSign(sign_cpu, test_str, StrLen(test_str), private_key) == false)\r
+                                                       {\r
+                                                               Print("RsaSign() Failed.\n");\r
+                                                       }\r
+                                                       else\r
+                                                       {\r
+                                                               Print("Ok.\n");\r
+                                                               Print("sign_cpu: ");\r
+                                                               PrintBin(sign_cpu, sizeof(sign_cpu));\r
+                                                               Print("Signing Data by %s..\n", sec->Dev->DeviceName);\r
+                                                               if (SignSec(sec, "test_key", sign_sec, test_str, StrLen(test_str)) == false)\r
+                                                               {\r
+                                                                       Print("SignSec() Failed.\n");\r
+                                                               }\r
+                                                               else\r
+                                                               {\r
+                                                                       Print("Ok.\n");\r
+                                                                       Print("sign_sec: ");\r
+                                                                       PrintBin(sign_sec, sizeof(sign_sec));\r
+                                                                       Print("Compare...");\r
+                                                                       if (Cmp(sign_sec, sign_cpu, sizeof(sign_cpu)) == 0)\r
+                                                                       {\r
+                                                                               Print("Ok.\n");\r
+                                                                               Print("Verify...");\r
+                                                                               if (RsaVerify(test_str, StrLen(test_str),\r
+                                                                                       sign_sec, pub) == false)\r
+                                                                               {\r
+                                                                                       Print("[FAILED]\n");\r
+                                                                               }\r
+                                                                               else\r
+                                                                               {\r
+                                                                                       Print("Ok.\n");\r
+                                                                               }\r
+                                                                       }\r
+                                                                       else\r
+                                                                       {\r
+                                                                               Print("[DIFFIRENT]\n");\r
+                                                                       }\r
+                                                               }\r
+                                                       }\r
+                                                       Print("Deleting test_key...\n");\r
+//                                                     DeleteSecKey(sec, "test_key");\r
+                                                       FreeK(pub);\r
+                                               }\r
+                                       }\r
+                                       FreeX(x);\r
+                               }\r
+                       }\r
+                       Print("Deleting Cert..\n");\r
+//                     DeleteSecCert(sec, "test_cer");\r
+                       FreeX(cert);\r
+               }\r
+               FreeName(name);\r
+               FreeK(private_key);\r
+               FreeK(public_key);\r
+       }\r
+}\r
+\r
+// セキュリティデバイスのテスト\r
+void TestSec()\r
+{\r
+       UINT i;\r
+       LIST *secure_device_list;\r
+       Print("Secure Device Test Program\n"\r
+               "Copyright (C) 2004-2010 SoftEther Corporation. All Rights Reserved.\n\n");\r
+\r
+       // セキュアデバイスリストの取得\r
+       secure_device_list = GetSecureDeviceList();\r
+       if (secure_device_list != NULL)\r
+       {\r
+               UINT use_device_id;\r
+               char tmp[MAX_SIZE];\r
+               Print("--- Secure Device List ---\n");\r
+               for (i = 0;i < LIST_NUM(secure_device_list);i++)\r
+               {\r
+                       SECURE_DEVICE *dev = LIST_DATA(secure_device_list, i);\r
+                       Print("%2u - %s\n", dev->Id, dev->DeviceName);\r
+               }\r
+               Print("\n");\r
+               Print("Device ID >");\r
+               GetLine(tmp, sizeof(tmp));\r
+               use_device_id = ToInt(tmp);\r
+               if (use_device_id == 0)\r
+               {\r
+                       Print("Canceled.\n");\r
+               }\r
+               else\r
+               {\r
+                       SECURE *sec = OpenSec(use_device_id);\r
+                       Print("Opening Device...\n");\r
+                       if (sec == NULL)\r
+                       {\r
+                               Print("OpenSec() Failed.\n");\r
+                       }\r
+                       else\r
+                       {\r
+                               Print("Opening Session...\n");\r
+                               if (OpenSecSession(sec, 0) == false)\r
+                               {\r
+                                       Print("OpenSecSession() Failed.\n");\r
+                               }\r
+                               else\r
+                               {\r
+                                       while (true)\r
+                                       {\r
+                                               char pin[MAX_SIZE];\r
+                                               Print("PIN Code >");\r
+                                               GetLine(pin, sizeof(pin));\r
+                                               Trim(pin);\r
+                                               if (StrLen(pin) == 0)\r
+                                               {\r
+                                                       Print("Canceled.\n");\r
+                                                       break;\r
+                                               }\r
+                                               else\r
+                                               {\r
+                                                       Print("Login...\n");\r
+                                                       if (LoginSec(sec, pin))\r
+                                                       {\r
+                                                               TestSecMain(sec);\r
+                                                               Print("Logout...\n");\r
+                                                               LogoutSec(sec);\r
+                                                               break;\r
+                                                       }\r
+                                                       else\r
+                                                       {\r
+                                                               Print("Login Failed. Please Try Again.\n");\r
+                                                       }\r
+                                               }\r
+                                       }\r
+                                       Print("Closing Session...\n");\r
+                                       CloseSecSession(sec);\r
+                               }\r
+                               Print("Closing Device...\n");\r
+                               CloseSec(sec);\r
+                       }\r
+               }\r
+               ReleaseList(secure_device_list);\r
+       }\r
+       else\r
+       {\r
+               Print("GetSecureDeviceList() Error.\n");\r
+       }\r
+}\r
+\r
+// セキュアデバイスリストの解放\r
+void FreeSecureDeviceList()\r
+{\r
+       ReleaseList(SecureDeviceList);\r
+}\r
+\r
+// セキュリティトークンモジュールの初期化\r
+void InitSecure()\r
+{\r
+       // セキュアデバイスリストの初期化\r
+       InitSecureDeviceList();\r
+}\r
+\r
+// セキュリティトークンモジュールの解放\r
+void FreeSecure()\r
+{\r
+       // セキュアデバイスリストの解放\r
+       FreeSecureDeviceList();\r
+}\r
+\r
+\r