source: lab.git/Dev/utvpn/utvpn-unix-v101-7101-public/src/Mayaqua/Tick64.c @ 86521dd

trunk
Last change on this file since 86521dd was a1bae3e, checked in by mitty <mitty@…>, 12 years ago
  • copy vendor drop to trunk

git-svn-id: https://lab.mitty.jp/svn/lab/trunk@147 7d2118f6-f56c-43e7-95a2-4bb3031d96e7

  • Property mode set to 100644
File size: 10.1 KB
Line 
1// SoftEther UT-VPN SourceCode
2//
3// Copyright (C) 2004-2010 SoftEther Corporation.
4// Copyright (C) 2004-2010 University of Tsukuba, Japan.
5// Copyright (C) 2003-2010 Daiyuu Nobori.
6// All Rights Reserved.
7//
8// http://utvpn.tsukuba.ac.jp/
9//
10// This program is free software; you can redistribute it and/or
11// modify it under the terms of the GNU General Public License
12// version 2 as published by the Free Software Foundation.
13//
14// This program is distributed in the hope that it will be useful,
15// but WITHOUT ANY WARRANTY; without even the implied warranty of
16// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17// GNU General Public License for more details.
18//
19// You should have received a copy of the GNU General Public License version 2
20// along with this program; if not, write to the Free Software
21// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
22//
23// このファイルは GPL バージョン 2 ライセンスで公開されています。
24// 誰でもこのファイルの内容を複製、改変したり、改変したバージョンを再配布
25// することができます。ただし、原著作物を改変した場合は、原著作物の著作権表示
26// を除去することはできません。改変した著作物を配布する場合は、改変実施者の
27// 著作権表示を原著作物の著作権表示に付随して記載するようにしてください。
28//
29// この SoftEther UT-VPN オープンソース・プロジェクトは、日本国の
30// ソフトイーサ株式会社 (SoftEther Corporation, http://www.softether.co.jp/ )
31// および筑波大学 (University of Tsukuba, http://www.tsukuba.ac.jp/ ) によって
32// ホストされています。
33// 本プログラムの配布者は、本プログラムを、業としての利用以外のため、
34// および、試験または研究のために利用が行われることを想定して配布
35// しています。
36// SoftEther UT-VPN プロジェクトの Web サイトは http://utvpn.tsukuba.ac.jp/ に
37// あります。
38// 本ソフトウェアの不具合の修正、機能改良、セキュリティホールの修復などのコード
39// の改変を行った場合で、その成果物を SoftEther UT-VPN プロジェクトに提出して
40// いただける場合は、 http://utvpn.tsukuba.ac.jp/ までソースコードを送付して
41// ください。SoftEther UT-VPN プロジェクトの本体リリースまたはブランチリリース
42// に組み込みさせていただきます。
43//
44// GPL に基づいて原著作物が提供される本ソフトウェアの改良版を配布、販売する
45// 場合は、そのソースコードを GPL に基づいて誰にでも開示する義務が生じます。
46//
47// 本ソフトウェアに関連する著作権、特許権、商標権はソフトイーサ株式会社
48// (SoftEther Corporation) およびその他の著作権保持者が保有しています。
49// ソフトイーサ株式会社等はこれらの権利を放棄していません。本ソフトウェアの
50// 二次著作物を配布、販売する場合は、これらの権利を侵害しないようにご注意
51// ください。
52//
53// お願い: どのような通信ソフトウェアにも通常は必ず未発見の
54// セキュリティホールが潜んでいます。本ソースコードをご覧いただいた結果、
55// UT-VPN にセキュリティホールを発見された場合は、当該セキュリティホールの
56// 情報を不特定多数に開示される前に、必ず、ソフトイーサ株式会社
57// および脆弱性情報の届出を受け付ける公的機関まで通報いただき、
58// 公益保護にご協力いただきますようお願い申し上げます。
59//
60// ソフトイーサ株式会社は、当該セキュリティホールについて迅速に対処を
61// 行い、UT-VPN および UT-VPN に関連するソフトウェアのユーザー・顧客
62// を保護するための努力を行います。
63//
64// ソフトイーサへの届出先: http://www.softether.co.jp/jp/contact/
65// 日本国内の脆弱性情報届出受付公的機関:
66//         独立行政法人 情報処理推進機構
67//         http://www.ipa.go.jp/security/vuln/report/
68//
69// 上記各事項について不明な点は、ソフトイーサ株式会社までご連絡ください。
70// 連絡先: http://www.softether.co.jp/jp/contact/
71
72// -----------------------------------------------
73// [ChangeLog]
74// 2010.05.20
75//  新規リリース by SoftEther
76// -----------------------------------------------
77
78// Tick64.c
79// 64bit リアルタイムクロックプログラム
80
81#ifdef  WIN32
82#include <windows.h>
83#endif  // WIN32
84
85
86#include <stdio.h>
87#include <stdlib.h>
88#include <string.h>
89#include <wchar.h>
90#include <stdarg.h>
91#include <locale.h>
92#include <time.h>
93#include <errno.h>
94#include <Mayaqua/Mayaqua.h>
95
96static TICK64 *tk64 = NULL;
97static EVENT *halt_tick_event = NULL;
98
99// 高解像度時刻を取得
100UINT64 TickHighres64()
101{
102    UINT64 ret = 0;
103
104#ifdef  OS_WIN32
105
106    ret = (UINT64)(MsGetHiResTimeSpan(MsGetHiResCounter()) * 1000.0f);
107
108#else   // OS_WIN32
109
110    return Tick64();
111
112#endif  // OS_WIN32
113
114    return ret;
115}
116
117// Tick 値を時刻に変換
118UINT64 Tick64ToTime64(UINT64 tick)
119{
120    UINT64 ret = 0;
121    if (tick == 0)
122    {
123        return 0;
124    }
125    LockList(tk64->AdjustTime);
126    {
127        UINT i;
128        for (i = 0;i < LIST_NUM(tk64->AdjustTime);i++)
129        {
130            ADJUST_TIME *t = LIST_DATA(tk64->AdjustTime, i);
131            if (t->Tick <= tick)
132            {
133                ret = t->Time + (tick - t->Tick);
134            }
135        }
136    }
137    UnlockList(tk64->AdjustTime);
138    if (ret == 0)
139    {
140        ret++;
141    }
142    return ret;
143}
144
145// Tick 値を時刻に変換
146UINT64 TickToTime(UINT64 tick)
147{
148    return Tick64ToTime64(tick);
149}
150
151// Tick 値を取得
152UINT64 Tick64()
153{
154#ifdef  OS_WIN32
155    return Win32FastTick64();
156#else   // OS_WIN32
157    UINT64 tick64;
158    if (tk64 == NULL)
159    {
160        return 0;
161    }
162    Lock(tk64->TickLock);
163    {
164        tick64 = tk64->Tick;
165    }
166    Unlock(tk64->TickLock);
167    return tick64;
168#endif  // OS_WIN32
169}
170
171// リアルタイムクロック測定用スレッド
172void Tick64Thread(THREAD *thread, void *param)
173{
174    UINT n = 0;
175    bool first = false;
176    bool create_first_entry = true;
177    UINT tick_span;
178    // 引数チェック
179    if (thread == NULL)
180    {
181        return;
182    }
183
184#ifdef  OS_WIN32
185
186    // Win32 スレッドの優先順位を上げる
187    MsSetThreadPriorityRealtime();
188
189    tick_span = TICK64_SPAN_WIN32;
190
191#else   // OS_WIN32
192
193    // POSIX スレッドの優先順位を上げる
194    UnixSetThreadPriorityRealtime();
195
196    tick_span = TICK64_SPAN;
197
198#endif  // OS_WIN32
199
200    while (true)
201    {
202        UINT tick;
203        UINT64 tick64;
204
205#ifndef OS_WIN32
206        tick = TickRealtime();      // 現在のシステムクロックを取得
207
208        if (tk64->LastTick > tick)
209        {
210            if ((tk64->LastTick - tick) >= (UINT64)0x0fffffff)
211            {
212                // tick の値が 1 周した
213                tk64->RoundCount++;
214            }
215            else
216            {
217                // tick の値が逆戻りした (システム管理者がハードウェア時計を変更した)
218                // 通常これは 1 秒以内の誤差として発生する
219                tick = tk64->LastTick;
220            }
221        }
222        tk64->LastTick = tick;
223
224        tick64 = (UINT64)tk64->RoundCount * (UINT64)4294967296LL + (UINT64)tick;
225
226        Lock(tk64->TickLock);
227        {
228            if (tk64->TickStart == 0)
229            {
230                tk64->TickStart = tick64;
231            }
232            tick64 = tk64->Tick = tick64 - tk64->TickStart + (UINT64)1;
233        }
234        Unlock(tk64->TickLock);
235#else   // OS_WIN32
236        tick64 = Win32FastTick64();
237        tick = (UINT)tick64;
238#endif  // OS_WIN32
239
240        if (create_first_entry)
241        {
242            ADJUST_TIME *t = ZeroMalloc(sizeof(ADJUST_TIME));
243            t->Tick = tick64;
244            t->Time = SystemTime64();
245            tk64->Tick64WithTime64 = tick64;
246            tk64->Time64 = t->Time;
247            Add(tk64->AdjustTime, t);
248
249            // 初期化完了を通知
250            NoticeThreadInit(thread);
251            create_first_entry = false;
252        }
253
254        // 時刻補正
255        n += tick_span;
256        if (n >= 1000 || first == false)
257        {
258            UINT64 now = SystemTime64();
259
260            if (now < tk64->Time64 ||
261                Diff64((now - tk64->Time64) + tk64->Tick64WithTime64, tick64) >= tick_span)
262            {
263                ADJUST_TIME *t = ZeroMalloc(sizeof(ADJUST_TIME));
264                LockList(tk64->AdjustTime);
265                {
266                    t->Tick = tick64;
267                    t->Time = now;
268                    Add(tk64->AdjustTime, t);
269                    Debug("Adjust Time: Tick = %I64u, Time = %I64u\n",
270                        t->Tick, t->Time);
271
272                    // 時計が狂っているシステムでメモリを無限に食わないように
273                    if (LIST_NUM(tk64->AdjustTime) > MAX_ADJUST_TIME)
274                    {
275                        // 2 個目を削除する
276                        ADJUST_TIME *t2 = LIST_DATA(tk64->AdjustTime, 1);
277
278                        Delete(tk64->AdjustTime, t2);
279
280                        Debug("NUM_ADJUST TIME: %u\n", LIST_NUM(tk64->AdjustTime));
281
282                        Free(t2);
283                    }
284                }
285                UnlockList(tk64->AdjustTime);
286                tk64->Time64 = now;
287                tk64->Tick64WithTime64 = tick64;
288            }
289            first = true;
290            n = 0;
291        }
292
293        if (tk64->Halt)
294        {
295            break;
296        }
297
298#ifdef  OS_WIN32
299        Wait(halt_tick_event, tick_span);
300#else   // OS_WIN32
301        SleepThread(tick_span);
302#endif  // OS_WIN32
303    }
304}
305
306// 2 つの 64 bit 整数の差の絶対値を取得
307UINT64 Diff64(UINT64 a, UINT64 b)
308{
309    if (a > b)
310    {
311        return a - b;
312    }
313    else
314    {
315        return b - a;
316    }
317}
318
319// Tick64 の初期化
320void InitTick64()
321{
322    if (tk64 != NULL)
323    {
324        // すでに初期化されている
325        return;
326    }
327
328    halt_tick_event = NewEvent();
329
330    // 構造体の初期化
331    tk64 = ZeroMalloc(sizeof(TICK64));
332    tk64->TickLock = NewLock();
333    tk64->AdjustTime = NewList(NULL);
334
335    // スレッドの作成
336    tk64->Thread = NewThread(Tick64Thread, NULL);
337    WaitThreadInit(tk64->Thread);
338}
339
340// Tick64 の解放
341void FreeTick64()
342{
343    UINT i;
344    if (tk64 == NULL)
345    {
346        // 初期化されていない
347        return;
348    }
349
350    // 終了処理
351    tk64->Halt = true;
352    Set(halt_tick_event);
353    WaitThread(tk64->Thread, INFINITE);
354    ReleaseThread(tk64->Thread);
355
356    // 解放処理
357    for (i = 0;i < LIST_NUM(tk64->AdjustTime);i++)
358    {
359        ADJUST_TIME *t = LIST_DATA(tk64->AdjustTime, i);
360        Free(t);
361    }
362    ReleaseList(tk64->AdjustTime);
363    DeleteLock(tk64->TickLock);
364    Free(tk64);
365    tk64 = NULL;
366
367    ReleaseEvent(halt_tick_event);
368    halt_tick_event = NULL;
369}
370
Note: See TracBrowser for help on using the repository browser.