//
//  SuperTuxKart - a fun racing game with go-kart
//  Copyright (C) 2004-2015 Ingo Ruhnke <grumbel@gmx.de>
//  Copyright (C) 2006-2015 SuperTuxKart-Team
//
//  This program 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; either version 3
//  of the License, or (at your option) any later version.
//
//  This program 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.  See the
//  GNU General Public License for more details.
//
//  You should have received a copy of the GNU General Public License
//  along with this program; if not, write to the Free Software
//  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.

#include "main_loop.hpp"

#ifdef MOBILE_STK
#include "addons/addons_manager.hpp"
#endif
#include "audio/music_manager.hpp"
#include "audio/sfx_manager.hpp"
#include "config/player_manager.hpp"
#include "config/user_config.hpp"
#include "graphics/central_settings.hpp"
#include "graphics/irr_driver.hpp"
#include "graphics/material_manager.hpp"
#include "graphics/sp/sp_texture_manager.hpp"
#include "guiengine/engine.hpp"
#include "guiengine/message_queue.hpp"
#include "guiengine/modaldialog.hpp"
#include "guiengine/screen_keyboard.hpp"
#include "input/input_manager.hpp"
#include "modes/world.hpp"
#include "modes/profile_world.hpp"
#include "network/network_config.hpp"
#include "network/network_timer_synchronizer.hpp"
#include "network/protocols/client_lobby.hpp"
#include "network/protocols/game_protocol.hpp"
#include "network/protocol_manager.hpp"
#include "network/race_event_manager.hpp"
#include "network/rewind_manager.hpp"
#include "network/server.hpp"
#include "network/stk_host.hpp"
#include "online/request_manager.hpp"
#include "race/history.hpp"
#include "race/race_manager.hpp"
#include "states_screens/dialogs/server_info_dialog.hpp"
#include "states_screens/online/server_selection.hpp"
#include "states_screens/state_manager.hpp"
#include "utils/profiler.hpp"
#include "utils/string_utils.hpp"
#include "utils/time.hpp"
#include "utils/translation.hpp"

#ifndef WIN32
#include <unistd.h>
#endif

MainLoop* main_loop = 0;

#ifdef WIN32
LRESULT CALLBACK separateProcessProc(_In_ HWND hwnd, _In_ UINT uMsg, 
                                     _In_ WPARAM wParam, _In_ LPARAM lParam)
{
    if (uMsg == WM_DESTROY)
    {
        PostQuitMessage(0);
        return 0;
    }
    return DefWindowProc(hwnd, uMsg, wParam, lParam);
};
#endif

// ----------------------------------------------------------------------------
MainLoop::MainLoop(unsigned parent_pid, bool download_assets)
        : m_abort(false), m_request_abort(false), m_paused(false),
          m_ticks_adjustment(0), m_parent_pid(parent_pid)
{
    m_curr_time       = 0;
    m_prev_time       = 0;
    m_throttle_fps    = true;
    m_allow_large_dt  = false;
    m_frame_before_loading_world = false;
    m_download_assets = download_assets;
#ifdef WIN32
    if (parent_pid != 0)
    {
        core::stringw class_name = L"separate_process";
        class_name += StringUtils::toWString(GetCurrentProcessId());
        WNDCLASSEX wx = {};
        wx.cbSize = sizeof(WNDCLASSEX);
        wx.lpfnWndProc = separateProcessProc;
        wx.hInstance = GetModuleHandle(0);
        wx.lpszClassName = class_name.c_str();
        if (RegisterClassEx(&wx))
        {
            CreateWindowEx(0, class_name.c_str(), L"stk_server_only",
                0, 0, 0, 0, 0, HWND_MESSAGE, NULL, NULL, NULL);
        }
    }
#endif
}  // MainLoop

//-----------------------------------------------------------------------------
MainLoop::~MainLoop()
{
}   // ~MainLoop

#ifdef MOBILE_STK
//-----------------------------------------------------------------------------
extern "C" void pause_mainloop()
{
    if (!main_loop)
        return;
    if (music_manager)
        music_manager->pauseMusic();
    if (SFXManager::get())
        SFXManager::get()->pauseAll();
    PlayerManager::get()->save();
    if (addons_manager->hasDownloadedIcons())
        addons_manager->saveInstalled();
    Online::RequestManager::get()->setPaused(true);
    IrrlichtDevice* dev = irr_driver->getDevice();
    if (dev)
        dev->resetPaused();
}  // pause_mainloop

//-----------------------------------------------------------------------------
extern "C" void resume_mainloop()
{
    if (!main_loop)
        return;
    IrrlichtDevice* dev = irr_driver->getDevice();
    if (dev)
        dev->resetUnpaused();
    if (music_manager)
        music_manager->resumeMusic();
    if (SFXManager::get())
        SFXManager::get()->resumeAll();
    // Improve rubber banding effects of rewinders when going
    // back to phone, because the smooth timer is paused
    if (World::getWorld() && RewindManager::isEnabled())
        RewindManager::get()->resetSmoothNetworkBody();
    Online::RequestManager::get()->setPaused(false);
}  // resume_mainloop
#endif

//-----------------------------------------------------------------------------
/** Returns the current dt, which guarantees a limited frame rate. If dt is
 *  too low (the frame rate too high), the process will sleep to reach the
 *  maximum frame rate.
 */
float MainLoop::getLimitedDt()
{
    m_prev_time = m_curr_time;

#ifdef IOS_STK
    if (m_paused.load())
    {
        // When iOS apps entering background it should not run any
        // opengl command from apple document, so we stop here
        pause_mainloop();
        while (true)
        {
            // iOS will pause this thread completely when fully in background
            IrrlichtDevice* dev = irr_driver->getDevice();
            bool quit = false;
            if (dev)
                quit = !dev->run();
            if (quit || !m_paused.load())
            {
                if (quit)
                    return 1.0f/60.0f;
                break;
            }
        }
        resume_mainloop();
    }
#endif
    float dt = 0;

    // In profile mode without graphics, run with a fixed dt of 1/60
    if ((ProfileWorld::isProfileMode() && GUIEngine::isNoGraphics()) ||
        UserConfigParams::m_arena_ai_stats)
    {
        return 1.0f/60.0f;
    }

    while( 1 )
    {
        m_curr_time = StkTime::getMonoTimeMs();
        if (m_prev_time > m_curr_time)
        {
            m_prev_time = m_curr_time;
            // If system time adjusted backwards, return fixed dt and
            // resynchronize network timer if exists in client
            if (STKHost::existHost())
            {
#ifndef SERVER_ONLY
                if (UserConfigParams::m_artist_debug_mode &&
                    !GUIEngine::isNoGraphics())
                {
                    core::stringw err = L"System clock running backwards in"
                        " networking game.";
                    MessageQueue::add(MessageQueue::MT_ERROR, err);
                }
#endif
                Log::error("MainLoop", "System clock running backwards in"
                    " networking game.");
                if (STKHost::get()->getNetworkTimerSynchronizer())
                {
                    STKHost::get()->getNetworkTimerSynchronizer()
                        ->resynchroniseTimer();
                }
            }
        }
        dt = (float)(m_curr_time - m_prev_time);
        // On a server (i.e. without graphics) the frame rate can be under
        // 1 ms, i.e. dt = 0. Additionally, the resolution of a sleep
        // statement is not that precise either: if the sleep statement
        // would be consistent < 1ms, but the stk time would increase by
        // 1 ms, the stk clock would be desynchronised from real time
        // (it would go faster), resulting in synchronisation problems
        // with clients (server time is supposed to be behind client time).
        // So we play it safe by adding a loop to make sure at least 1ms
        // (minimum time that can be handled by the integer timer) delay here.
        while (dt == 0)
        {
            StkTime::sleep(1);
            m_curr_time = StkTime::getMonoTimeMs();
            if (m_prev_time > m_curr_time)
            {
                Log::error("MainLopp", "System clock keeps backwards!");
                m_prev_time = m_curr_time;
            }
            dt = (float)(m_curr_time - m_prev_time);
        }

        const World* const world = World::getWorld();
        if (UserConfigParams::m_fps_debug && world)
        {
            const LinearWorld *lw = dynamic_cast<const LinearWorld*>(world);
            if (lw)
            {
                Log::verbose("fps", "time %f distance %f dt %f fps %f",
                             lw->getTime(),
                             lw->getDistanceDownTrackForKart(0, true),
                             dt*0.001f, 1000.0f / dt);
            }
            else
            {
                Log::verbose("fps", "time %f dt %f fps %f",
                             world->getTime(), dt*0.001f, 1000.0f / dt);
            }

        }

        // Don't allow the game to run slower than a certain amount.
        // when the computer can't keep it up, slow down the shown time instead
        // But this can not be done in networking, otherwise the game time on
        // client and server will not be in synch anymore
        if ((!NetworkConfig::get()->isNetworking() || !World::getWorld()) &&
            !m_allow_large_dt)
        {
            /* time 3 internal substeps take */
            const float MAX_ELAPSED_TIME = 3.0f*1.0f / 60.0f*1000.0f;
            if (dt > MAX_ELAPSED_TIME) dt = MAX_ELAPSED_TIME;
        }
        if (!m_throttle_fps || ProfileWorld::isProfileMode()) break;

        // Throttle fps if more than maximum, which can reduce
        // the noise the fan on a graphics card makes.
        // When in menus, reduce FPS much, it's not necessary to push to the
        // maximum for plain menus
#ifdef IOS_STK
        // For iOS devices most at locked at 60, for new iPad Pro supports 120
        // which is currently m_max_fps
        const int max_fps =
            UserConfigParams::m_swap_interval == 2 ? 30 :
            UserConfigParams::m_swap_interval == 1 ? 60 :
            UserConfigParams::m_max_fps;
#else
        const int max_fps = (irr_driver->isRecording() &&
                             UserConfigParams::m_limit_game_fps )
                          ? UserConfigParams::m_record_fps 
                          : ( StateManager::get()->throttleFPS() 
                              ? 60 
                              : UserConfigParams::m_max_fps     );
#endif
        const int current_fps = (int)(1000.0f / dt);
        if (!m_throttle_fps || current_fps <= max_fps ||
            ProfileWorld::isProfileMode()                )  break;

        int wait_time = 1000 / max_fps - 1000 / current_fps;
        if (wait_time < 1) wait_time = 1;

        PROFILER_PUSH_CPU_MARKER("Throttle framerate", 0, 0, 0);
        StkTime::sleep(wait_time);
        PROFILER_POP_CPU_MARKER();
    }   // while(1)

    dt *= 0.001f;
    return dt;
}   // getLimitedDt

//-----------------------------------------------------------------------------
/** Updates all race related objects.
 *  \param ticks Number of ticks (physics steps) to simulate - should be 1.
 *  \param fast_forward If true, then only rewinders in network will be
 *  updated, but not the physics.
 */
void MainLoop::updateRace(int ticks, bool fast_forward)
{
    if (!World::getWorld())  return;   // No race on atm - i.e. we are in menu

    // The race event manager will update world in case of an online race
    if (RaceEventManager::get() && RaceEventManager::get()->isRunning())
        RaceEventManager::get()->update(ticks, fast_forward);
    else
        World::getWorld()->updateWorld(ticks);
}   // updateRace

//-----------------------------------------------------------------------------
/** Run the actual main loop.
 *  The sequnce in which various parts of STK are updated is:
 *  - Determine next time step size (`getLimitedDt`). This takes maximum fps
 *    into account (i.e. sleep if the fps would be too high), and will actually
 *    slow down the in-game clock if the fps are too low (if more than 3/60 of
 *    a second have passed, more than 3 physics time steps would be needed, 
 *    and physics do at most 3 time steps).
 *  - if a race is taking place (i.e. not only a menu being shown), call
 *    `updateRace()`, which is a thin wrapper around a call to
 *    `World::updateWorld()`:
 *    - Update history manager (which will either set the kart position and/or
 *      controls when replaying, or store the current info for a replay).
 *      This is mostly for debugging only (though available even in release
 *      mode).
 *    - Updates Replays - either storing data when not replaying, or
 *      updating kart positions/control when replaying).
 *    - Calls `WorldStatus::update()`, which updates the race state (e.g.
 *      go from 'ready' to 'set' etc), and clock.
 *    - Updates the physics (`Physics::update()`). This will simulate all
 *      physical objects for the specified time with bullet.
 *    - Updates all karts (`Kart::update()`). Obviously the update function
 *      does a lot more than what is described here, this is only supposed to
 *      be a _very_ high level overview:
 *      - Updates its rewinder (to store potentially changed controls
 *        as events) in `KartRewinder::update()`.
 *      - Calls `Moveable::update()`, which takes the new position from
 *        the physics and saves it (and computes dependent values, like
 *        heading, local velocity).
 *      - Updates its controller. This is either:
 *        - an AI using `SkiddingController::update()` (which then will
 *          compute the new controls), or 
 *        - a player controller using `PlayerController::update()`, which will
 *          handle smooth steering (in case of digital input devices steering
 *          is adjusted a bit over time to avoid an instant change from all
 *          left to all right). Input events will be handled when updating
 *          the irrlicht driver later at the end of the main loop.
 *      - Updates kart animation (like rescue, ...) if one is shown atm.
 *      - Update attachments.
 *      - update physics, i.e. taking the current steering and updating
 *        the bullet raycast vehicle with that data. The settings are actually
 *        only used in the next frame when the physics are updated.
 *    - Updates all cameras via `Camera::update()`. The camera position and
 *      rotation is adjusted according to the position etc of the kart (and
 *      special circumstances like rescue, falling).
 *    - Updates all projectiles using the projectile manager. Some of the
 *      projectiles are mostly handled by the physics (e.g. a cake will mainly
 *      check if it's out of bounds), others (like basket ball) do all 
 *      their aiming and movement here.
 *    - Updates the rewind manager to store rewind states.
 *  - Updates the music manager.
 *  - Updates the input manager (which only updates internal time, actual
 *    input handling follows late)
 *  - Updates the wiimote manager. This will read the data of all wiimotes
 *    and feed the corresponding events to the irrlicht event system.
 *  - Updates the STK internal gui engine. This updates all widgets, and
 *    e.g. takes care of the rotation of the karts in the KartSelection
 *    screen using the ModelViewWidget.
 *  - Updates STK's irrlicht driver `IrrDriver::update()`:
 *    - Calls Irrlicht's `beginScene()` .
 *    - Renders the scene (several times with different viewport if
 *      split screen is being used)
 *    - Calls `GUIEngine::render()`, which renders all widgets with the
 *      help of Irrlicht's GUIEnvironment (`drawAll()`). This will also
 *      handle all events, i.e. all input is now handled (e.g. steering,
 *      firing etc are all set in the corresponding karts depending on
 *      user input).
 *    - Calls Irrlicht's `endScene()`
 */
void MainLoop::run()
{
    m_curr_time = StkTime::getMonoTimeMs();
    // DT keeps track of the leftover time, since the race update
    // happens in fixed timesteps
    float left_over_time = 0;

#ifdef WIN32
    HANDLE parent = 0;
    if (m_parent_pid != 0)
    {
        parent = OpenProcess(PROCESS_ALL_ACCESS, FALSE, m_parent_pid);
        if (parent == 0 || parent == INVALID_HANDLE_VALUE)
        {
            Log::warn("MainLoop", "Cannot open parent handle, this child "
                "may not be auto destroyed when parent is terminated");
        }
    }
#endif

    while (!m_abort)
    {
#ifdef WIN32
        if (parent != 0 && parent != INVALID_HANDLE_VALUE)
        {
            MSG msg;
            while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
            {
                TranslateMessage(&msg);
                DispatchMessage(&msg);
                if (msg.message == WM_QUIT)
                {
                    m_request_abort = true;
                }
            }
            // If parent is killed, abort the child main loop too
            if (WaitForSingleObject(parent, 0) != WAIT_TIMEOUT)
            {
                m_request_abort = true;
            }
        }
#else
        // POSIX equivalent
        if (m_parent_pid != 0 && getppid() != (int)m_parent_pid)
        {
            m_request_abort = true;
        }
#endif

        PROFILER_PUSH_CPU_MARKER("Main loop", 0xFF, 0x00, 0xF7);

        left_over_time += getLimitedDt();
        int num_steps   = stk_config->time2Ticks(left_over_time);
        float dt = stk_config->ticks2Time(1);
        left_over_time -= num_steps * dt ;

        // Shutdown next frame if shutdown request is sent while loading the
        // world
        bool was_server = NetworkConfig::get()->isNetworking() &&
            NetworkConfig::get()->isServer();
        if ((STKHost::existHost() && STKHost::get()->requestedShutdown()) ||
            m_request_abort)
        {
            bool was_lan = NetworkConfig::get()->isLAN();
            std::shared_ptr<Server> rejoin_server;
            if (auto cl = LobbyProtocol::get<ClientLobby>())
            {
                if (cl->getJoinedServer()->reconnectWhenQuitLobby())
                    rejoin_server = cl->getJoinedServer();
            }
            bool exist_host = STKHost::existHost();
            core::stringw msg = _("Server connection timed out.");

            if (!m_request_abort)
            {
                if (!GUIEngine::isNoGraphics())
                {
                    SFXManager::get()->quickSound("anvil");
                    if (!STKHost::get()->getErrorMessage().empty())
                    {
                        msg = STKHost::get()->getErrorMessage();
                    }
                }
            }

            if (exist_host == true)
            {
                STKHost::get()->shutdown();
            }

#ifndef SERVER_ONLY
            if (CVS->isGLSL() && !m_download_assets)
            {
                // Flush all command before delete world, avoid later access
                SP::SPTextureManager::get()
                    ->checkForGLCommand(true/*before_scene*/);
                // Reset screen in case the minimap was drawn
                glViewport(0, 0, irr_driver->getActualScreenSize().Width,
                    irr_driver->getActualScreenSize().Height);
            }
#endif

            // In case the user opened a race pause dialog
            GUIEngine::ModalDialog::dismiss();
            GUIEngine::ScreenKeyboard::dismiss();

            if (World::getWorld())
            {
                RaceManager::get()->clearNetworkGrandPrixResult();
                RaceManager::get()->exitRace();
            }

            if (exist_host == true)
            {
                if (!GUIEngine::isNoGraphics())
                {
                    StateManager::get()->resetAndSetStack(
                        NetworkConfig::get()->getResetScreens().data());
                    MessageQueue::add(MessageQueue::MT_ERROR, msg);
                }
                if (rejoin_server)
                {
                    if (was_lan)
                        NetworkConfig::get()->setIsLAN();
                    else
                        NetworkConfig::get()->setIsWAN();
                    NetworkConfig::get()->setIsServer(false);
                    ServerSelection::getInstance()->push();
                    rejoin_server->setReconnectWhenQuitLobby(false);
                    new ServerInfoDialog(rejoin_server);
                }
                else
                    NetworkConfig::get()->unsetNetworking();
            }

            if (m_request_abort)
            {
                m_abort = true;
            }
        }

        if (was_server && !STKHost::existHost())
            m_abort = true;

        if (!m_abort)
        {
            float frame_duration = num_steps * dt;
            if (!GUIEngine::isNoGraphics())
            {
                PROFILER_PUSH_CPU_MARKER("Update race", 0, 255, 255);
                if (World::getWorld())
                    World::getWorld()->updateGraphics(frame_duration);
                PROFILER_POP_CPU_MARKER();

                // Render the previous frame, and also handle all user input.
                PROFILER_PUSH_CPU_MARKER("IrrDriver update", 0x00, 0x00, 0x7F);
                irr_driver->update(frame_duration);
                PROFILER_POP_CPU_MARKER();

                PROFILER_PUSH_CPU_MARKER("Input/GUI", 0x7F, 0x00, 0x00);
                input_manager->update(frame_duration);
                GUIEngine::update(frame_duration);
                PROFILER_POP_CPU_MARKER();
                if (!m_download_assets)
                {
                    PROFILER_PUSH_CPU_MARKER("Music", 0x7F, 0x00, 0x00);
                    SFXManager::get()->update();
                    PROFILER_POP_CPU_MARKER();
                }
            }
            // Some protocols in network will use RequestManager
            if (!m_download_assets)
            {
                PROFILER_PUSH_CPU_MARKER("Database polling update", 0x00, 0x7F, 0x7F);
                Online::RequestManager::get()->update(frame_duration);
                PROFILER_POP_CPU_MARKER();
            }

            m_ticks_adjustment.lock();
            if (m_ticks_adjustment.getData() != 0)
            {
                if (m_ticks_adjustment.getData() > 0)
                {
                    num_steps += m_ticks_adjustment.getData();
                    m_ticks_adjustment.getData() = 0;
                }
                else if (m_ticks_adjustment.getData() < 0)
                {
                    int new_steps = num_steps + m_ticks_adjustment.getData();
                    if (new_steps < 0)
                    {
                        num_steps = 0;
                        m_ticks_adjustment.getData() = new_steps;
                    }
                    else
                    {
                        num_steps = new_steps;
                        m_ticks_adjustment.getData() = 0;
                    }
                }
            }
            m_ticks_adjustment.unlock();

            // Avoid hang when some function in world takes too long time or
            // when leave / come back from android home button
            bool fast_forward = NetworkConfig::get()->isNetworking() &&
                NetworkConfig::get()->isClient() &&
                num_steps > stk_config->time2Ticks(1.0f);
            for (int i = 0; i < num_steps; i++)
            {
                if (World::getWorld() && history->replayHistory())
                {
                    history->updateReplay(
                                       World::getWorld()->getTicksSinceStart());
                }

                PROFILER_PUSH_CPU_MARKER("Protocol manager update",
                                         0x7F, 0x00, 0x7F);
                if (auto pm = ProtocolManager::lock())
                {
                    pm->update(1);
                }
                PROFILER_POP_CPU_MARKER();

                PROFILER_PUSH_CPU_MARKER("Update race", 0, 255, 255);
                if (World::getWorld())
                {
                    updateRace(1, fast_forward);
                }
                PROFILER_POP_CPU_MARKER();

                // We need to check again because update_race may have requested
                // the main loop to abort; and it's not a good idea to continue
                // since the GUI engine is no more to be called then.
                if (m_abort || m_request_abort) 
                    break;

                if (m_frame_before_loading_world)
                {
                    // This will be called when changing introcutscene 1 and 2
                    // in CutsceneWorld::enterRaceOverState
                    // Reset the timer for correct time for cutscene
                    m_frame_before_loading_world = false;
                    m_curr_time = StkTime::getMonoTimeMs();
                    left_over_time = 0.0f;
                    break;
                }

                if (World::getWorld())
                {
                    if (World::getWorld()->getPhase()==WorldStatus::SETUP_PHASE)
                    {
                        // Skip the large num steps contributed by loading time
                        World::getWorld()->updateTime(1);
                        break;
                    }
                    World::getWorld()->updateTime(1);
                }
            }   // for i < num_steps

            // Do it after all pending rewinding is done
            if (World::getWorld() && RewindManager::isEnabled())
                 RewindManager::get()->handleResetSmoothNetworkBody();

            // Handle controller the last to avoid slow PC sending actions too 
            // late
            if (!GUIEngine::isNoGraphics())
            {
                // User aborted (e.g. closed window)
                bool abort = !irr_driver->getDevice()->run();

                if (m_frame_before_loading_world)
                {
                    // irr_driver->getDevice()->run() loads the world
                    m_frame_before_loading_world = false;
                    m_curr_time = StkTime::getMonoTimeMs();
                    left_over_time = 0.0f;
                }

                if (abort)
                {
                    m_request_abort = true;
                }
            }

            if (auto gp = GameProtocol::lock())
            {
                gp->sendActions();
            }
        }
        PROFILER_POP_CPU_MARKER();   // MainLoop pop
        PROFILER_SYNC_FRAME();
    }  // while !m_abort

#ifdef WIN32
    if (parent != 0 && parent != INVALID_HANDLE_VALUE)
        CloseHandle(parent);
#endif

}   // run

// ----------------------------------------------------------------------------
/** Renders the GUI. This function is used during loading a track to get a
 *  responsive GUI, and allow GUI animations (like a progress bar) to be
 *  shown.
 *  \param phase An integer indicated a phase. The maximum number of phases
 *         is used to show a progress bar. The values are between 0 and 8200.
 *  \param loop_index If the call is from a loop, the current loop index.
 *  \param loop_size The number of loop iterations. Used to smooth update
 *         e.g. a progress bar.
 */
void MainLoop::renderGUI(int phase, int loop_index, int loop_size)
{
#ifdef SERVER_ONLY
    return;
#else
    if ((NetworkConfig::get()->isNetworking() &&
        NetworkConfig::get()->isServer()) ||
        GUIEngine::isNoGraphics())
    {
        return;
    }
    // Atm ignore all input when loading only
    irr_driver->getDevice()->setEventReceiver(NULL);
    irr_driver->getDevice()->run();
    irr_driver->getDevice()->setEventReceiver(GUIEngine::EventHandler::get());
    return;
    // Rendering past phase 7000 causes the minimap to not work
    // on higher graphical settings
    if (phase > 7000)
    {
        m_request_abort = !irr_driver->getDevice()->run();
        return;
    }

    uint64_t now = StkTime::getMonoTimeMs();
    float dt = (now - m_curr_time)/1000.0f;
    
    if (dt < 1.0 / 30.0f) return;

    m_curr_time = now;
    
    // TODO: remove debug output
    //Log::verbose("mainloop", "Rendergui t %llu dt %f phase %d  index %d / %d",
    //             now, dt, phase, loop_index, loop_size);

    irr_driver->update(dt, /*is_loading*/true);
    GUIEngine::update(dt);
    m_request_abort = !irr_driver->getDevice()->run();
    
    //TODO: remove debug output
    // uint64_t now2 = StkTime::getMonoTimeMs();
    // Log::verbose("mainloop", "  duration t %llu dt %llu", now, now2-now);
#endif
}   // renderGUI
/* EOF */

#ifdef IOS_STK
// For iOS STK we need to make sure no rendering command is executed after pause
// so we need a handle_app_event callback
#include "SDL_events.h"
extern "C" int handle_app_event(void* userdata, SDL_Event* event)
{
    if (!main_loop)
        return 1;
    switch (event->type)
    {
    case SDL_APP_WILLENTERBACKGROUND:
        main_loop->setPaused(true);
        break;
    case SDL_APP_DIDENTERFOREGROUND:
        main_loop->setPaused(false);
        break;
    default:
        break;
    }
    return 1;
}
#endif
