2 * This file is part of OpenTTD.
3 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
4 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
8 /** @file video_driver.cpp Common code between video driver implementations. */
10 #include "../stdafx.h"
11 #include "../core/random_func.hpp"
12 #include "../network/network.h"
13 #include "../blitter/factory.hpp"
15 #include "../driver.h"
16 #include "../fontcache.h"
17 #include "../gfx_func.h"
18 #include "../gfxinit.h"
19 #include "../progress.h"
21 #include "../thread.h"
22 #include "../window_func.h"
23 #include "video_driver.hpp"
25 bool _video_hw_accel
; ///< Whether to consider hardware accelerated video drivers on startup.
26 bool _video_vsync
; ///< Whether we should use vsync (only if active video driver supports HW acceleration).
28 void VideoDriver::GameLoop()
30 this->next_game_tick
+= this->GetGameInterval();
32 /* Avoid next_game_tick getting behind more and more if it cannot keep up. */
33 auto now
= std::chrono::steady_clock::now();
34 if (this->next_game_tick
< now
- ALLOWED_DRIFT
* this->GetGameInterval()) this->next_game_tick
= now
;
37 std::lock_guard
<std::mutex
> lock(this->game_state_mutex
);
43 void VideoDriver::GameThread()
48 auto now
= std::chrono::steady_clock::now();
49 if (this->next_game_tick
> now
) {
50 std::this_thread::sleep_for(this->next_game_tick
- now
);
52 /* Ensure we yield this thread if drawings wants to take a lock on
53 * the game state. This is mainly because most OSes have an
54 * optimization that if you unlock/lock a mutex in the same thread
55 * quickly, it will never context switch even if there is another
56 * thread waiting to take the lock on the same mutex. */
57 std::lock_guard
<std::mutex
> lock(this->game_thread_wait_mutex
);
63 * Pause the game-loop for a bit, releasing the game-state lock. This allows,
64 * if the draw-tick requested this, the drawing to happen.
66 void VideoDriver::GameLoopPause()
68 /* If we are not called from the game-thread, ignore this request. */
69 if (std::this_thread::get_id() != this->game_thread
.get_id()) return;
71 this->game_state_mutex
.unlock();
74 /* See GameThread() for more details on this lock. */
75 std::lock_guard
<std::mutex
> lock(this->game_thread_wait_mutex
);
78 this->game_state_mutex
.lock();
81 /* static */ void VideoDriver::GameThreadThunk(VideoDriver
*drv
)
86 void VideoDriver::StartGameThread()
88 if (this->is_game_threaded
) {
89 this->is_game_threaded
= StartNewThread(&this->game_thread
, "ottd:game", &VideoDriver::GameThreadThunk
, this);
92 Debug(driver
, 1, "using {}thread for game-loop", this->is_game_threaded
? "" : "no ");
95 void VideoDriver::StopGameThread()
97 if (!this->is_game_threaded
) return;
99 this->game_thread
.join();
102 void VideoDriver::Tick()
104 if (!this->is_game_threaded
&& std::chrono::steady_clock::now() >= this->next_game_tick
) {
107 /* For things like dedicated server, don't run a separate draw-tick. */
108 if (!this->HasGUI()) {
111 this->next_draw_tick
= this->next_game_tick
;
115 auto now
= std::chrono::steady_clock::now();
116 if (this->HasGUI() && now
>= this->next_draw_tick
) {
117 this->next_draw_tick
+= this->GetDrawInterval();
118 /* Avoid next_draw_tick getting behind more and more if it cannot keep up. */
119 if (this->next_draw_tick
< now
- ALLOWED_DRIFT
* this->GetDrawInterval()) this->next_draw_tick
= now
;
121 /* Locking video buffer can block (especially with vsync enabled), do it before taking game state lock. */
122 this->LockVideoBuffer();
125 /* Tell the game-thread to stop so we can have a go. */
126 std::lock_guard
<std::mutex
> lock_wait(this->game_thread_wait_mutex
);
127 std::lock_guard
<std::mutex
> lock_state(this->game_state_mutex
);
129 /* Keep the interactive randomizer a bit more random by requesting
130 * new values when-ever we can. */
133 this->DrainCommandQueue();
135 while (this->PollEvent()) {}
138 /* Check if the fast-forward button is still pressed. */
139 if (fast_forward_key_pressed
&& !_networking
&& _game_mode
!= GM_MENU
) {
140 ChangeGameSpeed(true);
141 this->fast_forward_via_key
= true;
142 } else if (this->fast_forward_via_key
) {
143 ChangeGameSpeed(false);
144 this->fast_forward_via_key
= false;
149 /* Prevent drawing when switching mode, as windows can be removed when they should still appear. */
150 if (_game_mode
== GM_BOOTSTRAP
|| _switch_mode
== SM_NONE
|| HasModalProgress()) {
154 this->PopulateSystemSprites();
157 this->CheckPaletteAnim();
160 this->UnlockVideoBuffer();
162 /* Wait till the first successful drawing tick before marking the driver as operational. */
163 static bool first_draw_tick
= true;
164 if (first_draw_tick
) {
165 first_draw_tick
= false;
166 DriverFactoryBase::MarkVideoDriverOperational();
171 void VideoDriver::SleepTillNextTick()
173 auto next_tick
= this->next_draw_tick
;
174 auto now
= std::chrono::steady_clock::now();
176 if (!this->is_game_threaded
) {
177 next_tick
= min(next_tick
, this->next_game_tick
);
180 if (next_tick
> now
) {
181 std::this_thread::sleep_for(next_tick
- now
);
186 * Get the caption to use for the game's title bar.
187 * @return The caption.
189 /* static */ std::string
VideoDriver::GetCaption()
191 return fmt::format("OpenTTD {}", _openttd_revision
);