* tar xzf utvpn-src-unix-v101-7101-public-2010.06.27.tar.gz
[lab.git] / utvpn / utvpn-unix-v101-7101-public / src / Mayaqua / Tick64.c
diff --git a/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Tick64.c b/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Tick64.c
new file mode 100644 (file)
index 0000000..21869b0
--- /dev/null
@@ -0,0 +1,370 @@
+// 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
+// Tick64.c\r
+// 64bit リアルタイムクロックプログラム\r
+\r
+#ifdef WIN32\r
+#include <windows.h>\r
+#endif // WIN32\r
+\r
+\r
+#include <stdio.h>\r
+#include <stdlib.h>\r
+#include <string.h>\r
+#include <wchar.h>\r
+#include <stdarg.h>\r
+#include <locale.h>\r
+#include <time.h>\r
+#include <errno.h>\r
+#include <Mayaqua/Mayaqua.h>\r
+\r
+static TICK64 *tk64 = NULL;\r
+static EVENT *halt_tick_event = NULL;\r
+\r
+// 高解像度時刻を取得\r
+UINT64 TickHighres64()\r
+{\r
+       UINT64 ret = 0;\r
+\r
+#ifdef OS_WIN32\r
+\r
+       ret = (UINT64)(MsGetHiResTimeSpan(MsGetHiResCounter()) * 1000.0f);\r
+\r
+#else  // OS_WIN32\r
+\r
+       return Tick64();\r
+\r
+#endif // OS_WIN32\r
+\r
+       return ret;\r
+}\r
+\r
+// Tick 値を時刻に変換\r
+UINT64 Tick64ToTime64(UINT64 tick)\r
+{\r
+       UINT64 ret = 0;\r
+       if (tick == 0)\r
+       {\r
+               return 0;\r
+       }\r
+       LockList(tk64->AdjustTime);\r
+       {\r
+               UINT i;\r
+               for (i = 0;i < LIST_NUM(tk64->AdjustTime);i++)\r
+               {\r
+                       ADJUST_TIME *t = LIST_DATA(tk64->AdjustTime, i);\r
+                       if (t->Tick <= tick)\r
+                       {\r
+                               ret = t->Time + (tick - t->Tick);\r
+                       }\r
+               }\r
+       }\r
+       UnlockList(tk64->AdjustTime);\r
+       if (ret == 0)\r
+       {\r
+               ret++;\r
+       }\r
+       return ret;\r
+}\r
+\r
+// Tick 値を時刻に変換\r
+UINT64 TickToTime(UINT64 tick)\r
+{\r
+       return Tick64ToTime64(tick);\r
+}\r
+\r
+// Tick 値を取得\r
+UINT64 Tick64()\r
+{\r
+#ifdef OS_WIN32\r
+       return Win32FastTick64();\r
+#else  // OS_WIN32\r
+       UINT64 tick64;\r
+       if (tk64 == NULL)\r
+       {\r
+               return 0;\r
+       }\r
+       Lock(tk64->TickLock);\r
+       {\r
+               tick64 = tk64->Tick;\r
+       }\r
+       Unlock(tk64->TickLock);\r
+       return tick64;\r
+#endif // OS_WIN32\r
+}\r
+\r
+// リアルタイムクロック測定用スレッド\r
+void Tick64Thread(THREAD *thread, void *param)\r
+{\r
+       UINT n = 0;\r
+       bool first = false;\r
+       bool create_first_entry = true;\r
+       UINT tick_span;\r
+       // 引数チェック\r
+       if (thread == NULL)\r
+       {\r
+               return;\r
+       }\r
+\r
+#ifdef OS_WIN32\r
+\r
+       // Win32 スレッドの優先順位を上げる\r
+       MsSetThreadPriorityRealtime();\r
+\r
+       tick_span = TICK64_SPAN_WIN32;\r
+\r
+#else  // OS_WIN32\r
+\r
+       // POSIX スレッドの優先順位を上げる\r
+       UnixSetThreadPriorityRealtime();\r
+\r
+       tick_span = TICK64_SPAN;\r
+\r
+#endif // OS_WIN32\r
+\r
+       while (true)\r
+       {\r
+               UINT tick;\r
+               UINT64 tick64;\r
+\r
+#ifndef        OS_WIN32\r
+               tick = TickRealtime();          // 現在のシステムクロックを取得\r
+\r
+               if (tk64->LastTick > tick)\r
+               {\r
+                       if ((tk64->LastTick - tick) >= (UINT64)0x0fffffff)\r
+                       {\r
+                               // tick の値が 1 周した\r
+                               tk64->RoundCount++;\r
+                       }\r
+                       else\r
+                       {\r
+                               // tick の値が逆戻りした (システム管理者がハードウェア時計を変更した)\r
+                               // 通常これは 1 秒以内の誤差として発生する\r
+                               tick = tk64->LastTick;\r
+                       }\r
+               }\r
+               tk64->LastTick = tick;\r
+\r
+               tick64 = (UINT64)tk64->RoundCount * (UINT64)4294967296LL + (UINT64)tick;\r
+\r
+               Lock(tk64->TickLock);\r
+               {\r
+                       if (tk64->TickStart == 0)\r
+                       {\r
+                               tk64->TickStart = tick64;\r
+                       }\r
+                       tick64 = tk64->Tick = tick64 - tk64->TickStart + (UINT64)1;\r
+               }\r
+               Unlock(tk64->TickLock);\r
+#else  // OS_WIN32\r
+               tick64 = Win32FastTick64();\r
+               tick = (UINT)tick64;\r
+#endif // OS_WIN32\r
+\r
+               if (create_first_entry)\r
+               {\r
+                       ADJUST_TIME *t = ZeroMalloc(sizeof(ADJUST_TIME));\r
+                       t->Tick = tick64;\r
+                       t->Time = SystemTime64();\r
+                       tk64->Tick64WithTime64 = tick64;\r
+                       tk64->Time64 = t->Time;\r
+                       Add(tk64->AdjustTime, t);\r
+\r
+                       // 初期化完了を通知\r
+                       NoticeThreadInit(thread);\r
+                       create_first_entry = false;\r
+               }\r
+\r
+               // 時刻補正\r
+               n += tick_span;\r
+               if (n >= 1000 || first == false)\r
+               {\r
+                       UINT64 now = SystemTime64();\r
+\r
+                       if (now < tk64->Time64 ||\r
+                               Diff64((now - tk64->Time64) + tk64->Tick64WithTime64, tick64) >= tick_span)\r
+                       {\r
+                               ADJUST_TIME *t = ZeroMalloc(sizeof(ADJUST_TIME));\r
+                               LockList(tk64->AdjustTime);\r
+                               {\r
+                                       t->Tick = tick64;\r
+                                       t->Time = now;\r
+                                       Add(tk64->AdjustTime, t);\r
+                                       Debug("Adjust Time: Tick = %I64u, Time = %I64u\n",\r
+                                               t->Tick, t->Time);\r
+\r
+                                       // 時計が狂っているシステムでメモリを無限に食わないように\r
+                                       if (LIST_NUM(tk64->AdjustTime) > MAX_ADJUST_TIME)\r
+                                       {\r
+                                               // 2 個目を削除する\r
+                                               ADJUST_TIME *t2 = LIST_DATA(tk64->AdjustTime, 1);\r
+\r
+                                               Delete(tk64->AdjustTime, t2);\r
+\r
+                                               Debug("NUM_ADJUST TIME: %u\n", LIST_NUM(tk64->AdjustTime));\r
+\r
+                                               Free(t2);\r
+                                       }\r
+                               }\r
+                               UnlockList(tk64->AdjustTime);\r
+                               tk64->Time64 = now;\r
+                               tk64->Tick64WithTime64 = tick64;\r
+                       }\r
+                       first = true;\r
+                       n = 0;\r
+               }\r
+\r
+               if (tk64->Halt)\r
+               {\r
+                       break;\r
+               }\r
+\r
+#ifdef OS_WIN32\r
+               Wait(halt_tick_event, tick_span);\r
+#else  // OS_WIN32\r
+               SleepThread(tick_span);\r
+#endif // OS_WIN32\r
+       }\r
+}\r
+\r
+// 2 つの 64 bit 整数の差の絶対値を取得\r
+UINT64 Diff64(UINT64 a, UINT64 b)\r
+{\r
+       if (a > b)\r
+       {\r
+               return a - b;\r
+       }\r
+       else\r
+       {\r
+               return b - a;\r
+       }\r
+}\r
+\r
+// Tick64 の初期化\r
+void InitTick64()\r
+{\r
+       if (tk64 != NULL)\r
+       {\r
+               // すでに初期化されている\r
+               return;\r
+       }\r
+\r
+       halt_tick_event = NewEvent();\r
+\r
+       // 構造体の初期化\r
+       tk64 = ZeroMalloc(sizeof(TICK64));\r
+       tk64->TickLock = NewLock();\r
+       tk64->AdjustTime = NewList(NULL);\r
+\r
+       // スレッドの作成\r
+       tk64->Thread = NewThread(Tick64Thread, NULL);\r
+       WaitThreadInit(tk64->Thread);\r
+}\r
+\r
+// Tick64 の解放\r
+void FreeTick64()\r
+{\r
+       UINT i;\r
+       if (tk64 == NULL)\r
+       {\r
+               // 初期化されていない\r
+               return;\r
+       }\r
+\r
+       // 終了処理\r
+       tk64->Halt = true;\r
+       Set(halt_tick_event);\r
+       WaitThread(tk64->Thread, INFINITE);\r
+       ReleaseThread(tk64->Thread);\r
+\r
+       // 解放処理\r
+       for (i = 0;i < LIST_NUM(tk64->AdjustTime);i++)\r
+       {\r
+               ADJUST_TIME *t = LIST_DATA(tk64->AdjustTime, i);\r
+               Free(t);\r
+       }\r
+       ReleaseList(tk64->AdjustTime);\r
+       DeleteLock(tk64->TickLock);\r
+       Free(tk64);\r
+       tk64 = NULL;\r
+\r
+       ReleaseEvent(halt_tick_event);\r
+       halt_tick_event = NULL;\r
+}\r
+\r