diff --git a/Sources/Jazz2.vcxproj b/Sources/Jazz2.vcxproj index d0977e71..ef0c88af 100644 --- a/Sources/Jazz2.vcxproj +++ b/Sources/Jazz2.vcxproj @@ -384,6 +384,7 @@ + @@ -730,6 +731,7 @@ + diff --git a/Sources/Jazz2.vcxproj.filters b/Sources/Jazz2.vcxproj.filters index c59d9ba4..0bbb2ba8 100644 --- a/Sources/Jazz2.vcxproj.filters +++ b/Sources/Jazz2.vcxproj.filters @@ -1353,6 +1353,9 @@ Header Files\Shared\Threading + + Header Files\Jazz2\UI\Menu + @@ -2252,6 +2255,9 @@ Source Files\Jazz2\Actors + + Source Files\Jazz2\UI\Menu + diff --git a/Sources/Jazz2/Actors/Player.cpp b/Sources/Jazz2/Actors/Player.cpp index 78104bb4..bc8311bf 100644 --- a/Sources/Jazz2/Actors/Player.cpp +++ b/Sources/Jazz2/Actors/Player.cpp @@ -161,7 +161,7 @@ namespace Jazz2::Actors void Player::OnUpdate(float timeMult) { #if defined(DEATH_DEBUG) - if (_levelHandler->PlayerActionPressed(_playerIndex, PlayerActions::ChangeWeapon)) { + if (PreferencesCache::AllowCheats && _levelHandler->PlayerActionPressed(_playerIndex, PlayerActions::ChangeWeapon)) { float moveDistance = (_levelHandler->PlayerActionPressed(_playerIndex, PlayerActions::Run) ? 400.0f : 100.0f); if (_levelHandler->PlayerActionHit(_playerIndex, PlayerActions::Left)) { MoveInstantly(Vector2f(-moveDistance, 0.0f), MoveType::Relative | MoveType::Force); diff --git a/Sources/Jazz2/ILevelHandler.h b/Sources/Jazz2/ILevelHandler.h index 2a28bcba..024b3c6b 100644 --- a/Sources/Jazz2/ILevelHandler.h +++ b/Sources/Jazz2/ILevelHandler.h @@ -39,6 +39,7 @@ namespace Jazz2 virtual Tiles::TileMap* TileMap() = 0; virtual GameDifficulty Difficulty() const = 0; + virtual bool IsPausable() const = 0; virtual bool IsReforged() const = 0; virtual Recti LevelBounds() const = 0; virtual float ElapsedFrames() const = 0; diff --git a/Sources/Jazz2/IRootController.h b/Sources/Jazz2/IRootController.h index 74792ceb..ee9cf710 100644 --- a/Sources/Jazz2/IRootController.h +++ b/Sources/Jazz2/IRootController.h @@ -33,7 +33,7 @@ namespace Jazz2 virtual void ChangeLevel(LevelInitialization&& levelInit) = 0; #if defined(WITH_MULTIPLAYER) - virtual bool ConnectToServer(const char* address, std::uint16_t port) = 0; + virtual bool ConnectToServer(const StringView& address, std::uint16_t port) = 0; virtual bool CreateServer(std::uint16_t port) = 0; #endif diff --git a/Sources/Jazz2/LevelHandler.cpp b/Sources/Jazz2/LevelHandler.cpp index 18b9c547..aaf88c77 100644 --- a/Sources/Jazz2/LevelHandler.cpp +++ b/Sources/Jazz2/LevelHandler.cpp @@ -216,17 +216,20 @@ namespace Jazz2 { float timeMult = theApplication().timeMult(); - UpdatePressedActions(); + if (_pauseMenu == nullptr) { + UpdatePressedActions(); - if (PlayerActionHit(0, PlayerActions::Menu) && _pauseMenu == nullptr && _nextLevelType == ExitType::None) { - PauseGame(); - } + if (PlayerActionHit(0, PlayerActions::Menu) && _nextLevelType == ExitType::None) { + PauseGame(); + } #if defined(DEATH_DEBUG) - if (PlayerActionPressed(0, PlayerActions::ChangeWeapon) && PlayerActionHit(0, PlayerActions::Jump)) { - _cheatsUsed = true; - BeginLevelChange(ExitType::Warp | ExitType::FastTransition, nullptr); - } + if (PreferencesCache::AllowCheats && PlayerActionPressed(0, PlayerActions::ChangeWeapon) && PlayerActionHit(0, PlayerActions::Jump)) { + _cheatsUsed = true; + BeginLevelChange(ExitType::Warp | ExitType::FastTransition, nullptr); + } #endif + } + #if defined(WITH_AUDIO) // Destroy stopped players and resume music after Sugar Rush if (_sugarRushMusic != nullptr && _sugarRushMusic->isStopped()) { @@ -245,7 +248,7 @@ namespace Jazz2 } #endif - if (_pauseMenu == nullptr) { + if (!IsPausable() || _pauseMenu == nullptr) { if (_nextLevelType != ExitType::None) { _nextLevelTime -= timeMult; @@ -291,7 +294,7 @@ namespace Jazz2 } } - if (_difficulty != GameDifficulty::Multiplayer) { + if (/*_difficulty != GameDifficulty::Multiplayer*/true) { if (!_players.empty()) { auto& pos = _players[0]->GetPos(); int32_t tx1 = (int32_t)pos.X / Tiles::TileSet::DefaultTileSize; @@ -1613,22 +1616,26 @@ namespace Jazz2 { // Show in-game pause menu _pauseMenu = std::make_shared(this); - // Prevent updating of all level objects - _rootNode->setUpdateEnabled(false); + if (IsPausable()) { + // Prevent updating of all level objects + _rootNode->setUpdateEnabled(false); + } #if defined(WITH_AUDIO) // Use low-pass filter on music and pause all SFX if (_music != nullptr) { _music->setLowPass(0.1f); } - for (auto& sound : _playingSounds) { - if (sound->isPlaying()) { - sound->pause(); + if (IsPausable()) { + for (auto& sound : _playingSounds) { + if (sound->isPlaying()) { + sound->pause(); + } + } + // If Sugar Rush music is playing, pause it and play normal music instead + if (_sugarRushMusic != nullptr && _music != nullptr) { + _music->play(); } - } - // If Sugar Rush music is playing, pause it and play normal music instead - if (_sugarRushMusic != nullptr && _music != nullptr) { - _music->play(); } #endif } diff --git a/Sources/Jazz2/LevelHandler.h b/Sources/Jazz2/LevelHandler.h index 2f74800e..1bfea8a8 100644 --- a/Sources/Jazz2/LevelHandler.h +++ b/Sources/Jazz2/LevelHandler.h @@ -80,6 +80,10 @@ namespace Jazz2 return _difficulty; } + bool IsPausable() const override { + return true; + } + bool IsReforged() const override { return _isReforged; } diff --git a/Sources/Jazz2/Multiplayer/MultiLevelHandler.h b/Sources/Jazz2/Multiplayer/MultiLevelHandler.h index 87723dc4..5a7c3073 100644 --- a/Sources/Jazz2/Multiplayer/MultiLevelHandler.h +++ b/Sources/Jazz2/Multiplayer/MultiLevelHandler.h @@ -35,6 +35,10 @@ namespace Jazz2::Multiplayer MultiLevelHandler(IRootController* root, NetworkManager* networkManager, const LevelInitialization& levelInit); ~MultiLevelHandler() override; + bool IsPausable() const override { + return false; + } + float GetAmbientLight() const override; void SetAmbientLight(float value) override; diff --git a/Sources/Jazz2/Multiplayer/NetworkManager.cpp b/Sources/Jazz2/Multiplayer/NetworkManager.cpp index 580d89f9..56977d9e 100644 --- a/Sources/Jazz2/Multiplayer/NetworkManager.cpp +++ b/Sources/Jazz2/Multiplayer/NetworkManager.cpp @@ -18,6 +18,8 @@ // Undefine it again after include #undef far +#include + #define MAX_CLIENTS 64 namespace Jazz2::Multiplayer @@ -39,7 +41,7 @@ namespace Jazz2::Multiplayer } } - bool NetworkManager::CreateClient(INetworkHandler* handler, const char* address, std::uint16_t port, std::uint32_t clientData) + bool NetworkManager::CreateClient(INetworkHandler* handler, const StringView& address, std::uint16_t port, std::uint32_t clientData) { if (!_initialized || _host != nullptr) { return false; @@ -53,7 +55,7 @@ namespace Jazz2::Multiplayer _state = NetworkState::Connecting; ENetAddress addr = { }; - enet_address_set_host(&addr, address); + enet_address_set_host(&addr, String::nullTerminatedView(address).data()); addr.port = port; ENetPeer* peer = enet_host_connect(_host, &addr, (std::size_t)NetworkChannel::Count, clientData); diff --git a/Sources/Jazz2/Multiplayer/NetworkManager.h b/Sources/Jazz2/Multiplayer/NetworkManager.h index 67356a19..d9931907 100644 --- a/Sources/Jazz2/Multiplayer/NetworkManager.h +++ b/Sources/Jazz2/Multiplayer/NetworkManager.h @@ -8,6 +8,7 @@ #include "../../nCine/Threading/ThreadSync.h" #include +#include struct _ENetHost; @@ -39,7 +40,7 @@ namespace Jazz2::Multiplayer NetworkManager(); ~NetworkManager(); - bool CreateClient(INetworkHandler* handler, const char* address, std::uint16_t port, std::uint32_t clientData); + bool CreateClient(INetworkHandler* handler, const StringView& address, std::uint16_t port, std::uint32_t clientData); bool CreateServer(INetworkHandler* handler, std::uint16_t port); void Dispose(); diff --git a/Sources/Jazz2/PreferencesCache.cpp b/Sources/Jazz2/PreferencesCache.cpp index 1196995e..d0c4252a 100644 --- a/Sources/Jazz2/PreferencesCache.cpp +++ b/Sources/Jazz2/PreferencesCache.cpp @@ -14,6 +14,9 @@ using namespace Death::IO; namespace Jazz2 { +#if defined(WITH_MULTIPLAYER) + String PreferencesCache::InitialState; +#endif UnlockableEpisodes PreferencesCache::UnlockedEpisodes = UnlockableEpisodes::None; RescaleMode PreferencesCache::ActiveRescaleMode = RescaleMode::None; bool PreferencesCache::EnableFullscreen = false; @@ -296,6 +299,11 @@ namespace Jazz2 } else if (arg == "/mute"_s) { MasterVolume = 0.0f; } +#if defined(WITH_MULTIPLAYER) + else if (InitialState.empty() && (arg == "/server"_s || arg.hasPrefix("/connect:"_s))) { + InitialState = arg; + } +#endif } } diff --git a/Sources/Jazz2/PreferencesCache.h b/Sources/Jazz2/PreferencesCache.h index 18d34a79..770bfeec 100644 --- a/Sources/Jazz2/PreferencesCache.h +++ b/Sources/Jazz2/PreferencesCache.h @@ -79,6 +79,9 @@ namespace Jazz2 static constexpr std::int32_t UnlimitedFps = 0; static constexpr std::int32_t UseVsync = -1; +#if defined(WITH_MULTIPLAYER) + static String InitialState; +#endif static UnlockableEpisodes UnlockedEpisodes; // Graphics @@ -148,9 +151,7 @@ namespace Jazz2 static constexpr float TouchPaddingMultiplier = 0.003f; - /// Deleted copy constructor PreferencesCache(const PreferencesCache&) = delete; - /// Deleted assignment operator PreferencesCache& operator=(const PreferencesCache&) = delete; static String _configPath; diff --git a/Sources/Jazz2/UI/Menu/BeginSection.cpp b/Sources/Jazz2/UI/Menu/BeginSection.cpp index c45bfe86..d967692d 100644 --- a/Sources/Jazz2/UI/Menu/BeginSection.cpp +++ b/Sources/Jazz2/UI/Menu/BeginSection.cpp @@ -333,7 +333,7 @@ namespace Jazz2::UI::Menu // TODO: Multiplayer case (int32_t)Item::TODO_ConnectTo: // TODO: Hardcoded address and port - _root->ConnectToServer("127.0.0.1", 10666); + _root->ConnectToServer("127.0.0.1"_s, 7438); break; case (int32_t)Item::TODO_CreateServer: // TODO: Hardcoded address and port diff --git a/Sources/Jazz2/UI/Menu/IMenuContainer.h b/Sources/Jazz2/UI/Menu/IMenuContainer.h index 4b1a361c..032c36cb 100644 --- a/Sources/Jazz2/UI/Menu/IMenuContainer.h +++ b/Sources/Jazz2/UI/Menu/IMenuContainer.h @@ -42,7 +42,7 @@ namespace Jazz2::UI::Menu virtual void LeaveSection() = 0; virtual void ChangeLevel(Jazz2::LevelInitialization&& levelInit) = 0; #if defined(WITH_MULTIPLAYER) - virtual bool ConnectToServer(const char* address, std::uint16_t port) = 0; + virtual bool ConnectToServer(const StringView& address, std::uint16_t port) = 0; virtual bool CreateServer(std::uint16_t port) = 0; #endif virtual void ApplyPreferencesChanges(ChangedPreferencesType type) = 0; diff --git a/Sources/Jazz2/UI/Menu/InGameMenu.cpp b/Sources/Jazz2/UI/Menu/InGameMenu.cpp index ad6724a3..20751e6f 100644 --- a/Sources/Jazz2/UI/Menu/InGameMenu.cpp +++ b/Sources/Jazz2/UI/Menu/InGameMenu.cpp @@ -235,7 +235,7 @@ namespace Jazz2::UI::Menu } #if defined(WITH_MULTIPLAYER) - bool InGameMenu::ConnectToServer(const char* address, std::uint16_t port) + bool InGameMenu::ConnectToServer(const StringView& address, std::uint16_t port) { return _root->_root->ConnectToServer(address, port); } @@ -264,6 +264,7 @@ namespace Jazz2::UI::Menu } if ((type & ChangedPreferencesType::Language) == ChangedPreferencesType::Language) { + // All sections have to be recreated to load new language _sections.clear(); SwitchToSection(); } diff --git a/Sources/Jazz2/UI/Menu/InGameMenu.h b/Sources/Jazz2/UI/Menu/InGameMenu.h index eac7b866..c16b783f 100644 --- a/Sources/Jazz2/UI/Menu/InGameMenu.h +++ b/Sources/Jazz2/UI/Menu/InGameMenu.h @@ -31,7 +31,7 @@ namespace Jazz2::UI::Menu void LeaveSection() override; void ChangeLevel(Jazz2::LevelInitialization&& levelInit) override; #if defined(WITH_MULTIPLAYER) - bool ConnectToServer(const char* address, std::uint16_t port) override; + bool ConnectToServer(const StringView& address, std::uint16_t port) override; bool CreateServer(std::uint16_t port) override; #endif void ApplyPreferencesChanges(ChangedPreferencesType type) override; diff --git a/Sources/Jazz2/UI/Menu/LoadingSection.cpp b/Sources/Jazz2/UI/Menu/LoadingSection.cpp new file mode 100644 index 00000000..2c1157f6 --- /dev/null +++ b/Sources/Jazz2/UI/Menu/LoadingSection.cpp @@ -0,0 +1,35 @@ +#include "LoadingSection.h" + +namespace Jazz2::UI::Menu +{ + LoadingSection::LoadingSection(const StringView& message) + : _message(message) + { + } + + LoadingSection::LoadingSection(String&& message) + : _message(std::move(message)) + { + + } + + void LoadingSection::OnUpdate(float timeMult) + { + } + + void LoadingSection::OnDraw(Canvas* canvas) + { + Vector2i viewSize = canvas->ViewSize; + Vector2f center = Vector2f(viewSize.X * 0.5f, viewSize.Y * 0.5f); + + constexpr float topLine = 131.0f; + std::int32_t charOffset = 0; + + _root->DrawStringShadow(_message, charOffset, center.X, topLine + 40.0f, IMenuContainer::FontLayer, + Alignment::Top, Font::DefaultColor, 1.2f, 0.4f, 0.6f, 0.6f, 0.6f, 0.9f, 1.2f); + } + + void LoadingSection::OnTouchEvent(const nCine::TouchEvent& event, const Vector2i& viewSize) + { + } +} \ No newline at end of file diff --git a/Sources/Jazz2/UI/Menu/LoadingSection.h b/Sources/Jazz2/UI/Menu/LoadingSection.h new file mode 100644 index 00000000..c67f0d4e --- /dev/null +++ b/Sources/Jazz2/UI/Menu/LoadingSection.h @@ -0,0 +1,20 @@ +#pragma once + +#include "MenuSection.h" + +namespace Jazz2::UI::Menu +{ + class LoadingSection : public MenuSection + { + public: + LoadingSection(const StringView& message); + LoadingSection(String&& message); + + void OnUpdate(float timeMult) override; + void OnDraw(Canvas* canvas) override; + void OnTouchEvent(const nCine::TouchEvent& event, const Vector2i& viewSize) override; + + private: + String _message; + }; +} \ No newline at end of file diff --git a/Sources/Jazz2/UI/Menu/MainMenu.cpp b/Sources/Jazz2/UI/Menu/MainMenu.cpp index ce4dcced..28b9b9c2 100644 --- a/Sources/Jazz2/UI/Menu/MainMenu.cpp +++ b/Sources/Jazz2/UI/Menu/MainMenu.cpp @@ -66,6 +66,30 @@ namespace Jazz2::UI::Menu _canvasOverlay->setParent(nullptr); } + void MainMenu::Reset() + { + bool shouldSwitch = false; + while (!_sections.empty()) { + if (_sections.size() == 1 && dynamic_cast(_sections.back().get())) { + if (shouldSwitch) { + auto& lastSection = _sections.back(); + lastSection->OnShow(this); + + if (_canvasBackground->ViewSize != Vector2i::Zero) { + Recti clipRectangle = lastSection->GetClipRectangle(_canvasBackground->ViewSize); + _upscalePass.SetClipRectangle(clipRectangle); + } + } + return; + } + + _sections.pop_back(); + shouldSwitch = true; + } + + SwitchToSection(); + } + void MainMenu::OnBeginFrame() { float timeMult = theApplication().timeMult(); @@ -327,7 +351,7 @@ namespace Jazz2::UI::Menu } #if defined(WITH_MULTIPLAYER) - bool MainMenu::ConnectToServer(const char* address, std::uint16_t port) + bool MainMenu::ConnectToServer(const StringView& address, std::uint16_t port) { return _root->ConnectToServer(address, port); } @@ -353,6 +377,7 @@ namespace Jazz2::UI::Menu } if ((type & ChangedPreferencesType::Language) == ChangedPreferencesType::Language) { + // All sections have to be recreated to load new language _sections.clear(); SwitchToSection(); } diff --git a/Sources/Jazz2/UI/Menu/MainMenu.h b/Sources/Jazz2/UI/Menu/MainMenu.h index aa694c95..f5b0edb9 100644 --- a/Sources/Jazz2/UI/Menu/MainMenu.h +++ b/Sources/Jazz2/UI/Menu/MainMenu.h @@ -35,6 +35,8 @@ namespace Jazz2::UI::Menu MainMenu(IRootController* root, bool afterIntro); ~MainMenu() override; + void Reset(); + void OnBeginFrame() override; void OnInitializeViewport(int32_t width, int32_t height) override; @@ -46,7 +48,7 @@ namespace Jazz2::UI::Menu void LeaveSection() override; void ChangeLevel(Jazz2::LevelInitialization&& levelInit) override; #if defined(WITH_MULTIPLAYER) - bool ConnectToServer(const char* address, std::uint16_t port) override; + bool ConnectToServer(const StringView& address, std::uint16_t port) override; bool CreateServer(std::uint16_t port) override; #endif void ApplyPreferencesChanges(ChangedPreferencesType type) override; diff --git a/Sources/Main.cpp b/Sources/Main.cpp index e7ed17d4..4787ed79 100644 --- a/Sources/Main.cpp +++ b/Sources/Main.cpp @@ -25,6 +25,7 @@ #include "Jazz2/UI/Cinematics.h" #include "Jazz2/UI/ControlScheme.h" #include "Jazz2/UI/Menu/MainMenu.h" +#include "Jazz2/UI/Menu/LoadingSection.h" #include "Jazz2/UI/Menu/SimpleMessageSection.h" #include "Jazz2/Compatibility/JJ2Anims.h" @@ -76,6 +77,7 @@ class GameEventHandler : public IAppEventHandler, public IInputEventHandler, pub static constexpr std::int32_t DefaultHeight = 405; #if defined(WITH_MULTIPLAYER) + static constexpr std::uint16_t MultiplayerDefaultPort = 7438; static constexpr std::uint32_t MultiplayerProtocolVersion = 1; #endif @@ -98,7 +100,7 @@ class GameEventHandler : public IAppEventHandler, public IInputEventHandler, pub void ChangeLevel(LevelInitialization&& levelInit) override; #if defined(WITH_MULTIPLAYER) - bool ConnectToServer(const char* address, std::uint16_t port) override; + bool ConnectToServer(const StringView& address, std::uint16_t port) override; bool CreateServer(std::uint16_t port) override; bool OnPeerConnected(const Peer& peer, std::uint32_t clientData) override; @@ -134,9 +136,11 @@ class GameEventHandler : public IAppEventHandler, public IInputEventHandler, pub void RefreshCache(); void CheckUpdates(); #endif + static void WriteCacheDescriptor(const StringView& path, std::uint64_t currentVersion, std::int64_t animsModified); static void SaveEpisodeEnd(const LevelInitialization& pendingLevelChange); static void SaveEpisodeContinue(const LevelInitialization& pendingLevelChange); static void UpdateRichPresence(const LevelInitialization& levelInit); + static bool TryParseAddressAndPort(const StringView& input, String& address, std::uint16_t& port); }; void GameEventHandler::OnPreInit(AppConfiguration& config) @@ -218,11 +222,40 @@ void GameEventHandler::OnInit() thread.SetName("Parallel initialization"); # endif +# if defined(WITH_MULTIPLAYER) + if (PreferencesCache::InitialState == "/server"_s) { + thread.Join(); + + auto mainMenu = std::make_unique(this, false); + mainMenu->SwitchToSection(_("Creating server...")); + SetStateHandler(std::move(mainMenu)); + + // TODO: Hardcoded port + CreateServer(MultiplayerDefaultPort); + } else if (PreferencesCache::InitialState.hasPrefix("/connect:"_s)) { + thread.Join(); + + String address; std::uint16_t port; + if (TryParseAddressAndPort(PreferencesCache::InitialState.exceptPrefix(9), address, port)) { + if (port == 0) { + port = MultiplayerDefaultPort; + } + + auto mainMenu = std::make_unique(this, false); + mainMenu->SwitchToSection(_f("Connecting to %s:%u...", address.data(), port)); + SetStateHandler(std::move(mainMenu)); + + ConnectToServer(address.data(), (std::uint16_t)port); + return; + } + } +# endif + SetStateHandler(std::make_unique(this, "intro"_s, [thread](IRootController* root, bool endOfStream) mutable { if ((root->GetFlags() & Jazz2::IRootController::Flags::IsVerified) != Jazz2::IRootController::Flags::IsVerified) { return false; } - + thread.Join(); root->GoToMainMenu(endOfStream); return true; @@ -251,6 +284,33 @@ void GameEventHandler::OnInit() CheckUpdates(); # endif +# if defined(WITH_MULTIPLAYER) + if (PreferencesCache::InitialState == "/server"_s) { + LOGI("Starting server on port %u...", MultiplayerDefaultPort); + + auto mainMenu = std::make_unique(this, false); + mainMenu->SwitchToSection(_("Creating server...")); + SetStateHandler(std::move(mainMenu)); + + // TODO: Hardcoded port + CreateServer(MultiplayerDefaultPort); + } else if (PreferencesCache::InitialState.hasPrefix("/connect:"_s)) { + String address; std::uint16_t port; + if (TryParseAddressAndPort(PreferencesCache::InitialState.exceptPrefix(9), address, port)) { + if (port == 0) { + port = MultiplayerDefaultPort; + } + + auto mainMenu = std::make_unique(this, false); + mainMenu->SwitchToSection(_f("Connecting to %s:%u...", address.data(), port)); + SetStateHandler(std::move(mainMenu)); + + ConnectToServer(address.data(), (std::uint16_t)port); + return; + } + } +# endif + SetStateHandler(std::make_unique(this, "intro"_s, [](IRootController* root, bool endOfStream) { root->GoToMainMenu(endOfStream); return true; @@ -373,9 +433,12 @@ void GameEventHandler::GoToMainMenu(bool afterIntro) #if defined(WITH_MULTIPLAYER) _networkManager = nullptr; #endif - - SetStateHandler(std::make_unique(this, afterIntro)); - UpdateRichPresence({}); + if (auto mainMenu = dynamic_cast(_currentHandler.get())) { + mainMenu->Reset(); + } else { + SetStateHandler(std::make_unique(this, afterIntro)); + UpdateRichPresence({}); + } }); } @@ -457,8 +520,10 @@ void GameEventHandler::ChangeLevel(LevelInitialization&& levelInit) } #if defined(WITH_MULTIPLAYER) -bool GameEventHandler::ConnectToServer(const char* address, std::uint16_t port) +bool GameEventHandler::ConnectToServer(const StringView& address, std::uint16_t port) { + LOGI("Connecting to %s:%u...", address, port); + if (_networkManager == nullptr) { _networkManager = std::make_unique(); } @@ -468,6 +533,8 @@ bool GameEventHandler::ConnectToServer(const char* address, std::uint16_t port) bool GameEventHandler::CreateServer(std::uint16_t port) { + LOGI("Creating server on port %u...", port); + if (_networkManager == nullptr) { _networkManager = std::make_unique(); } @@ -600,23 +667,26 @@ void GameEventHandler::RefreshCache() return; } + constexpr std::uint64_t currentVersion = parseVersion({ NCINE_VERSION, countof(NCINE_VERSION) - 1 }); + auto& resolver = ContentResolver::Get(); + auto cachePath = fs::CombinePath({ resolver.GetCachePath(), "Animations"_s, "cache.index"_s }); // Check cache state { - auto s = fs::Open(fs::CombinePath({ resolver.GetCachePath(), "Animations"_s, "cache.index"_s }), FileAccessMode::Read); + auto s = fs::Open(cachePath, FileAccessMode::Read); if (s->GetSize() < 16) { goto RecreateCache; } - uint64_t signature = s->ReadValue(); - uint8_t fileType = s->ReadValue(); - uint16_t version = s->ReadValue(); + std::uint64_t signature = s->ReadValue(); + std::uint8_t fileType = s->ReadValue(); + std::uint16_t version = s->ReadValue(); if (signature != 0x2095A59FF0BFBBEF || fileType != ContentResolver::CacheIndexFile || version != Compatibility::JJ2Anims::CacheVersion) { goto RecreateCache; } - uint8_t flags = s->ReadValue(); + std::uint8_t flags = s->ReadValue(); if ((flags & 0x01) == 0x01) { // Don't overwrite cache LOGI("Cache is protected"); @@ -628,20 +698,39 @@ void GameEventHandler::RefreshCache() if (!fs::IsReadableFile(animsPath)) { animsPath = fs::FindPathCaseInsensitive(fs::CombinePath(resolver.GetSourcePath(), "AnimsSw.j2a"_s)); } - int64_t animsCached = s->ReadValue(); - int64_t animsModified = fs::GetLastModificationTime(animsPath).GetValue(); + std::int64_t animsCached = s->ReadValue(); + std::int64_t animsModified = fs::GetLastModificationTime(animsPath).GetValue(); if (animsModified != 0 && animsCached != animsModified) { goto RecreateCache; } // If some events were added, recreate cache - uint16_t eventTypeCount = s->ReadValue(); - if (eventTypeCount != (uint16_t)EventType::Count) { + std::uint16_t eventTypeCount = s->ReadValue(); + if (eventTypeCount != (std::uint16_t)EventType::Count) { goto RecreateCache; } // Cache is up-to-date - LOGI("Cache is already up-to-date"); + std::uint64_t lastVersion = s->ReadValue(); + + // Close the file, so it can be writable for possible update + s = nullptr; + + if (currentVersion != lastVersion) { + if ((lastVersion & 0xFFFFFFFFULL) == 0x0FFFFFFFULL) { + LOGI("Cache is already up-to-date, but created in experimental build v%i.%i.0", (lastVersion >> 48) & 0xFFFFULL, (lastVersion >> 32) & 0xFFFFULL); + } else { + LOGI("Cache is already up-to-date, but created in different build v%i.%i.%i", (lastVersion >> 48) & 0xFFFFULL, (lastVersion >> 32) & 0xFFFFULL, lastVersion & 0xFFFFFFFFULL); + } + + WriteCacheDescriptor(cachePath, currentVersion, animsModified); + + LOGI("Pruning binary shader cache..."); + RenderResources::binaryShaderCache().prune(); + } else { + LOGI("Cache is already up-to-date"); + } + _flags |= Flags::IsVerified | Flags::IsPlayable; return; } @@ -668,18 +757,13 @@ void GameEventHandler::RefreshCache() RefreshCacheLevels(); - // Create cache index - auto so = fs::Open(fs::CombinePath({ resolver.GetCachePath(), "Animations"_s, "cache.index"_s }), FileAccessMode::Write); + LOGI("Cache was recreated"); + std::int64_t animsModified = fs::GetLastModificationTime(animsPath).GetValue(); + WriteCacheDescriptor(cachePath, currentVersion, animsModified); - so->WriteValue(0x2095A59FF0BFBBEF); // Signature - so->WriteValue(ContentResolver::CacheIndexFile); - so->WriteValue(Compatibility::JJ2Anims::CacheVersion); - so->WriteValue(0x00); // Flags - int64_t animsModified = fs::GetLastModificationTime(animsPath).GetValue(); - so->WriteValue(animsModified); - so->WriteValue((uint16_t)EventType::Count); + LOGI("Pruning binary shader cache..."); + RenderResources::binaryShaderCache().prune(); - LOGI("Cache was recreated"); _flags |= Flags::IsVerified | Flags::IsPlayable; } @@ -921,9 +1005,18 @@ void GameEventHandler::RefreshCacheLevels() } } } - - LOGI("Pruning binary shader cache..."); - RenderResources::binaryShaderCache().prune(); +} + +void GameEventHandler::WriteCacheDescriptor(const StringView& path, std::uint64_t currentVersion, std::int64_t animsModified) +{ + auto so = fs::Open(path, FileAccessMode::Write); + so->WriteValue(0x2095A59FF0BFBBEF); // Signature + so->WriteValue(ContentResolver::CacheIndexFile); + so->WriteValue(Compatibility::JJ2Anims::CacheVersion); + so->WriteValue(0x00); // Flags + so->WriteValue(animsModified); + so->WriteValue((std::uint16_t)EventType::Count); + so->WriteValue(currentVersion); } void GameEventHandler::CheckUpdates() @@ -1102,7 +1195,7 @@ void GameEventHandler::CheckUpdates() Http::Request req(url, Http::InternetProtocol::V4); Http::Response resp = req.Send("GET"_s, std::chrono::seconds(10)); if (resp.Status.Code == Http::HttpStatus::Ok && !resp.Body.empty() && resp.Body.size() < sizeof(_newestVersion) - 1) { - std::uint64_t currentVersion = parseVersion(NCINE_VERSION); + constexpr std::uint64_t currentVersion = parseVersion({ NCINE_VERSION, countof(NCINE_VERSION) - 1 }); std::uint64_t latestVersion = parseVersion(StringView(reinterpret_cast(resp.Body.data()), resp.Body.size())); if (currentVersion < latestVersion) { std::memcpy(_newestVersion, resp.Body.data(), resp.Body.size()); @@ -1272,6 +1365,23 @@ void GameEventHandler::UpdateRichPresence(const LevelInitialization& levelInit) #endif } +bool GameEventHandler::TryParseAddressAndPort(const StringView& input, String& address, std::uint16_t& port) +{ + auto portSep = input.findLast(':'); + if (portSep == nullptr) { + return false; + } + + address = String(input.prefix(portSep.begin())); + if (address.empty()) { + return false; + } + + auto portString = input.suffix(portSep.begin() + 1); + port = (std::uint16_t)stou32(portString.data(), portString.size()); + return true; +} + #if defined(DEATH_TARGET_ANDROID) std::unique_ptr CreateAppEventHandler() { diff --git a/Sources/Shared/Containers/ArrayView.h b/Sources/Shared/Containers/ArrayView.h index 3c879364..7579ea67 100644 --- a/Sources/Shared/Containers/ArrayView.h +++ b/Sources/Shared/Containers/ArrayView.h @@ -54,7 +54,10 @@ namespace Death::Containers public: typedef T Type; - constexpr /*implicit*/ ArrayView(std::nullptr_t = nullptr) noexcept : _data{}, _size {} {} + /* To avoid ambiguity in certain cases of passing 0 to overloads that take either a ArrayView or std::size_t */ + template::value>::type> constexpr /*implicit*/ ArrayView(U) noexcept : _data{}, _size{} {} + + constexpr /*implicit*/ ArrayView() noexcept : _data{}, _size{} {} constexpr /*implicit*/ ArrayView(T* data, std::size_t size) noexcept : _data(data), _size(size) {} @@ -172,7 +175,10 @@ namespace Death::Containers public: typedef void Type; - constexpr /*implicit*/ ArrayView(std::nullptr_t = nullptr) noexcept : _data{}, _size{} {} + /* To avoid ambiguity in certain cases of passing 0 to overloads that take either a ArrayView or std::size_t */ + template::value>::type> constexpr /*implicit*/ ArrayView(U) noexcept : _data{}, _size{} {} + + constexpr /*implicit*/ ArrayView() noexcept : _data{}, _size{} {} constexpr /*implicit*/ ArrayView(void* data, std::size_t size) noexcept : _data(data), _size(size) {} @@ -233,7 +239,10 @@ namespace Death::Containers public: typedef const void Type; - constexpr /*implicit*/ ArrayView(std::nullptr_t = nullptr) noexcept : _data{}, _size{} {} + /* To avoid ambiguity in certain cases of passing 0 to overloads that take either a ArrayView or std::size_t */ + template::value>::type> constexpr /*implicit*/ ArrayView(U) noexcept : _data{}, _size{} {} + + constexpr /*implicit*/ ArrayView() noexcept : _data{}, _size{} {} constexpr /*implicit*/ ArrayView(const void* data, std::size_t size) noexcept : _data(data), _size(size) {} @@ -356,7 +365,10 @@ namespace Death::Containers Size = size_ }; - constexpr /*implicit*/ StaticArrayView(std::nullptr_t = nullptr) noexcept : _data{} {} + /* To avoid ambiguity in certain cases of passing 0 to overloads that take either a ArrayView or std::size_t */ + template::value>::type> constexpr /*implicit*/ StaticArrayView(U) noexcept : _data{} {} + + constexpr /*implicit*/ StaticArrayView() noexcept : _data{} {} template::value && !std::is_same::value>::type> constexpr explicit StaticArrayView(U data) noexcept : _data(data) {} diff --git a/Sources/Shared/Containers/String.cpp b/Sources/Shared/Containers/String.cpp index 827df3ea..d2bb044d 100644 --- a/Sources/Shared/Containers/String.cpp +++ b/Sources/Shared/Containers/String.cpp @@ -27,7 +27,7 @@ namespace Death::Containers out._large.size |= std::size_t(view.flags() & StringViewFlags::Global); return out; } - return String { view }; + return String{view}; } String String::nullTerminatedView(AllocatedInitT, StringView view) { @@ -36,7 +36,7 @@ namespace Death::Containers out._large.size |= std::size_t(view.flags() & StringViewFlags::Global); return out; } - return String { AllocatedInit, view }; + return String{AllocatedInit, view}; } String String::nullTerminatedGlobalView(StringView view) { @@ -45,7 +45,7 @@ namespace Death::Containers out._large.size |= std::size_t(StringViewFlags::Global); return out; } - return String { view }; + return String{view}; } String String::nullTerminatedGlobalView(AllocatedInitT, StringView view) { @@ -54,7 +54,7 @@ namespace Death::Containers out._large.size |= std::size_t(StringViewFlags::Global); return out; } - return String { AllocatedInit, view }; + return String{AllocatedInit, view}; } inline void String::construct(NoInitT, const std::size_t size) { @@ -112,7 +112,7 @@ namespace Death::Containers String::String(const ArrayView view) : String{view.data(), view.size()} {} - String::String(const char* const data) : String{data, data ? std::strlen(data) : 0} {} + String::String(std::nullptr_t, std::nullptr_t, std::nullptr_t, const char* const data) : String{data, data ? std::strlen(data) : 0} {} String::String(const char* const data, const std::size_t size) : _large{} @@ -120,7 +120,7 @@ namespace Death::Containers #if defined(DEATH_TARGET_32BIT) // Compared to StringView construction which happens a lot this shouldn't, and the chance of strings > 1 GB on 32-bit // is rare but possible and thus worth checking even in release - DEATH_ASSERT(size < std::size_t { 1 } << (sizeof(std::size_t) * 8 - 2), , "Containers::String: String expected to be smaller than 2^%zu bytes, got %zu", sizeof(std::size_t) * 8 - 2, size); + DEATH_ASSERT(size < std::size_t{1} << (sizeof(std::size_t) * 8 - 2), , "Containers::String: String expected to be smaller than 2^%zu bytes, got %zu", sizeof(std::size_t) * 8 - 2, size); #endif DEATH_ASSERT(data || size == 0, , "Containers::String: Received a null string of size %zu", size); @@ -193,7 +193,7 @@ namespace Death::Containers // Compared to StringView construction which happens a lot this shouldn't, the chance of strings > 1 GB on 32-bit // is rare but possible and thus worth checking even in release; but most importantly checking for null // termination outweighs potential speed issues - DEATH_ASSERT(size < std::size_t { 1 } << (sizeof(std::size_t) * 8 - 2), , "Containers::String: String expected to be smaller than 2^%zu bytes, got %zu", sizeof(std::size_t) * 8 - 2, size); + DEATH_ASSERT(size < std::size_t{1} << (sizeof(std::size_t) * 8 - 2), , "Containers::String: String expected to be smaller than 2^%zu bytes, got %zu", sizeof(std::size_t) * 8 - 2, size); DEATH_ASSERT(data && !data[size], , "Containers::String: Can only take ownership of a non-null null-terminated array"); _large.data = data; @@ -213,7 +213,7 @@ namespace Death::Containers { // Compared to StringView construction which happens a lot this shouldn't, and the chance of strings > 1 GB on 32-bit // is rare but possible and thus worth checking even in release - DEATH_ASSERT(size < std::size_t { 1 } << (sizeof(std::size_t) * 8 - 2), , "Containers::String: String expected to be smaller than 2^%zu bytes, got %zu", sizeof(std::size_t) * 8 - 2, size); + DEATH_ASSERT(size < std::size_t{1} << (sizeof(std::size_t) * 8 - 2), , "Containers::String: String expected to be smaller than 2^%zu bytes, got %zu", sizeof(std::size_t) * 8 - 2, size); if (size < Implementation::SmallStringSize) { // Everything already zero-init'd in the constructor init list @@ -229,7 +229,7 @@ namespace Death::Containers { // Compared to StringView construction which happens a lot this shouldn't, and the chance of strings > 1 GB on 32-bit // is rare but possible and thus worth checking even in release - DEATH_ASSERT(size < std::size_t { 1 } << (sizeof(std::size_t) * 8 - 2), , "Containers::String: String expected to be smaller than 2^%zu bytes, got %zu", sizeof(std::size_t) * 8 - 2, size); + DEATH_ASSERT(size < std::size_t{1} << (sizeof(std::size_t) * 8 - 2), , "Containers::String: String expected to be smaller than 2^%zu bytes, got %zu", sizeof(std::size_t) * 8 - 2, size); construct(NoInit, size); } @@ -299,11 +299,11 @@ namespace Death::Containers if (_small.size & Implementation::SmallStringBit) { const std::size_t size = _small.size & ~SmallSizeMask; // Allocate the output including a null terminator at the end, but don't include it in the size - out = Array { Array{NoInit, size + 1}.release(), size }; + out = Array{Array{NoInit, size + 1}.release(), size}; out[size] = '\0'; std::memcpy(out.data(), _small.data, size); } else { - out = Array { _large.data, _large.size & ~LargeSizeMask, deleter() }; + out = Array{_large.data, _large.size & ~LargeSizeMask, deleter()}; } // Same as in release(). Create a zero-size small string to fullfil the guarantee of data() being always non-null @@ -461,119 +461,119 @@ namespace Death::Containers } MutableStringView String::slice(char* const begin, char* const end) { - return MutableStringView { *this }.slice(begin, end); + return MutableStringView{*this}.slice(begin, end); } StringView String::slice(const char* const begin, const char* const end) const { - return StringView { *this }.slice(begin, end); + return StringView{*this}.slice(begin, end); } MutableStringView String::slice(const std::size_t begin, const std::size_t end) { - return MutableStringView { *this }.slice(begin, end); + return MutableStringView{*this}.slice(begin, end); } StringView String::slice(const std::size_t begin, const std::size_t end) const { - return StringView { *this }.slice(begin, end); + return StringView{*this}.slice(begin, end); } MutableStringView String::prefix(char* const end) { - return MutableStringView { *this }.prefix(end); + return MutableStringView{*this}.prefix(end); } StringView String::prefix(const char* const end) const { - return StringView { *this }.prefix(end); + return StringView{*this}.prefix(end); } MutableStringView String::suffix(char* const begin) { - return MutableStringView { *this }.suffix(begin); + return MutableStringView{*this}.suffix(begin); } StringView String::suffix(const char* const begin) const { - return StringView { *this }.suffix(begin); + return StringView{*this}.suffix(begin); } MutableStringView String::prefix(const std::size_t count) { - return MutableStringView { *this }.prefix(count); + return MutableStringView{*this}.prefix(count); } StringView String::prefix(const std::size_t count) const { - return StringView { *this }.prefix(count); + return StringView{*this}.prefix(count); } MutableStringView String::exceptPrefix(const std::size_t count) { - return MutableStringView { *this }.exceptPrefix(count); + return MutableStringView{*this}.exceptPrefix(count); } StringView String::exceptPrefix(const std::size_t count) const { - return StringView { *this }.exceptPrefix(count); + return StringView{*this}.exceptPrefix(count); } MutableStringView String::exceptSuffix(const std::size_t count) { - return MutableStringView { *this }.exceptSuffix(count); + return MutableStringView{*this}.exceptSuffix(count); } StringView String::exceptSuffix(const std::size_t count) const { - return StringView { *this }.exceptSuffix(count); + return StringView{*this}.exceptSuffix(count); } Array String::split(const char delimiter) { - return MutableStringView { *this }.split(delimiter); + return MutableStringView{*this}.split(delimiter); } Array String::split(const char delimiter) const { - return StringView { *this }.split(delimiter); + return StringView{*this}.split(delimiter); } Array String::split(const StringView delimiter) { - return MutableStringView { *this }.split(delimiter); + return MutableStringView{*this}.split(delimiter); } Array String::split(const StringView delimiter) const { - return StringView { *this }.split(delimiter); + return StringView{*this}.split(delimiter); } Array String::splitWithoutEmptyParts(const char delimiter) { - return MutableStringView { *this }.splitWithoutEmptyParts(delimiter); + return MutableStringView{*this}.splitWithoutEmptyParts(delimiter); } Array String::splitWithoutEmptyParts(const char delimiter) const { - return StringView { *this }.splitWithoutEmptyParts(delimiter); + return StringView{*this}.splitWithoutEmptyParts(delimiter); } Array String::splitOnAnyWithoutEmptyParts(const StringView delimiters) { - return MutableStringView { *this }.splitOnAnyWithoutEmptyParts(delimiters); + return MutableStringView{*this}.splitOnAnyWithoutEmptyParts(delimiters); } Array String::splitOnAnyWithoutEmptyParts(const StringView delimiters) const { - return StringView { *this }.splitOnAnyWithoutEmptyParts(delimiters); + return StringView{*this}.splitOnAnyWithoutEmptyParts(delimiters); } Array String::splitOnWhitespaceWithoutEmptyParts() { - return MutableStringView { *this }.splitOnWhitespaceWithoutEmptyParts(); + return MutableStringView{*this}.splitOnWhitespaceWithoutEmptyParts(); } Array String::splitOnWhitespaceWithoutEmptyParts() const { - return StringView { *this }.splitOnWhitespaceWithoutEmptyParts(); + return StringView{*this}.splitOnWhitespaceWithoutEmptyParts(); } StaticArray<3, MutableStringView> String::partition(const char separator) { - return MutableStringView { *this }.partition(separator); + return MutableStringView{*this}.partition(separator); } StaticArray<3, StringView> String::partition(const char separator) const { - return StringView { *this }.partition(separator); + return StringView{*this}.partition(separator); } StaticArray<3, MutableStringView> String::partition(const StringView separator) { - return MutableStringView { *this }.partition(separator); + return MutableStringView{*this}.partition(separator); } StaticArray<3, StringView> String::partition(const StringView separator) const { - return StringView { *this }.partition(separator); + return StringView{*this}.partition(separator); } String String::join(const ArrayView strings) const { - return StringView { *this }.join(strings); + return StringView{*this}.join(strings); } String String::join(const std::initializer_list strings) const { @@ -582,7 +582,7 @@ namespace Death::Containers } String String::joinWithoutEmptyParts(const ArrayView strings) const { - return StringView { *this }.joinWithoutEmptyParts(strings); + return StringView{*this}.joinWithoutEmptyParts(strings); } String String::joinWithoutEmptyParts(const std::initializer_list strings) const { @@ -591,199 +591,199 @@ namespace Death::Containers } bool String::hasPrefix(const StringView prefix) const { - return StringView { *this }.hasPrefix(prefix); + return StringView{*this}.hasPrefix(prefix); } bool String::hasPrefix(const char prefix) const { - return StringView { *this }.hasPrefix(prefix); + return StringView{*this}.hasPrefix(prefix); } bool String::hasSuffix(const StringView suffix) const { - return StringView { *this }.hasSuffix(suffix); + return StringView{*this}.hasSuffix(suffix); } bool String::hasSuffix(const char suffix) const { - return StringView { *this }.hasSuffix(suffix); + return StringView{*this}.hasSuffix(suffix); } MutableStringView String::exceptPrefix(const StringView prefix) { - return MutableStringView { *this }.exceptPrefix(prefix); + return MutableStringView{*this}.exceptPrefix(prefix); } StringView String::exceptPrefix(const StringView prefix) const { - return StringView { *this }.exceptPrefix(prefix); + return StringView{*this}.exceptPrefix(prefix); } MutableStringView String::exceptSuffix(const StringView suffix) { - return MutableStringView { *this }.exceptSuffix(suffix); + return MutableStringView{*this}.exceptSuffix(suffix); } StringView String::exceptSuffix(const StringView suffix) const { - return StringView { *this }.exceptSuffix(suffix); + return StringView{*this}.exceptSuffix(suffix); } MutableStringView String::trimmed(const StringView characters) { - return MutableStringView { *this }.trimmed(characters); + return MutableStringView{*this}.trimmed(characters); } StringView String::trimmed(const StringView characters) const { - return StringView { *this }.trimmed(characters); + return StringView{*this}.trimmed(characters); } MutableStringView String::trimmed() { - return MutableStringView { *this }.trimmed(); + return MutableStringView{*this}.trimmed(); } StringView String::trimmed() const { - return StringView { *this }.trimmed(); + return StringView{*this}.trimmed(); } MutableStringView String::trimmedPrefix(const StringView characters) { - return MutableStringView { *this }.trimmedPrefix(characters); + return MutableStringView{*this}.trimmedPrefix(characters); } StringView String::trimmedPrefix(const StringView characters) const { - return StringView { *this }.trimmedPrefix(characters); + return StringView{*this}.trimmedPrefix(characters); } MutableStringView String::trimmedPrefix() { - return MutableStringView { *this }.trimmedPrefix(); + return MutableStringView{*this}.trimmedPrefix(); } StringView String::trimmedPrefix() const { - return StringView { *this }.trimmedPrefix(); + return StringView{*this}.trimmedPrefix(); } MutableStringView String::trimmedSuffix(const StringView characters) { - return MutableStringView { *this }.trimmedSuffix(characters); + return MutableStringView{*this}.trimmedSuffix(characters); } StringView String::trimmedSuffix(const StringView characters) const { - return StringView { *this }.trimmedSuffix(characters); + return StringView{*this}.trimmedSuffix(characters); } MutableStringView String::trimmedSuffix() { - return MutableStringView { *this }.trimmedSuffix(); + return MutableStringView{*this}.trimmedSuffix(); } StringView String::trimmedSuffix() const { - return StringView { *this }.trimmedSuffix(); + return StringView{*this}.trimmedSuffix(); } MutableStringView String::find(const StringView substring) { // Calling straight into the concrete implementation to reduce call stack depth - return MutableStringView { *this }.findOr(substring, nullptr); + return MutableStringView{*this}.findOr(substring, nullptr); } StringView String::find(const StringView substring) const { // Calling straight into the concrete implementation to reduce call stack depth - return StringView { *this }.findOr(substring, nullptr); + return StringView{*this}.findOr(substring, nullptr); } MutableStringView String::find(const char character) { // Calling straight into the concrete implementation to reduce call stack depth - return MutableStringView { *this }.findOr(character, nullptr); + return MutableStringView{*this}.findOr(character, nullptr); } StringView String::find(const char character) const { // Calling straight into the concrete implementation to reduce call stack depth - return StringView { *this }.findOr(character, nullptr); + return StringView{*this}.findOr(character, nullptr); } MutableStringView String::findOr(const StringView substring, char* const fail) { - return MutableStringView { *this }.findOr(substring, fail); + return MutableStringView{*this}.findOr(substring, fail); } StringView String::findOr(const StringView substring, const char* const fail) const { - return StringView { *this }.findOr(substring, fail); + return StringView{*this}.findOr(substring, fail); } MutableStringView String::findOr(const char character, char* const fail) { - return MutableStringView { *this }.findOr(character, fail); + return MutableStringView{*this}.findOr(character, fail); } StringView String::findOr(const char character, const char* const fail) const { - return StringView { *this }.findOr(character, fail); + return StringView{*this}.findOr(character, fail); } MutableStringView String::findLast(const StringView substring) { // Calling straight into the concrete implementation to reduce call stack depth - return MutableStringView { *this }.findLastOr(substring, nullptr); + return MutableStringView{*this}.findLastOr(substring, nullptr); } StringView String::findLast(const StringView substring) const { // Calling straight into the concrete implementation to reduce call stack depth - return StringView { *this }.findLastOr(substring, nullptr); + return StringView{*this}.findLastOr(substring, nullptr); } MutableStringView String::findLast(const char character) { // Calling straight into the concrete implementation to reduce call stack depth - return MutableStringView { *this }.findLastOr(character, nullptr); + return MutableStringView{*this}.findLastOr(character, nullptr); } StringView String::findLast(const char character) const { /* Calling straight into the concrete implementation to reduce call stack depth */ - return StringView { *this }.findLastOr(character, nullptr); + return StringView{*this}.findLastOr(character, nullptr); } MutableStringView String::findLastOr(const StringView substring, char* const fail) { - return MutableStringView { *this }.findLastOr(substring, fail); + return MutableStringView{*this}.findLastOr(substring, fail); } StringView String::findLastOr(const StringView substring, const char* const fail) const { - return StringView { *this }.findLastOr(substring, fail); + return StringView{*this}.findLastOr(substring, fail); } MutableStringView String::findLastOr(const char character, char* const fail) { - return MutableStringView { *this }.findLastOr(character, fail); + return MutableStringView{*this}.findLastOr(character, fail); } StringView String::findLastOr(const char character, const char* const fail) const { - return StringView { *this }.findLastOr(character, fail); + return StringView{*this}.findLastOr(character, fail); } bool String::contains(const StringView substring) const { - return StringView { *this }.contains(substring); + return StringView{*this}.contains(substring); } bool String::contains(const char character) const { - return StringView { *this }.contains(character); + return StringView{*this}.contains(character); } MutableStringView String::findAny(const StringView characters) { - return MutableStringView { *this }.findAny(characters); + return MutableStringView{*this}.findAny(characters); } StringView String::findAny(const StringView characters) const { - return StringView { *this }.findAny(characters); + return StringView{*this}.findAny(characters); } MutableStringView String::findAnyOr(const StringView characters, char* fail) { - return MutableStringView { *this }.findAnyOr(characters, fail); + return MutableStringView{*this}.findAnyOr(characters, fail); } StringView String::findAnyOr(const StringView characters, const char* fail) const { - return StringView { *this }.findAnyOr(characters, fail); + return StringView{*this}.findAnyOr(characters, fail); } MutableStringView String::findLastAny(const StringView characters) { - return MutableStringView { *this }.findLastAny(characters); + return MutableStringView{*this}.findLastAny(characters); } StringView String::findLastAny(const StringView characters) const { - return StringView { *this }.findLastAny(characters); + return StringView{*this}.findLastAny(characters); } MutableStringView String::findLastAnyOr(const StringView characters, char* fail) { - return MutableStringView { *this }.findLastAnyOr(characters, fail); + return MutableStringView{*this}.findLastAnyOr(characters, fail); } StringView String::findLastAnyOr(const StringView characters, const char* fail) const { - return StringView { *this }.findLastAnyOr(characters, fail); + return StringView{*this}.findLastAnyOr(characters, fail); } bool String::containsAny(const StringView substring) const { - return StringView { *this }.containsAny(substring); + return StringView{*this}.containsAny(substring); } char* String::release() { @@ -802,11 +802,11 @@ namespace Death::Containers namespace Implementation { String StringConverter::from(const std::string& other) { - return String { other.data(), other.size() }; + return String{other.data(), other.size()}; } std::string StringConverter::to(const String& other) { - return std::string { other.data(), other.size() }; + return std::string{other.data(), other.size()}; } } } \ No newline at end of file diff --git a/Sources/Shared/Containers/String.h b/Sources/Shared/Containers/String.h index 5174e0a0..d3bf6b1e 100644 --- a/Sources/Shared/Containers/String.h +++ b/Sources/Shared/Containers/String.h @@ -165,7 +165,8 @@ namespace Death::Containers * @cpp nullptr @ce --- in that case an empty string is constructed. * Depending on the size, it's either stored allocated or in a SSO. */ - /*implicit*/ String(const char* data); + /* To avoid ambiguity in certain cases of passing 0 to overloads that take either a String or std::size_t */ + template::value && !std::is_convertible::value>::type> /*implicit*/ String(T data) : String{nullptr, nullptr, nullptr, data} {} /** * @brief Construct from a sized C string @@ -860,6 +861,8 @@ namespace Death::Containers char* release(); private: + // Delegated to from the (templated) String(const char*). THREE extra nullptr arguments to avoid accidental ambiguous overloads. + explicit String(std::nullptr_t, std::nullptr_t, std::nullptr_t, const char* data); // Delegated to from the (templated) String(char*, Deleter). Argument order shuffled together with a null parameter to avoid accidental ambiguous overloads. explicit String(Deleter deleter, std::nullptr_t, char* data) noexcept; diff --git a/Sources/Shared/Containers/StringView.h b/Sources/Shared/Containers/StringView.h index b9791c87..76b92ebd 100644 --- a/Sources/Shared/Containers/StringView.h +++ b/Sources/Shared/Containers/StringView.h @@ -47,14 +47,14 @@ namespace Death::Containers * string view with this flag set doesn't need to have a copy allocated in * order to ensure it stays in scope. */ - Global = std::size_t { 1 } << (sizeof(std::size_t) * 8 - 1), + Global = std::size_t{1} << (sizeof(std::size_t) * 8 - 1), /** * The referenced string is null-terminated. A string view with this flag * set doesn't need to have a null-terminated copy allocated in order to * pass to an API that expects only null-terminated strings. */ - NullTerminated = std::size_t { 1 } << (sizeof(std::size_t) * 8 - 2) + NullTerminated = std::size_t{1} << (sizeof(std::size_t) * 8 - 2) }; DEFINE_ENUM_OPERATORS(StringViewFlags); @@ -75,12 +75,15 @@ namespace Death::Containers template class BasicStringView { public: + /* To avoid ambiguity in certain cases of passing 0 to overloads that take either a StringView or std::size_t */ + template::value>::type> constexpr /*implicit*/ BasicStringView(U) noexcept : _data{}, _sizePlusFlags{std::size_t(StringViewFlags::Global)} {} + /** * @brief Default constructor * * A default-constructed instance has @ref StringViewFlags::Global set. */ - constexpr /*implicit*/ BasicStringView(std::nullptr_t = nullptr) noexcept : _data{}, _sizePlusFlags{std::size_t(StringViewFlags::Global)} {} + constexpr /*implicit*/ BasicStringView() noexcept : _data{}, _sizePlusFlags{std::size_t(StringViewFlags::Global)} {} /** * @brief Construct from a C string of known size @@ -104,7 +107,7 @@ namespace Death::Containers constexpr /*implicit*/ BasicStringView(T* data, std::size_t size, StringViewFlags flags = {}) noexcept : _data{data}, _sizePlusFlags{ // This ends up being called from BasicStringView(T*, Flags), so basically on every implicit conversion // from a C string, thus the release build perf aspect wins over safety - (size | (std::size_t(flags) & Implementation::StringViewSizeMask))} { } + (size | (std::size_t(flags) & Implementation::StringViewSizeMask))} {} /** * @brief Construct from a @ref String @@ -141,7 +144,7 @@ namespace Death::Containers // It's also explicitly disallowing T[] arguments (which are implicitly convertible to an ArrayView), because those should be picking the T* // overload and rely on strlen(), consistently with how C string literals work; and disallowing construction from a StringView // because it'd get preferred over the implicit copy constructor. - template::type>::value && !std::is_same::type, BasicStringView>::value, decltype(ArrayView{std::declval()})>::type> constexpr /*implicit*/ BasicStringView(U&& data, StringViewFlags flags = { }) noexcept : BasicStringView{flags, ArrayView(data)} {} + template::type>::value && !std::is_same::type, BasicStringView>::value && !std::is_same::type, std::nullptr_t>::value, decltype(ArrayView{std::declval()})>::type> constexpr /*implicit*/ BasicStringView(U&& data, StringViewFlags flags = {}) noexcept: BasicStringView{flags, ArrayView(data)} {} /** @brief Construct a @ref StringView from a @ref MutableStringView */ template::value>::type> constexpr /*implicit*/ BasicStringView(BasicStringView mutable_) noexcept : _data{mutable_._data}, _sizePlusFlags{mutable_._sizePlusFlags} {} @@ -162,7 +165,7 @@ namespace Death::Containers * The @ref BasicStringView(std::nullptr_t) overload (which is a * default constructor) is additionally @cpp constexpr @ce. */ - DEATH_CONSTEXPR14 /*implicit*/ BasicStringView(T* data, StringViewFlags extraFlags = { }) noexcept : BasicStringView {data, extraFlags, nullptr} {} + template::value && std::is_convertible::value>::type> /*implicit*/ BasicStringView(U data, StringViewFlags extraFlags = {}) noexcept : BasicStringView{data, extraFlags, nullptr} {} /** * @brief Construct a view on an external type / from an external representation @@ -283,7 +286,7 @@ namespace Death::Containers * is @cpp nullptr @ce, returns zero-sized @cpp nullptr @ce view. */ constexpr BasicStringView prefix(T* end) const { - return end ? slice(_data, end) : BasicStringView {}; + return end ? slice(_data, end) : BasicStringView{}; } /** @@ -761,13 +764,13 @@ namespace Death::Containers // Called from BasicStringView(U&&, StringViewFlags), see its comment for details; arguments in a flipped order to avoid accidental // ambiguity. The ArrayView type is a template to avoid having to include ArrayView.h. - template::value>::type> constexpr explicit BasicStringView(StringViewFlags flags, ArrayView data) noexcept : BasicStringView { data.data(), data.size(), flags } {} + template::value>::type> constexpr explicit BasicStringView(StringViewFlags flags, ArrayView data) noexcept : BasicStringView{data.data(), data.size(), flags} {} // Used by the char* constructor, delinlined because it calls into std::strlen() explicit BasicStringView(T* data, StringViewFlags flags, std::nullptr_t) noexcept; // Used by slice() to skip unneeded checks in the public constexpr constructor - constexpr explicit BasicStringView(T* data, std::size_t sizePlusFlags, std::nullptr_t) noexcept : _data { data }, _sizePlusFlags { sizePlusFlags } {} + constexpr explicit BasicStringView(T* data, std::size_t sizePlusFlags, std::nullptr_t) noexcept : _data{data}, _sizePlusFlags{sizePlusFlags} {} T* _data; std::size_t _sizePlusFlags; @@ -825,7 +828,7 @@ namespace Death::Containers */ constexpr StringView operator"" _s(const char* data, std::size_t size) { // Using plain bit ops instead of EnumSet to speed up debug builds - return StringView { data, size, StringViewFlags(std::size_t(StringViewFlags::Global) | std::size_t(StringViewFlags::NullTerminated)) }; + return StringView{data, size, StringViewFlags(std::size_t(StringViewFlags::Global) | std::size_t(StringViewFlags::NullTerminated))}; } } @@ -966,7 +969,7 @@ namespace Death::Containers template<> struct ArrayViewConverter> { static ArrayView from(const BasicStringView& other); }; - template struct ErasedArrayViewConverter> : ArrayViewConverter> { }; - template struct ErasedArrayViewConverter> : ArrayViewConverter> { }; + template struct ErasedArrayViewConverter> : ArrayViewConverter> {}; + template struct ErasedArrayViewConverter> : ArrayViewConverter> {}; } } \ No newline at end of file diff --git a/Sources/Shared/IO/Stream.h b/Sources/Shared/IO/Stream.h index 33ade27b..12ffcb5d 100644 --- a/Sources/Shared/IO/Stream.h +++ b/Sources/Shared/IO/Stream.h @@ -75,7 +75,7 @@ namespace Death::IO template::value>::type* = nullptr> DEATH_ALWAYS_INLINE T ReadValue() { - T buffer; + T buffer = { }; Read(&buffer, sizeof(T)); return buffer; } diff --git a/Sources/nCine/AppConfiguration.cpp b/Sources/nCine/AppConfiguration.cpp index 7be38f51..42e37842 100644 --- a/Sources/nCine/AppConfiguration.cpp +++ b/Sources/nCine/AppConfiguration.cpp @@ -52,7 +52,6 @@ namespace nCine glMajorVersion_(3), glMinorVersion_(3), #endif - argc_(0), argv_(nullptr) { #if defined(DEATH_TARGET_ANDROID) @@ -81,21 +80,8 @@ namespace nCine return dataPath_; } -#if defined(DEATH_TARGET_WINDOWS) - const String AppConfiguration::argv(int index) const - { - if (index < argc_) { - return Death::Utf8::FromUtf16(argv_[index]); - } - return { }; - } -#else const StringView AppConfiguration::argv(int index) const { - if (index < argc_) { - return argv_[index]; - } - return { }; + return argv_[index]; } -#endif } diff --git a/Sources/nCine/AppConfiguration.h b/Sources/nCine/AppConfiguration.h index c9ee5cba..cd8c631c 100644 --- a/Sources/nCine/AppConfiguration.h +++ b/Sources/nCine/AppConfiguration.h @@ -9,12 +9,6 @@ using namespace Death::Containers; namespace nCine { -#if defined(DEATH_TARGET_WINDOWS) - typedef wchar_t* NativeArgument; -#else - typedef char* NativeArgument; -#endif - /// The class storing initialization settings for an nCine application class AppConfiguration { @@ -96,15 +90,11 @@ namespace nCine } /// \returns The number of arguments passed on the command-line - inline int argc() const { - return argc_; + inline std::size_t argc() const { + return argv_.size(); } /// \returns The selected argument from the ones passed on the command-line -#if defined(DEATH_TARGET_WINDOWS) - const String argv(int index) const; -#else const StringView argv(int index) const; -#endif private: // Pre-configured compile-time variables @@ -113,8 +103,11 @@ namespace nCine const unsigned int glMajorVersion_; const unsigned int glMinorVersion_; - int argc_; - NativeArgument* argv_; +#if defined(DEATH_TARGET_WINDOWS) + Array argv_; +#else + Array argv_; +#endif String dataPath_; friend class MainApplication; diff --git a/Sources/nCine/Application.cpp b/Sources/nCine/Application.cpp index 12e121da..3df36e99 100644 --- a/Sources/nCine/Application.cpp +++ b/Sources/nCine/Application.cpp @@ -1,5 +1,13 @@ #include "../Common.h" +#if defined(DEATH_TARGET_WINDOWS) +extern "C" +{ + _declspec(dllexport) unsigned long int NvOptimusEnablement = 0x00000001; + _declspec(dllexport) int AmdPowerXpressRequestHighPerformance = 0x00000001; +}; +#endif + #if defined(DEATH_TARGET_WINDOWS) && !defined(CMAKE_BUILD) # pragma comment(lib, "opengl32.lib") # if defined(_M_X64) @@ -33,13 +41,6 @@ # else # error Unsupported architecture # endif - -extern "C" -{ - _declspec(dllexport) unsigned long int NvOptimusEnablement = 0x00000001; - _declspec(dllexport) int AmdPowerXpressRequestHighPerformance = 1; -}; - #endif #include "Application.h" diff --git a/Sources/nCine/Base/Algorithms.cpp b/Sources/nCine/Base/Algorithms.cpp index 10aefa5a..adbc3391 100644 --- a/Sources/nCine/Base/Algorithms.cpp +++ b/Sources/nCine/Base/Algorithms.cpp @@ -12,7 +12,7 @@ namespace nCine { - int copyStringFirst(char* dest, int destSize, const char* source, int count) + std::int32_t copyStringFirst(char* dest, std::int32_t destSize, const char* source, std::int32_t count) { if (destSize == 0) { return 0; @@ -44,24 +44,24 @@ namespace nCine return n; } - int formatString(char* buffer, size_t maxLen, const char* format, ...) + std::int32_t formatString(char* buffer, std::size_t maxLen, const char* format, ...) { va_list args; va_start(args, format); #if defined(DEATH_TARGET_WINDOWS) && !defined(DEATH_TARGET_MINGW) - const int writtenChars = vsnprintf_s(buffer, maxLen, maxLen - 1, format, args); - const int result = (writtenChars > -1 ? writtenChars : maxLen - 1); + const std::int32_t writtenChars = vsnprintf_s(buffer, maxLen, maxLen - 1, format, args); + const std::int32_t result = (writtenChars > -1 ? writtenChars : maxLen - 1); #else - const int result = ::vsnprintf(buffer, maxLen, format, args); + const std::int32_t result = ::vsnprintf(buffer, maxLen, format, args); #endif va_end(args); return result; } - inline unsigned CountDecimalDigit32(uint32_t n) + inline std::uint32_t CountDecimalDigit32(std::uint32_t n) { #if defined(DEATH_TARGET_MSVC) || defined(DEATH_TARGET_GCC) - static constexpr uint32_t powers_of_10[] = { + static constexpr std::uint32_t powers_of_10[] = { 0, 10, 100, @@ -77,9 +77,9 @@ namespace nCine # if defined(DEATH_TARGET_MSVC) unsigned long i = 0; _BitScanReverse(&i, n | 1); - uint32_t t = (i + 1) * 1233 >> 12; + std::uint32_t t = (i + 1) * 1233 >> 12; # elif defined(DEATH_TARGET_GCC) - uint32_t t = (32 - __builtin_clz(n | 1)) * 1233 >> 12; + std::uint32_t t = (32 - __builtin_clz(n | 1)) * 1233 >> 12; # endif return t - (n < powers_of_10[t]) + 1; #else @@ -97,10 +97,10 @@ namespace nCine #endif } - inline unsigned CountDecimalDigit64(uint64_t n) + inline std::uint32_t CountDecimalDigit64(std::uint64_t n) { #if defined(DEATH_TARGET_MSVC) || defined(DEATH_TARGET_GCC) - static constexpr uint64_t powers_of_10[] = { + static constexpr std::uint64_t powers_of_10[] = { 0, 10, 100, @@ -124,20 +124,20 @@ namespace nCine }; # if defined(DEATH_TARGET_GCC) - uint32_t t = (64 - __builtin_clzll(n | 1)) * 1233 >> 12; + std::uint32_t t = (64 - __builtin_clzll(n | 1)) * 1233 >> 12; # elif defined(DEATH_TARGET_32BIT) unsigned long i = 0; - uint64_t m = n | 1; + std::uint64_t m = n | 1; if (_BitScanReverse(&i, m >> 32)) { i += 32; } else { _BitScanReverse(&i, m & 0xFFFFFFFF); } - uint32_t t = (i + 1) * 1233 >> 12; + std::uint32_t t = (i + 1) * 1233 >> 12; # else unsigned long i = 0; _BitScanReverse64(&i, n | 1); - uint32_t t = (i + 1) * 1233 >> 12; + std::uint32_t t = (i + 1) * 1233 >> 12; # endif return t - (n < powers_of_10[t]) + 1; #else @@ -165,9 +165,9 @@ namespace nCine #endif } - void u32tos(uint32_t value, char* buffer) + void u32tos(std::uint32_t value, char* buffer) { - unsigned digit = CountDecimalDigit32(value); + std::uint32_t digit = CountDecimalDigit32(value); buffer += digit; *buffer = '\0'; @@ -177,9 +177,9 @@ namespace nCine } while (value > 0); } - void i32tos(int32_t value, char* buffer) + void i32tos(std::int32_t value, char* buffer) { - uint32_t u = static_cast(value); + std::uint32_t u = static_cast(value); if (value < 0) { *buffer++ = '-'; u = ~u + 1; @@ -187,9 +187,9 @@ namespace nCine u32tos(u, buffer); } - void u64tos(uint64_t value, char* buffer) + void u64tos(std::uint64_t value, char* buffer) { - unsigned digit = CountDecimalDigit64(value); + std::uint32_t digit = CountDecimalDigit64(value); buffer += digit; *buffer = '\0'; @@ -199,9 +199,9 @@ namespace nCine } while (value > 0); } - void i64tos(int64_t value, char* buffer) + void i64tos(std::int64_t value, char* buffer) { - uint64_t u = static_cast(value); + std::uint64_t u = static_cast(value); if (value < 0) { *buffer++ = '-'; u = ~u + 1; @@ -209,26 +209,26 @@ namespace nCine u64tos(u, buffer); } - void ftos(double value, char* buffer, int bufferSize) + void ftos(double value, char* buffer, std::int32_t bufferSize) { #if defined(DEATH_TARGET_WINDOWS) && !defined(DEATH_TARGET_MINGW) - int length = sprintf_s(buffer, bufferSize, "%f", value); + std::int32_t length = sprintf_s(buffer, bufferSize, "%f", value); #else - int length = snprintf(buffer, bufferSize, "%f", value); + std::int32_t length = snprintf(buffer, bufferSize, "%f", value); #endif if (length <= 0) { buffer[0] = '\0'; return; } - int n = length - 1; + std::int32_t n = length - 1; while (n >= 0 && buffer[n] == '0') { n--; } n++; bool separatorFound = false; - for (int i = 0; i < n; i++) { + for (std::int32_t i = 0; i < n; i++) { if (buffer[i] == '.' || buffer[i] == ',') { separatorFound = true; break; @@ -244,42 +244,4 @@ namespace nCine buffer[n] = '\0'; } } - - uint64_t parseVersion(const Containers::StringView version) - { - auto parts = version.split('.'); - size_t partsCount = parts.size(); - if (partsCount == 0) { - return 0; - } - - uint64_t major = 0, minor = 0, patch = 0; - size_t partSize; - char stringBuffer[32]; - - partSize = std::min(parts[0].size(), sizeof(stringBuffer) - 1); - std::memcpy(stringBuffer, parts[0].data(), partSize); - stringBuffer[partSize] = '\0'; - major = strtol(stringBuffer, nullptr, 10); - - if (partsCount >= 2 && !parts[1].empty()) { - partSize = std::min(parts[1].size(), sizeof(stringBuffer) - 1); - std::memcpy(stringBuffer, parts[1].data(), partSize); - stringBuffer[partSize] = '\0'; - minor = strtol(stringBuffer, nullptr, 10); - } - if (partsCount >= 3 && !parts[2].empty()) { - if (parts[2][0] == 'r') { - // GIT Revision - always the latest - patch = 0x0FFFFFFFull; - } else { - partSize = std::min(parts[2].size(), sizeof(stringBuffer) - 1); - std::memcpy(stringBuffer, parts[2].data(), partSize); - stringBuffer[partSize] = '\0'; - patch = strtol(stringBuffer, nullptr, 10); - } - } - - return (patch & 0xFFFFFFFFull) | ((minor & 0xFFFFull) << 32) | ((major & 0xFFFFull) << 48); - } } diff --git a/Sources/nCine/Base/Algorithms.h b/Sources/nCine/Base/Algorithms.h index ee007803..df9ab338 100644 --- a/Sources/nCine/Base/Algorithms.h +++ b/Sources/nCine/Base/Algorithms.h @@ -13,32 +13,32 @@ using namespace Death; namespace nCine { // Traits - template + template struct isIntegral { static constexpr bool value = false; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; @@ -48,46 +48,46 @@ namespace nCine { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template <> + template<> struct isIntegral { static constexpr bool value = true; }; - template + template inline bool IsLess(const T &a, const T &b) { return a < b; } - template + template inline bool IsNotLess(const T &a, const T &b) { return !(a < b); } /// Returns true if the range is sorted into ascending order - template + template inline bool isSorted(Iterator first, const Iterator last) { if (first == last) @@ -104,7 +104,7 @@ namespace nCine } /// Returns true if the range is sorted, using a custom comparison - template + template inline bool isSorted(Iterator first, const Iterator last, Compare comp) { if (first == last) @@ -121,7 +121,7 @@ namespace nCine } /// Returns an iterator to the first element in the range which does not follow an ascending order, or last if sorted - template + template inline const Iterator isSortedUntil(Iterator first, const Iterator last) { if (first == last) @@ -138,7 +138,7 @@ namespace nCine } /// Returns an iterator to the first element in the range which does not follow the custom comparison, or last if sorted - template + template inline const Iterator isSortedUntil(Iterator first, const Iterator last, Compare comp) { if (first == last) @@ -155,7 +155,7 @@ namespace nCine } /// Partition function for quicksort with iterators - template + template inline Iterator partition(Iterator first, Iterator last, Compare comp) { Iterator pivot = last; @@ -183,10 +183,10 @@ namespace nCine namespace { /// Quicksort implementation with random access iterators and custom compare function - template + template inline void quicksort(Iterator first, Iterator last, RandomAccessIteratorTag, Compare comp) { - int size = distance(first, last, RandomAccessIteratorTag()); + std::int32_t size = distance(first, last, RandomAccessIteratorTag()); if (size > 1) { Iterator p = prev(last); std::swap(*next(first, size / 2), *p); @@ -198,7 +198,7 @@ namespace nCine } /// Quicksort implementation with bidirectional iterators and custom compare function - template + template inline void quicksort(Iterator first, Iterator last, BidirectionalIteratorTag, Compare comp) { if (first != last) { @@ -214,28 +214,28 @@ namespace nCine } /// Quicksort implementation with iterators and custom compare function - template + template inline void quicksort(Iterator first, Iterator last, Compare comp) { quicksort(first, last, IteratorTraits::IteratorCategory(), comp); } /// Quicksort implementation with iterators, ascending order - template + template inline void quicksort(Iterator first, Iterator last) { quicksort(first, last, IteratorTraits::IteratorCategory(), IsLess::ValueType>); } /// Quicksort implementation with iterators, descending order - template + template inline void quicksortDesc(Iterator first, Iterator last) { quicksort(first, last, IteratorTraits::IteratorCategory(), IsNotLess::ValueType>); } /// A container for functions to destruct objects and arrays of objects - template + template struct destructHelpers { template @@ -245,16 +245,16 @@ namespace nCine } template - inline static void destructArray(T* ptr, unsigned int numElements) + inline static void destructArray(T* ptr, std::uint32_t numElements) { - for (unsigned int i = 0; i < numElements; i++) + for (std::uint32_t i = 0; i < numElements; i++) ptr[numElements - i - 1].~T(); } }; namespace detail { - template + template struct typeIdentity { using type = T; @@ -266,18 +266,18 @@ namespace nCine auto tryAddRValueReference(...)->typeIdentity; } - template + template struct addRValueReference : decltype(detail::tryAddRValueReference(0)) {}; - template + template typename addRValueReference::type declVal(); /// Specialization for trivially destructible types - template + template struct isDestructible { static constexpr bool value = false; }; - template + template struct isDestructible().~T())> { static constexpr bool value = (true && !__is_union(T)); @@ -285,47 +285,47 @@ namespace nCine // Use `__has_trivial_destructor()` only on GCC #if defined(__GNUC__) && !defined(__clang__) && !defined(__INTEL_COMPILER) - template + template struct hasTrivialDestructor { static constexpr bool value = __has_trivial_destructor(T); }; - template + template struct isTriviallyDestructible { static constexpr bool value = isDestructible::value && hasTrivialDestructor::value; }; #else - template + template struct isTriviallyDestructible { static constexpr bool value = __is_trivially_destructible(T); }; #endif - template <> + template<> struct destructHelpers { - template + template inline static void destructObject(T* ptr) { } - template + template inline static void destructArray(T* ptr, unsigned int numElements) { } }; - template + template void destructObject(T* ptr) { destructHelpers::value>::destructObject(ptr); } - template - void destructArray(T* ptr, unsigned int numElements) + template + void destructArray(T* ptr, std::uint32_t numElements) { destructHelpers::value>::destructArray(ptr, numElements); } @@ -335,9 +335,9 @@ namespace nCine return a + ratio * (b - a); } - inline int lerp(int a, int b, float ratio) + inline std::int32_t lerp(std::int32_t a, std::int32_t b, float ratio) { - return (int)std::round(a + ratio * (float)(b - a)); + return (std::int32_t)std::round(a + ratio * (float)(b - a)); } inline void lowercaseInPlace(const Containers::MutableStringView string) @@ -363,20 +363,80 @@ namespace nCine } } - int copyStringFirst(char* dest, int destSize, const char* source, int count = -1); + std::int32_t copyStringFirst(char* dest, std::int32_t destSize, const char* source, std::int32_t count = -1); + + template + inline std::int32_t copyStringFirst(char(&dest)[size], const char* source, std::int32_t count = -1) { + return copyStringFirst(dest, size, source, count); + } + + int formatString(char* buffer, std::size_t maxLen, const char* format, ...); - template - inline int copyStringFirst(char(&dest)[N], const char* source, int count = -1) { - return copyStringFirst(dest, N, source, count); + void u32tos(std::uint32_t value, char* buffer); + void i32tos(std::int32_t value, char* buffer); + void u64tos(std::uint64_t value, char* buffer); + void i64tos(std::int64_t value, char* buffer); + void ftos(double value, char* buffer, std::int32_t bufferSize); + + constexpr bool isDigit(char c) + { + return (c >= '0' && c <= '9'); + } + + constexpr std::uint32_t stou32(const char* str, std::size_t length) + { + std::uint32_t n = 0; + while (length > 0) { + if (!isDigit(*str)) { + break; + } + n *= 10; + n += (*str++ - '0'); + length--; + } + return n; + } + + constexpr std::uint64_t stou64(const char* str, std::size_t length) + { + std::uint64_t n = 0; + while (length > 0) { + if (!isDigit(*str)) { + break; + } + n *= 10; + n += (*str++ - '0'); + length--; + } + return n; } - int formatString(char* buffer, size_t maxLen, const char* format, ...); + constexpr std::uint64_t parseVersion(const Containers::StringView& version) + { + std::size_t versionLength = version.size(); + std::size_t dotIndices[3] { }; + std::size_t foundCount = 0; - void u32tos(uint32_t value, char* buffer); - void i32tos(int32_t value, char* buffer); - void u64tos(uint64_t value, char* buffer); - void i64tos(int64_t value, char* buffer); - void ftos(double value, char* buffer, int bufferSize); + for (std::size_t i = 0; i < versionLength; i++) { + if (version[i] == '.') { + dotIndices[foundCount++] = i; + if (foundCount >= countof(dotIndices) - 1) { + // Save only indices of the first 2 dots and keep the last index for string length + break; + } + } + } + + dotIndices[foundCount] = versionLength; - uint64_t parseVersion(const Containers::StringView version); + std::uint64_t major = stou32(&version[0], dotIndices[0]); + std::uint64_t minor = (foundCount >= 1 ? stou32(&version[dotIndices[0] + 1], dotIndices[1] - dotIndices[0] - 1) : 0); + std::uint64_t patch = (foundCount >= 2 + ? (version[dotIndices[1] + 1] != 'r' + ? stou32(&version[dotIndices[1] + 1], dotIndices[2] - dotIndices[1] - 1) + : 0x0FFFFFFFULL) // GIT Revision - use special value, so it's always the latest (without upper 4 bits) + : 0); + + return (patch & 0xFFFFFFFFULL) | ((minor & 0xFFFFULL) << 32) | ((major & 0xFFFFULL) << 48); + } } diff --git a/Sources/nCine/Graphics/BinaryShaderCache.cpp b/Sources/nCine/Graphics/BinaryShaderCache.cpp index e22b7859..2bf2d10c 100644 --- a/Sources/nCine/Graphics/BinaryShaderCache.cpp +++ b/Sources/nCine/Graphics/BinaryShaderCache.cpp @@ -179,6 +179,10 @@ namespace nCine void BinaryShaderCache::prune() { + if (path_.empty()) { + return; + } + fs::Directory dir(path_); while (const StringView shaderPath = dir.GetNext()) { if (fs::GetExtension(shaderPath) != "shader"_s) { @@ -186,17 +190,20 @@ namespace nCine } StringView filename = fs::GetFileNameWithoutExtension(shaderPath); + + bool shouldRemove; if (filename.size() != 32) { - fs::RemoveFile(shaderPath); - continue; + shouldRemove = true; + } else { + char componentString[17]; + std::memcpy(componentString, &filename[16], 16); + componentString[16] = '\0'; + + std::uint64_t platformHash = strtoull(componentString, nullptr, 16); + shouldRemove = (platformHash != platformHash_); } - char componentString[17]; - std::memcpy(componentString, &filename[16], 16); - componentString[16] = '\0'; - - std::uint64_t platformHash = strtoull(componentString, nullptr, 16); - if (platformHash != platformHash_) { + if (shouldRemove) { fs::RemoveFile(shaderPath); } } @@ -204,8 +211,16 @@ namespace nCine void BinaryShaderCache::clear() { + if (path_.empty()) { + return; + } + fs::Directory dir(path_); while (const StringView shaderPath = dir.GetNext()) { + if (fs::GetExtension(shaderPath) != "shader"_s) { + continue; + } + fs::RemoveFile(shaderPath); } } diff --git a/Sources/nCine/Input/JoyMapping.cpp b/Sources/nCine/Input/JoyMapping.cpp index 80c8bb4d..8b3af82b 100644 --- a/Sources/nCine/Input/JoyMapping.cpp +++ b/Sources/nCine/Input/JoyMapping.cpp @@ -4,7 +4,6 @@ #include "../Primitives/Vector2.h" #include // for memcpy() -#include // for strtoul() #include #include diff --git a/Sources/nCine/MainApplication.cpp b/Sources/nCine/MainApplication.cpp index 8ac9915a..29fd4c55 100644 --- a/Sources/nCine/MainApplication.cpp +++ b/Sources/nCine/MainApplication.cpp @@ -17,13 +17,15 @@ #if defined(DEATH_TARGET_EMSCRIPTEN) # include -#endif -#if defined(DEATH_TARGET_SWITCH) +#elif defined(DEATH_TARGET_SWITCH) # include +#elif defined(DEATH_TARGET_WINDOWS) +# include #endif #include "tracy.h" +using namespace Death; using namespace Death::Containers::Literals; using namespace Death::IO; @@ -44,6 +46,7 @@ extern "C" IMAGE_DOS_HEADER __ImageBase; bool __showLogConsole; bool __hasVirtualTerminal; +Array __consolePrompt; static bool CreateLogConsole(const StringView& title) { @@ -68,6 +71,51 @@ static bool CreateLogConsole(const StringView& title) ::setvbuf(stdin, NULL, _IONBF, 0); } + // Try to get command prompt to be able to reprint it when the game exits + CONSOLE_SCREEN_BUFFER_INFO csbi; + if (::GetConsoleScreenBufferInfo(consoleHandleOut, &csbi)) { + DWORD dwConsoleColumnWidth = (DWORD)(csbi.srWindow.Right - csbi.srWindow.Left + 1); + SHORT xEnd = csbi.dwCursorPosition.X; + SHORT yEnd = csbi.dwCursorPosition.Y; + if (xEnd != 0 || yEnd != 0) { + DWORD dwNumberOfChars; + SHORT yBegin = yEnd; + if (dwConsoleColumnWidth > 16) { + Array tmp(NoInit, dwConsoleColumnWidth); + while (yBegin > 0) { + COORD dwReadCoord = { 0, yBegin }; + if (!::ReadConsoleOutputCharacter(consoleHandleOut, tmp.data(), dwConsoleColumnWidth, dwReadCoord, &dwNumberOfChars)) { + break; + } + + for (DWORD i = dwNumberOfChars - 8; i < dwNumberOfChars; i++) { + wchar_t wchar = tmp[i]; + if (wchar != L' ') { + yBegin--; + continue; + } + } + + if (yBegin < yEnd) { + yBegin++; + } + break; + } + } + + DWORD promptLength = (yEnd - yBegin) * dwConsoleColumnWidth + xEnd; + __consolePrompt = Array(NoInit, promptLength); + COORD dwPromptCoord = { 0, yEnd }; + if (::ReadConsoleOutputCharacter(consoleHandleOut, __consolePrompt.data(), promptLength, dwPromptCoord, &dwNumberOfChars)) { + if (::SetConsoleCursorPosition(consoleHandleOut, dwPromptCoord)) { + ::FillConsoleOutputCharacter(consoleHandleOut, L' ', promptLength, dwPromptCoord, &dwNumberOfChars); + } + } else { + __consolePrompt = {}; + } + } + } + return true; } else if (::AllocConsole()) { ::freopen_s(&fDummy, "CONOUT$", "w", stdout); @@ -98,21 +146,20 @@ static bool CreateLogConsole(const StringView& title) static void DestroyLogConsole() { - // The "Enter" key is only sent if the console window is in focus - if (::GetConsoleWindow() == ::GetForegroundWindow()) { - // Send the "Enter" key to the console to release the command prompt - INPUT ip; - ip.type = INPUT_KEYBOARD; - ip.ki.wScan = 0; - ip.ki.time = 0; - ip.ki.dwExtraInfo = 0; - - ip.ki.wVk = 0x0D; // virtual-key code for the "Enter" key - ip.ki.dwFlags = 0; // 0 for key press - ::SendInput(1, &ip, sizeof(INPUT)); - - ip.ki.dwFlags = KEYEVENTF_KEYUP; // `KEYEVENTF_KEYUP` for key release - ::SendInput(1, &ip, sizeof(INPUT)); + if (!__consolePrompt.empty()) { + HANDLE consoleHandleOut = ::GetStdHandle(STD_OUTPUT_HANDLE); + if (consoleHandleOut != INVALID_HANDLE_VALUE) { + CONSOLE_SCREEN_BUFFER_INFO csbi; + if (::GetConsoleScreenBufferInfo(consoleHandleOut, &csbi)) { + DWORD xEnd = csbi.dwCursorPosition.X; + DWORD yEnd = csbi.dwCursorPosition.Y; + if (xEnd != 0 || yEnd != 0) { + DWORD dwNumberOfCharsWritten; + ::WriteConsole(consoleHandleOut, L"\r\n", countof(L"\r\n") - 1, &dwNumberOfCharsWritten, NULL); + ::WriteConsole(consoleHandleOut, __consolePrompt.data(), (DWORD)__consolePrompt.size(), &dwNumberOfCharsWritten, NULL); + } + } + } } ::FreeConsole(); @@ -120,18 +167,14 @@ static void DestroyLogConsole() static bool EnableVirtualTerminalProcessing() { - HANDLE hOut = ::GetStdHandle(STD_OUTPUT_HANDLE); - if (hOut == INVALID_HANDLE_VALUE) { + HANDLE consoleHandleOut = ::GetStdHandle(STD_OUTPUT_HANDLE); + if (consoleHandleOut == INVALID_HANDLE_VALUE) { return false; } + DWORD dwMode = 0; - if (!::GetConsoleMode(hOut, &dwMode)) { - return false; - } - if (!::SetConsoleMode(hOut, dwMode | ENABLE_VIRTUAL_TERMINAL_PROCESSING)) { - return false; - } - return true; + return (::GetConsoleMode(consoleHandleOut, &dwMode) && + ::SetConsoleMode(consoleHandleOut, dwMode | ENABLE_VIRTUAL_TERMINAL_PROCESSING)); } #elif defined(DEATH_TRACE) && (defined(DEATH_TARGET_APPLE) || defined(DEATH_TARGET_EMSCRIPTEN) || defined(DEATH_TARGET_UNIX)) @@ -228,7 +271,8 @@ namespace nCine #if defined(DEATH_TRACE) # if defined(DEATH_TARGET_APPLE) - __hasVirtualTerminal = isatty(1); + // Xcode's console reports that it is a TTY, but it doesn't support colors, TERM is not defined in this case + __hasVirtualTerminal = isatty(1) && ::getenv("TERM"); # elif defined(DEATH_TARGET_EMSCRIPTEN) char* userAgent = (char*)EM_ASM_PTR({ return (typeof navigator !== 'undefined' && navigator !== null && @@ -244,14 +288,14 @@ namespace nCine } # elif defined(DEATH_TARGET_WINDOWS) && !defined(DEATH_TARGET_WINDOWS_RT) __showLogConsole = false; - for (int i = 0; i < argc; i++) { + for (std::int32_t i = 0; i < argc; i++) { if (wcscmp(argv[i], L"/log") == 0) { __showLogConsole = true; break; } } if (__showLogConsole) { - CreateLogConsole(NCINE_APP_NAME " Console"); + CreateLogConsole(NCINE_APP_NAME " [Console]"); __hasVirtualTerminal = EnableVirtualTerminalProcessing(); } else { __hasVirtualTerminal = false; @@ -259,17 +303,24 @@ namespace nCine # elif defined(DEATH_TARGET_UNIX) ::setvbuf(stdout, nullptr, _IONBF, 0); ::setvbuf(stderr, nullptr, _IONBF, 0); - - // Xcode's console reports that it is a TTY, but it doesn't support colors, but TERM is not defined - __hasVirtualTerminal = isatty(1) && std::getenv("TERM"); + __hasVirtualTerminal = isatty(1); # endif #endif appEventHandler_ = createAppEventHandler(); // Only `OnPreInit()` can modify the application configuration - appCfg_.argc_ = argc; - appCfg_.argv_ = argv; +#if defined(DEATH_TARGET_WINDOWS) + appCfg_.argv_ = Array(argc - 1); + for (std::int32_t i = 1; i < argc; i++) { + appCfg_.argv_[i - 1] = Utf8::FromUtf16(argv[i]); + } +#else + appCfg_.argv_ = Array(argc - 1); + for (std::int32_t i = 1; i < argc; i++) { + appCfg_.argv_[i - 1] = argv[i]; + } +#endif appEventHandler_->OnPreInit(appCfg_); LOGI("IAppEventHandler::OnPreInit() invoked"); diff --git a/Sources/nCine/MainApplication.h b/Sources/nCine/MainApplication.h index 78ec3201..1993d101 100644 --- a/Sources/nCine/MainApplication.h +++ b/Sources/nCine/MainApplication.h @@ -4,6 +4,12 @@ namespace nCine { +#if defined(DEATH_TARGET_WINDOWS) + typedef wchar_t* NativeArgument; +#else + typedef char* NativeArgument; +#endif + #if defined(WITH_QT5) class Qt5Widget; #endif diff --git a/cmake/ncine_headers.cmake b/cmake/ncine_headers.cmake index fa7c4ee9..e44b701c 100644 --- a/cmake/ncine_headers.cmake +++ b/cmake/ncine_headers.cmake @@ -331,6 +331,7 @@ list(APPEND HEADERS ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/InGameMenu.h ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/InputDiagnosticsSection.h ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/LanguageSelectSection.h + ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/LoadingSection.h ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/MainMenu.h ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/OptionsSection.h ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/PauseSection.h diff --git a/cmake/ncine_sources.cmake b/cmake/ncine_sources.cmake index 200ac429..e6fcbfbe 100644 --- a/cmake/ncine_sources.cmake +++ b/cmake/ncine_sources.cmake @@ -256,6 +256,7 @@ list(APPEND SOURCES ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/InGameMenu.cpp ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/InputDiagnosticsSection.cpp ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/LanguageSelectSection.cpp + ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/LoadingSection.cpp ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/MainMenu.cpp ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/OptionsSection.cpp ${NCINE_SOURCE_DIR}/Jazz2/UI/Menu/PauseSection.cpp