/* ** eventthread.h ** ** This file is part of mkxp. ** ** Copyright (C) 2013 Jonas Kulla ** ** mkxp 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 2 of the License, or ** (at your option) any later version. ** ** mkxp 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 mkxp. If not, see . */ #ifndef EVENTTHREAD_H #define EVENTTHREAD_H #include "config.h" #include "etc-internal.h" #include "sdl-util.h" #include "keybindings.h" #include #include #include #include #include #include struct RGSSThreadData; typedef struct ALCdevice_struct ALCdevice; struct SDL_Window; union SDL_Event; #define MAX_FINGERS 4 class EventThread { public: struct JoyState { int axes[256]; uint8_t hats[256]; bool buttons[256]; }; struct MouseState { int x, y; bool inWindow; bool buttons[32]; }; struct FingerState { bool down; int x, y; }; struct TouchState { FingerState fingers[MAX_FINGERS]; }; static uint8_t keyStates[SDL_NUM_SCANCODES]; static JoyState joyState; static MouseState mouseState; static TouchState touchState; static bool allocUserEvents(); EventThread(); void process(RGSSThreadData &rtData); void cleanup(); /* Called from RGSS thread */ void requestFullscreenMode(bool mode); void requestWindowResize(int width, int height); void requestShowCursor(bool mode); void requestTerminate(); bool getFullscreen() const; bool getShowCursor() const; void showMessageBox(const char *body, int flags = 0); /* RGSS thread calls this once per frame */ void notifyFrame(); private: static int eventFilter(void *, SDL_Event*); void resetInputStates(); void setFullscreen(SDL_Window *, bool mode); void updateCursorState(bool inWindow); bool fullscreen; bool showCursor; AtomicFlag msgBoxDone; struct { uint64_t lastFrame; uint64_t displayCounter; AtomicFlag sendUpdates; AtomicFlag immInitFlag; AtomicFlag immFiniFlag; double acc; uint32_t accDiv; } fps; }; /* Used to asynchronously inform the RGSS thread * about certain value changes */ template struct UnidirMessage { UnidirMessage() : mutex(SDL_CreateMutex()), current(T()) {} ~UnidirMessage() { SDL_DestroyMutex(mutex); } /* Done from the sending side */ void post(const T &value) { SDL_LockMutex(mutex); changed.set(); current = value; SDL_UnlockMutex(mutex); } /* Done from the receiving side */ bool poll(T &out) const { if (!changed) return false; SDL_LockMutex(mutex); out = current; changed.clear(); SDL_UnlockMutex(mutex); return true; } /* Done from either */ void get(T &out) const { SDL_LockMutex(mutex); out = current; SDL_UnlockMutex(mutex); } private: SDL_mutex *mutex; mutable AtomicFlag changed; T current; }; struct SyncPoint { /* Used by eventFilter to control sleep/wakeup */ void haltThreads(); void resumeThreads(); /* Used by RGSS thread */ bool mainSyncLocked(); void waitMainSync(); /* Used by secondary (audio) threads */ void passSecondarySync(); private: struct Util { Util(); ~Util(); void lock(); void unlock(bool multi); void waitForUnlock(); AtomicFlag locked; SDL_mutex *mut; SDL_cond *cond; }; Util mainSync; Util reply; Util secondSync; }; struct RGSSThreadData { /* Main thread sets this to request RGSS thread to terminate */ AtomicFlag rqTerm; /* In response, RGSS thread sets this to confirm * that it received the request and isn't stuck */ AtomicFlag rqTermAck; /* Set when F12 is pressed */ AtomicFlag rqReset; /* Set when F12 is released */ AtomicFlag rqResetFinish; EventThread *ethread; UnidirMessage windowSizeMsg; UnidirMessage bindingUpdateMsg; SyncPoint syncPoint; const char *argv0; SDL_Window *window; ALCdevice *alcDev; Vec2 sizeResoRatio; Vec2i screenOffset; const int refreshRate; Config config; std::string rgssErrorMsg; RGSSThreadData(EventThread *ethread, const char *argv0, SDL_Window *window, ALCdevice *alcDev, int refreshRate, const Config& newconf) : ethread(ethread), argv0(argv0), window(window), alcDev(alcDev), sizeResoRatio(1, 1), refreshRate(refreshRate), config(newconf) {} }; #endif // EVENTTHREAD_H