]> git.sesse.net Git - casparcg/blob - common/prec_timer.cpp
- Fixed diag to work with new SFML version.
[casparcg] / common / prec_timer.cpp
1 /*
2 * Copyright (c) 2011 Sveriges Television AB <info@casparcg.com>
3 *
4 * This file is part of CasparCG (www.casparcg.com).
5 *
6 * CasparCG is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * CasparCG is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with CasparCG. If not, see <http://www.gnu.org/licenses/>.
18 *
19 * Author: Robert Nagy, ronag89@gmail.com
20 */
21
22 #include "stdafx.h"
23
24 #include "prec_timer.h"
25
26 #include "os/windows/windows.h"
27
28 #include <Mmsystem.h>
29
30 namespace caspar {
31         
32 prec_timer::prec_timer()
33         : time_(0)
34 {
35 }
36
37 // Author: Ryan M. Geiss
38 // http://www.geisswerks.com/ryan/FAQS/timing.html
39 void prec_timer::tick(double interval)
40 {
41         tick_millis(static_cast<int64_t>(interval * 1000.0));
42 }
43
44 void prec_timer::tick_millis(int64_t ticks_to_wait)
45 {
46         auto t = ::timeGetTime();
47
48         if (time_ != 0)
49         {
50                 bool done = 0;
51                 do
52                 {
53                         auto ticks_passed = t - time_;
54                         auto ticks_left = ticks_to_wait - ticks_passed;
55
56                         if (t < time_)    // time wrap
57                                 done = 1;
58                         if (ticks_passed >= ticks_to_wait)
59                                 done = 1;
60
61                         if (!done)
62                         {
63                                 // if > 0.002s left, do Sleep(1), which will actually sleep some 
64                                 //   steady amount, probably 1-2 ms,
65                                 //   and do so in a nice way (cpu meter drops; laptop battery spared).
66                                 // otherwise, do a few Sleep(0)'s, which just give up the timeslice,
67                                 //   but don't really save cpu or battery, but do pass a tiny
68                                 //   amount of time.
69                                 if (ticks_left > 2)
70                                         Sleep(1);
71                                 else
72                                         for (int i = 0; i < 10; ++i)
73                                                 Sleep(0);  // causes thread to give up its timeslice
74                         }
75
76                         t = ::timeGetTime();
77                 } while (!done);
78         }
79
80         time_ = t;
81 }
82
83 }