#include "discord-rpc.h" #include "backoff.h" #include "discord-register.h" #include "rpc_connection.h" #include "serialization.h" #include #include #ifndef DISCORD_DISABLE_IO_THREAD #include #include #endif constexpr size_t MaxMessageSize{16 * 1024}; constexpr size_t MessageQueueSize{8}; struct QueuedMessage { size_t length; char buffer[MaxMessageSize]; }; static RpcConnection* Connection{nullptr}; static DiscordEventHandlers Handlers{}; static std::atomic_bool WasJustConnected{false}; static std::atomic_bool WasJustDisconnected{false}; static std::atomic_bool GotErrorMessage{false}; static std::atomic_bool WasJoinGame{false}; static std::atomic_bool WasSpectateGame{false}; static char JoinGameSecret[256]; static char SpectateGameSecret[256]; static int LastErrorCode{0}; static char LastErrorMessage[256]; static int LastDisconnectErrorCode{0}; static char LastDisconnectErrorMessage[256]; static QueuedMessage SendQueue[MessageQueueSize]{}; static std::atomic_uint SendQueueNextAdd{0}; static std::atomic_uint SendQueueNextSend{0}; static std::atomic_uint SendQueuePendingSends{0}; static Backoff ReconnectTimeMs(500, 60 * 1000); static auto NextConnect{std::chrono::system_clock::now()}; static int Pid{0}; static int Nonce{1}; #ifndef DISCORD_DISABLE_IO_THREAD static std::atomic_bool KeepRunning{true}; static std::mutex WaitForIOMutex; static std::condition_variable WaitForIOActivity; static std::thread IoThread; #endif // DISCORD_DISABLE_IO_THREAD static void UpdateReconnectTime() { NextConnect = std::chrono::system_clock::now() + std::chrono::duration{ReconnectTimeMs.nextDelay()}; } static QueuedMessage* SendQueueGetNextAddMessage() { // if we are falling behind, bail if (SendQueuePendingSends.load() >= MessageQueueSize) { return nullptr; } auto index = (SendQueueNextAdd++) % MessageQueueSize; return &SendQueue[index]; } static QueuedMessage* SendQueueGetNextSendMessage() { auto index = (SendQueueNextSend++) % MessageQueueSize; return &SendQueue[index]; } static void SendQueueCommitMessage() { SendQueuePendingSends++; } extern "C" void Discord_UpdateConnection() { if (!Connection->IsOpen()) { if (std::chrono::system_clock::now() >= NextConnect) { UpdateReconnectTime(); Connection->Open(); } } else { // reads for (;;) { JsonDocument message; if (!Connection->Read(message)) { break; } const char* evtName = nullptr; auto evt = message.FindMember("evt"); if (evt != message.MemberEnd() && evt->value.IsString()) { evtName = evt->value.GetString(); } auto nonce = message.FindMember("nonce"); if (nonce != message.MemberEnd() && nonce->value.IsString()) { // in responses only -- should use to match up response when needed. if (evtName && strcmp(evtName, "ERROR") == 0) { auto data = message.FindMember("data"); LastErrorCode = data->value["code"].GetInt(); StringCopy(LastErrorMessage, data->value["message"].GetString()); GotErrorMessage.store(true); } } else { // should have evt == name of event, optional data if (evtName == nullptr) { continue; } if (strcmp(evtName, "GAME_JOIN") == 0) { auto data = message.FindMember("data"); auto secret = data->value["secret"].GetString(); StringCopy(JoinGameSecret, secret); WasJoinGame.store(true); } else if (strcmp(evtName, "GAME_SPECTATE") == 0) { auto data = message.FindMember("data"); auto secret = data->value["secret"].GetString(); StringCopy(SpectateGameSecret, secret); WasSpectateGame.store(true); } } } // writes while (SendQueuePendingSends.load()) { auto qmessage = SendQueueGetNextSendMessage(); Connection->Write(qmessage->buffer, qmessage->length); --SendQueuePendingSends; } } } #ifndef DISCORD_DISABLE_IO_THREAD void DiscordRpcIo() { const std::chrono::duration maxWait{500LL}; while (KeepRunning.load()) { Discord_UpdateConnection(); std::unique_lock lock(WaitForIOMutex); WaitForIOActivity.wait_for(lock, maxWait); } } #endif void SignalIOActivity() { #ifndef DISCORD_DISABLE_IO_THREAD WaitForIOActivity.notify_all(); #endif } bool RegisterForEvent(const char* evtName) { auto qmessage = SendQueueGetNextAddMessage(); if (qmessage) { qmessage->length = JsonWriteSubscribeCommand(qmessage->buffer, sizeof(qmessage->buffer), Nonce++, evtName); SendQueueCommitMessage(); SignalIOActivity(); return true; } return false; } extern "C" void Discord_Initialize(const char* applicationId, DiscordEventHandlers* handlers, int autoRegister) { if (autoRegister) { Discord_Register(applicationId); } Pid = GetProcessId(); if (handlers) { Handlers = *handlers; } else { Handlers = {}; } Connection = RpcConnection::Create(applicationId); Connection->onConnect = []() { WasJustConnected.exchange(true); ReconnectTimeMs.reset(); if (Handlers.joinGame) { RegisterForEvent("GAME_JOIN"); } if (Handlers.spectateGame) { RegisterForEvent("GAME_SPECTATE"); } }; Connection->onDisconnect = [](int err, const char* message) { LastDisconnectErrorCode = err; StringCopy(LastDisconnectErrorMessage, message); WasJustDisconnected.exchange(true); UpdateReconnectTime(); }; #ifndef DISCORD_DISABLE_IO_THREAD IoThread = std::thread(DiscordRpcIo); #endif } extern "C" void Discord_Shutdown() { Connection->onConnect = nullptr; Connection->onDisconnect = nullptr; Handlers = {}; #ifndef DISCORD_DISABLE_IO_THREAD KeepRunning.exchange(false); SignalIOActivity(); if (IoThread.joinable()) { IoThread.join(); } #endif RpcConnection::Destroy(Connection); } extern "C" void Discord_UpdatePresence(const DiscordRichPresence* presence) { auto qmessage = SendQueueGetNextAddMessage(); if (qmessage) { qmessage->length = JsonWriteRichPresenceObj( qmessage->buffer, sizeof(qmessage->buffer), Nonce++, Pid, presence); SendQueueCommitMessage(); SignalIOActivity(); } } extern "C" void Discord_RunCallbacks() { if (GotErrorMessage.exchange(false) && Handlers.errored) { Handlers.errored(LastErrorCode, LastErrorMessage); } if (WasJustDisconnected.exchange(false) && Handlers.disconnected) { Handlers.disconnected(LastDisconnectErrorCode, LastDisconnectErrorMessage); } if (WasJustConnected.exchange(false) && Handlers.ready) { Handlers.ready(); } if (WasJoinGame.exchange(false) && Handlers.joinGame) { Handlers.joinGame(JoinGameSecret); } if (WasSpectateGame.exchange(false) && Handlers.spectateGame) { Handlers.spectateGame(SpectateGameSecret); } }