* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Cedar / Sam.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Cedar/Sam.c b/utvpn/utvpn-unix-v101-7101-public/src/Cedar/Sam.c
new file mode 100644 (file)
index 0000000..2edc304
--- /dev/null
@@ -0,0 +1,347 @@
+// 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
+// Sam.c\r
+// セキュリティアカウントマネージャ\r
+\r
+// Windows NT の SAM を真似してかっこいい名前を付けてみたが、\r
+// 実装してみると結局大した処理はしていない。\r
+\r
+#include "CedarPch.h"\r
+\r
+// パスワードの暗号化\r
+void SecurePassword(void *secure_password, void *password, void *random)\r
+{\r
+       BUF *b;\r
+       // 引数チェック\r
+       if (secure_password == NULL || password == NULL || random == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       b = NewBuf();\r
+       WriteBuf(b, password, SHA1_SIZE);\r
+       WriteBuf(b, random, SHA1_SIZE);\r
+       Hash(secure_password, b->Buf, b->Size, true);\r
+\r
+       FreeBuf(b);\r
+}\r
+\r
+// 160bit 乱数の生成\r
+void GenRamdom(void *random)\r
+{\r
+       // 引数チェック\r
+       if (random == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+       Rand(random, SHA1_SIZE);\r
+}\r
+\r
+// ユーザーの匿名認証\r
+bool SamAuthUserByAnonymous(HUB *h, char *username)\r
+{\r
+       bool b = false;\r
+       // 引数チェック\r
+       if (h == NULL || username == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       AcLock(h);\r
+       {\r
+               USER *u = AcGetUser(h, username);\r
+               if (u)\r
+               {\r
+                       Lock(u->lock);\r
+                       {\r
+                               if (u->AuthType == AUTHTYPE_ANONYMOUS)\r
+                               {\r
+                                       b = true;\r
+                               }\r
+                       }\r
+                       Unlock(u->lock);\r
+               }\r
+               ReleaseUser(u);\r
+       }\r
+       AcUnlock(h);\r
+\r
+       return b;\r
+}\r
+\r
+// 指定した証明書を署名したルート証明書をリストから取得する\r
+X *GetIssuerFromList(LIST *cert_list, X *cert)\r
+{\r
+       UINT i;\r
+       X *ret = NULL;\r
+       // 引数チェック\r
+       if (cert_list == NULL || cert == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       for (i = 0;i < LIST_NUM(cert_list);i++)\r
+       {\r
+               X *x = LIST_DATA(cert_list, i);\r
+               // 名前の比較\r
+               if (CheckXDateNow(x))\r
+               {\r
+                       if (CompareName(x->subject_name, cert->issuer_name))\r
+                       {\r
+                               // ルート証明書の公開鍵を取得\r
+                               K *k = GetKFromX(x);\r
+\r
+                               if (k != NULL)\r
+                               {\r
+                                       // 署名のチェック\r
+                                       if (CheckSignature(cert, k))\r
+                                       {\r
+                                               ret = x;\r
+                                       }\r
+                                       FreeK(k);\r
+                               }\r
+                       }\r
+               }\r
+               if (CompareX(x, cert))\r
+               {\r
+                       // 完全同一\r
+                       ret = x;\r
+               }\r
+       }\r
+\r
+       return ret;\r
+}\r
+\r
+// ユーザーに適用されるべきポリシーを取得\r
+POLICY *SamGetUserPolicy(HUB *h, char *username)\r
+{\r
+       POLICY *ret = NULL;\r
+       // 引数チェック\r
+       if (h == NULL || username == NULL)\r
+       {\r
+               return NULL;\r
+       }\r
+\r
+       AcLock(h);\r
+       {\r
+               USER *u;\r
+               u = AcGetUser(h, username);\r
+               if (u)\r
+               {\r
+                       USERGROUP *g = NULL;\r
+                       Lock(u->lock);\r
+                       {\r
+                               if (u->Policy != NULL)\r
+                               {\r
+                                       ret = ClonePolicy(u->Policy);\r
+                               }\r
+\r
+                               g = u->Group;\r
+\r
+                               if (g != NULL)\r
+                               {\r
+                                       AddRef(g->ref);\r
+                               }\r
+                       }\r
+                       Unlock(u->lock);\r
+\r
+                       ReleaseUser(u);\r
+                       u = NULL;\r
+\r
+                       if (ret == NULL)\r
+                       {\r
+                               if (g != NULL)\r
+                               {\r
+                                       Lock(g->lock);\r
+                                       {\r
+                                               ret = ClonePolicy(g->Policy);\r
+                                       }\r
+                                       Unlock(g->lock);\r
+                               }\r
+                       }\r
+\r
+                       if (g != NULL)\r
+                       {\r
+                               ReleaseGroup(g);\r
+                       }\r
+               }\r
+       }\r
+       AcUnlock(h);\r
+\r
+       return ret;\r
+}\r
+\r
+// ユーザーのパスワード認証\r
+bool SamAuthUserByPassword(HUB *h, char *username, void *random, void *secure_password)\r
+{\r
+       bool b = false;\r
+       UCHAR secure_password_check[SHA1_SIZE];\r
+       // 引数チェック\r
+       if (h == NULL || username == NULL || secure_password == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       if (StrCmpi(username, ADMINISTRATOR_USERNAME) == 0)\r
+       {\r
+               // Administrator モード\r
+               SecurePassword(secure_password_check, h->SecurePassword, random);\r
+               if (Cmp(secure_password_check, secure_password, SHA1_SIZE) == 0)\r
+               {\r
+                       return true;\r
+               }\r
+               else\r
+               {\r
+                       return false;\r
+               }\r
+       }\r
+\r
+       AcLock(h);\r
+       {\r
+               USER *u;\r
+               u = AcGetUser(h, username);\r
+               if (u)\r
+               {\r
+                       Lock(u->lock);\r
+                       {\r
+                               if (u->AuthType == AUTHTYPE_PASSWORD)\r
+                               {\r
+                                       AUTHPASSWORD *auth = (AUTHPASSWORD *)u->AuthData;\r
+                                       SecurePassword(secure_password_check, auth->HashedKey, random);\r
+                                       if (Cmp(secure_password_check, secure_password, SHA1_SIZE) == 0)\r
+                                       {\r
+                                               b = true;\r
+                                       }\r
+                               }\r
+                       }\r
+                       Unlock(u->lock);\r
+                       ReleaseUser(u);\r
+               }\r
+       }\r
+       AcUnlock(h);\r
+\r
+       return b;\r
+}\r
+\r
+// ユーザーが存在することを確認\r
+bool SamIsUser(HUB *h, char *username)\r
+{\r
+       bool b;\r
+       // 引数チェック\r
+       if (h == NULL || username == NULL)\r
+       {\r
+               return false;\r
+       }\r
+\r
+       AcLock(h);\r
+       {\r
+               b = AcIsUser(h, username);\r
+       }\r
+       AcUnlock(h);\r
+\r
+       return b;\r
+}\r
+\r
+// ユーザーが使用する認証の種類を取得\r
+UINT SamGetUserAuthType(HUB *h, char *username)\r
+{\r
+       UINT authtype;\r
+       // 引数チェック\r
+       if (h == NULL || username == NULL)\r
+       {\r
+               return INFINITE;\r
+       }\r
+\r
+       AcLock(h);\r
+       {\r
+               USER *u = AcGetUser(h, username);\r
+               if (u == NULL)\r
+               {\r
+                       authtype = INFINITE;\r
+               }\r
+               else\r
+               {\r
+                       authtype = u->AuthType;\r
+                       ReleaseUser(u);\r
+               }\r
+       }\r
+       AcUnlock(h);\r
+\r
+       return authtype;\r
+}\r
+\r