diff --git a/Activities/GibEditor.cpp b/Activities/GibEditor.cpp index 6248f7a07..fbfa030a0 100644 --- a/Activities/GibEditor.cpp +++ b/Activities/GibEditor.cpp @@ -511,7 +511,7 @@ void GibEditor::Update() std::list *pEditedGibList = m_pEditorGUI->GetPlacedGibs(); MovableObject *pGibCopy = 0; - for (std::list::iterator gItr = pLoadedGibList->begin(); gItr != pLoadedGibList->end(); ++gItr) + for (auto gItr = pLoadedGibList->begin(); gItr != pLoadedGibList->end(); ++gItr) { pGibCopy = dynamic_cast((*gItr).GetParticlePreset()->Clone()); if (pGibCopy) diff --git a/CHANGELOG.md b/CHANGELOG.md index 63cb4ec79..5747794fa 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -12,7 +12,12 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - Added a save/load game menu which can be accessed from the main or pause menus, which allows the user to save their current progress in an activity to resume at a later date. -- New `Settings.ini` property `EnableMultithreadedLua`, which can be used to enable multithreaded Lua scripts and AI, for any lua script with the `--[[MULTITHREAD]]--` tag. Defaults to true. +- Massive performance improvements, especially in very large scenes with lots of actors. + +- New multithreaded AI and Lua scripts. + Lua scripts now have extra callback functions `ThreadedUpdateAI(self)`, `ThreadedUpdate(self)` and `SyncedUpdate(self)`. + The `Threaded` callback functions are run in a multithreaded fashion, whereas `Update` runs in a singlethreaded fashion (where it's safe to modify global state or affect other objects). + The `SyncedUpdate` callback is called in a single-threaded fashion, but only when an MO directly requests it by calling `self:RequestSyncedUpdate()`. This gives greater performance, as the script can avoid any single-threaded updates being called on it until it explicitly needs it. - New generic Lua messaging system, to allow scripts on objects to communicate with other objects or scripts. Scripts on `MovableObject` now have new callback functions `OnMessage(self, message, context)` and `OnGlobalMessage(self, message, context)`. @@ -28,8 +33,6 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), New INI and Lua (R/W) property `CanAdjustAngleWhileFiring`, which defines whether the jet angle can change while the jetpack is active. Defaults to true. New INI and Lua (R/W) property `AdjustsThrottleForWeight`, which defines whether the jetpack will adjust it's throttle (between `NegativeThrottleMultiplier` and `PositiveThrottleMultiplier`) to account for any extra inventory mass. Increased throttle will decrease jet time accordingly. Defaults to true. -- New Lua event function `SyncedUpdate()`, which will be called in a thread-safe synchronized manner and allows multithreaded scripts to modify game state in a safe and consistent way. - - Multithreaded asynchronous pathfinding, which dramatically improves performance on large maps and improves AI responsiveness. New `Actor` Lua property (R) `IsWaitingOnNewMovePath`, which returns true while the actor is currently calculating a new path. New Lua `SceneMan` function `CalculatePathAsync` for asynchronous pathfinding. This function has no return value, and is used as follows: diff --git a/Entities/ACDropShip.cpp b/Entities/ACDropShip.cpp index 5e653f459..b3db4b68e 100644 --- a/Entities/ACDropShip.cpp +++ b/Entities/ACDropShip.cpp @@ -18,6 +18,8 @@ #include "AEmitter.h" #include "PresetMan.h" +#include "tracy/Tracy.hpp" + namespace RTE { ConcreteClassInfo(ACDropShip, ACraft, 10); @@ -272,6 +274,8 @@ MOID ACDropShip::DetectObstacle(float distance) void ACDropShip::PreControllerUpdate() { + ZoneScoped; + ACraft::PreControllerUpdate(); // TODO: Improve and make optional thrusters more robust! diff --git a/Entities/ACrab.cpp b/Entities/ACrab.cpp index 2ec81ce3a..7a563ec95 100644 --- a/Entities/ACrab.cpp +++ b/Entities/ACrab.cpp @@ -31,6 +31,8 @@ #include "GUI.h" #include "AllegroBitmap.h" +#include "tracy/Tracy.hpp" + namespace RTE { ConcreteClassInfo(ACrab, Actor, 20); @@ -997,6 +999,8 @@ void ACrab::OnNewMovePath() void ACrab::PreControllerUpdate() { + ZoneScoped; + Actor::PreControllerUpdate(); float deltaTime = g_TimerMan.GetDeltaTimeSecs(); @@ -1383,6 +1387,8 @@ void ACrab::PreControllerUpdate() void ACrab::Update() { + ZoneScoped; + Actor::Update(); //////////////////////////////////// diff --git a/Entities/ADoor.cpp b/Entities/ADoor.cpp index b8fa4157d..b2907ae84 100644 --- a/Entities/ADoor.cpp +++ b/Entities/ADoor.cpp @@ -6,6 +6,8 @@ #include "PresetMan.h" #include "SettingsMan.h" +#include "tracy/Tracy.hpp" + namespace RTE { ConcreteClassInfo(ADoor, Actor, 20); @@ -379,6 +381,8 @@ namespace RTE { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void ADoor::Update() { + ZoneScoped; + if (m_Door) { if (m_DoorState != STOPPED && m_Status != Actor::Status::INACTIVE && m_SensorTimer.IsPastSimMS(m_SensorInterval)) { UpdateSensors(); diff --git a/Entities/AEmitter.cpp b/Entities/AEmitter.cpp index a5fae556d..c997f5f77 100644 --- a/Entities/AEmitter.cpp +++ b/Entities/AEmitter.cpp @@ -591,6 +591,17 @@ void AEmitter::Update() m_WasEmitting = false; } } + + // Set the screen flash effect to draw at the final post processing stage + if (m_EmitEnabled && (!m_FlashOnlyOnBurst || m_BurstTriggered) && m_pFlash && m_pFlash->GetScreenEffect()) { + // Fudge the glow pos forward a bit so it aligns nicely with the flash + Vector emitPos(m_pFlash->GetScreenEffect()->w * 0.3F * m_FlashScale, 0); + emitPos.RadRotate(m_HFlipped ? c_PI + m_Rotation.GetRadAngle() - m_EmitAngle.GetRadAngle() : m_Rotation.GetRadAngle() + m_EmitAngle.GetRadAngle()); + emitPos = m_Pos + RotateOffset(m_EmissionOffset) + emitPos; + if (m_EffectAlwaysShows || !g_SceneMan.ObscuredPoint(emitPos)) { + g_PostProcessMan.RegisterPostEffect(emitPos, m_pFlash->GetScreenEffect(), m_pFlash->GetScreenEffectHash(), RandomNum(m_pFlash->GetEffectStopStrength(), m_pFlash->GetEffectStartStrength()) * std::clamp(m_FlashScale, 0.0F, 1.0F), m_pFlash->GetEffectRotAngle()); + } + } } @@ -616,17 +627,6 @@ void AEmitter::Draw(BITMAP *pTargetBitmap, if (m_pFlash && m_pFlash->IsDrawnAfterParent() && !onlyPhysical && mode == g_DrawColor && m_EmitEnabled && (!m_FlashOnlyOnBurst || m_BurstTriggered)) m_pFlash->Draw(pTargetBitmap, targetPos, mode, onlyPhysical); - - // Set the screen flash effect to draw at the final post processing stage - if (m_EmitEnabled && (!m_FlashOnlyOnBurst || m_BurstTriggered) && m_pFlash && m_pFlash->GetScreenEffect() && mode == g_DrawColor && !onlyPhysical) { - // Fudge the glow pos forward a bit so it aligns nicely with the flash - Vector emitPos(m_pFlash->GetScreenEffect()->w * 0.3F * m_FlashScale, 0); - emitPos.RadRotate(m_HFlipped ? c_PI + m_Rotation.GetRadAngle() - m_EmitAngle.GetRadAngle() : m_Rotation.GetRadAngle() + m_EmitAngle.GetRadAngle()); - emitPos = m_Pos + RotateOffset(m_EmissionOffset) + emitPos; - if (!g_SceneMan.ObscuredPoint(emitPos)) { - g_PostProcessMan.RegisterPostEffect(emitPos, m_pFlash->GetScreenEffect(), m_pFlash->GetScreenEffectHash(), RandomNum(m_pFlash->GetEffectStopStrength(), m_pFlash->GetEffectStartStrength()) * std::clamp(m_FlashScale, 0.0F, 1.0F), m_pFlash->GetEffectRotAngle()); - } - } } } // namespace RTE \ No newline at end of file diff --git a/Entities/AHuman.cpp b/Entities/AHuman.cpp index e581e0fac..7497ba440 100644 --- a/Entities/AHuman.cpp +++ b/Entities/AHuman.cpp @@ -30,6 +30,8 @@ #include "GUI.h" #include "AllegroBitmap.h" +#include "tracy/Tracy.hpp" + namespace RTE { ConcreteClassInfo(AHuman, Actor, 20); @@ -1710,6 +1712,8 @@ void AHuman::UpdateWalkAngle(AHuman::Layer whichLayer) { void AHuman::PreControllerUpdate() { + ZoneScoped; + Actor::PreControllerUpdate(); float deltaTime = g_TimerMan.GetDeltaTimeSecs(); @@ -2142,7 +2146,7 @@ void AHuman::PreControllerUpdate() reach += m_pFGArm ? m_pFGArm->GetMaxLength() : m_pBGArm->GetMaxLength(); reachPoint = m_pFGArm ? m_pFGArm->GetJointPos() : m_pBGArm->GetJointPos(); - MOID itemMOID = g_SceneMan.CastMORay(reachPoint, Vector(reach * RandomNum(0.5F, 1.0F) * GetFlipFactor(), 0).RadRotate(m_pItemInReach ? adjustedAimAngle : RandomNum(-(c_HalfPI + c_EighthPI), m_AimAngle * 0.75F + c_EighthPI) * GetFlipFactor()), m_MOID, Activity::NoTeam, g_MaterialGrass, true, 3); + MOID itemMOID = g_SceneMan.CastMORay(reachPoint, Vector(reach * RandomNum(0.5F, 1.0F) * GetFlipFactor(), 0).RadRotate(m_pItemInReach ? adjustedAimAngle : RandomNum(-(c_HalfPI + c_EighthPI), m_AimAngle * 0.75F + c_EighthPI) * GetFlipFactor()), m_MOID, m_Team, g_MaterialGrass, true, 3); if (MovableObject *foundMO = g_MovableMan.GetMOFromID(itemMOID)) { if (HeldDevice *foundDevice = dynamic_cast(foundMO->GetRootParent())) { @@ -2499,6 +2503,8 @@ void AHuman::PreControllerUpdate() void AHuman::Update() { + ZoneScoped; + float rot = m_Rotation.GetRadAngle(); // eugh, for backwards compat to be the same behaviour as with multithreaded AI Actor::Update(); diff --git a/Entities/Actor.cpp b/Entities/Actor.cpp index a854f83b3..e4ee5fadb 100644 --- a/Entities/Actor.cpp +++ b/Entities/Actor.cpp @@ -37,6 +37,8 @@ #include "GUI.h" #include "AllegroBitmap.h" +#include "tracy/Tracy.hpp" + namespace RTE { ConcreteClassInfo(Actor, MOSRotating, 20); @@ -1248,16 +1250,6 @@ float Actor::EstimateDigStrength() const { return m_AIBaseDigStrength; } -///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// - -void Actor::UpdateAIScripted(ThreadScriptsToRun scriptsToRun) { - RunScriptedFunctionInAppropriateScripts("UpdateAI", false, true, {}, {}, {}, scriptsToRun); - if (scriptsToRun == ThreadScriptsToRun::SingleThreaded) { - // If we're in a SingleThreaded context, we run the MultiThreaded scripts synced updates - RunScriptedFunctionInAppropriateScripts("SyncedUpdateAI", false, true, {}, {}, {}, ThreadScriptsToRun::Both); - } -} - ////////////////////////////////////////////////////////////////////////////////////////// // Method: VerifyMOIDs ////////////////////////////////////////////////////////////////////////////////////////// @@ -1309,6 +1301,8 @@ void Actor::PreControllerUpdate() { void Actor::Update() { + ZoneScoped; + ///////////////////////////////// // Hit Body update and handling MOSRotating::Update(); diff --git a/Entities/Actor.h b/Entities/Actor.h index 0439eaa60..3cf1bc7de 100644 --- a/Entities/Actor.h +++ b/Entities/Actor.h @@ -91,7 +91,7 @@ class Actor : public MOSRotating { // Concrete allocation and cloning definitions EntityAllocation(Actor); -AddScriptFunctionNames(MOSRotating, "UpdateAI", "SyncedUpdateAI", "OnControllerInputModeChange"); +AddScriptFunctionNames(MOSRotating, "ThreadedUpdateAI", "UpdateAI", "OnControllerInputModeChange"); SerializableOverrideMethods; ClassInfoGetters; @@ -1155,18 +1155,6 @@ ClassInfoGetters; /// The new base dig strength for this Actor. void SetAIBaseDigStrength(float newAIBaseDigStrength) { m_AIBaseDigStrength = newAIBaseDigStrength; } - -////////////////////////////////////////////////////////////////////////////////////////// -// Method: UpdateAIScripted -////////////////////////////////////////////////////////////////////////////////////////// -// Description: Updates this' AI state with the provided scripted AI Update function. -// Arguments: None. -// Return value: Whether there was an AI Update function defined for this in its script, -// and if it was executed successfully. - - void UpdateAIScripted(ThreadScriptsToRun scriptsToRun); - - ////////////////////////////////////////////////////////////////////////////////////////// // Virtual method: PreControllerUpdate ////////////////////////////////////////////////////////////////////////////////////////// diff --git a/Entities/Attachable.cpp b/Entities/Attachable.cpp index 5be8eeb39..53a77be3f 100644 --- a/Entities/Attachable.cpp +++ b/Entities/Attachable.cpp @@ -335,12 +335,12 @@ namespace RTE { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// - int Attachable::UpdateScripts(ThreadScriptsToRun scriptsToRun) { + int Attachable::UpdateScripts() { if (m_Parent && !m_AllLoadedScripts.empty() && !ObjectScriptsInitialized()) { - RunScriptedFunctionInAppropriateScripts("OnAttach", false, false, { m_Parent }, {}, {}, scriptsToRun); + RunScriptedFunctionInAppropriateScripts("OnAttach", false, false, { m_Parent }, {}, {}); } - return MOSRotating::UpdateScripts(scriptsToRun); + return MOSRotating::UpdateScripts(); } ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// @@ -389,13 +389,12 @@ namespace RTE { } // If we're attached to something, MovableMan doesn't own us, and therefore isn't calling our UpdateScripts method (and neither is our parent), so we should here. - // We run our single-threaded scripts here, so that single-threaded behaviour is unchanged from prior to the multithreaded lua implementation if (m_Parent && GetRootParent()->HasEverBeenAddedToMovableMan()) { g_PerformanceMan.StartPerformanceMeasurement(PerformanceMan::ScriptsUpdate); if (!m_AllLoadedScripts.empty() && !ObjectScriptsInitialized()) { - RunScriptedFunctionInAppropriateScripts("OnAttach", false, false, { m_Parent }, {}, {}, ThreadScriptsToRun::SingleThreaded); + RunScriptedFunctionInAppropriateScripts("OnAttach", false, false, { m_Parent }, {}, {}); } - UpdateScripts(ThreadScriptsToRun::SingleThreaded); + UpdateScripts(); g_PerformanceMan.StopPerformanceMeasurement(PerformanceMan::ScriptsUpdate); } diff --git a/Entities/Attachable.h b/Entities/Attachable.h index f89773772..7a3d702ed 100644 --- a/Entities/Attachable.h +++ b/Entities/Attachable.h @@ -455,9 +455,8 @@ namespace RTE { /// /// Updates this Attachable's Lua scripts. /// - /// Whether to run this objects single-threaded or multi-threaded scripts. /// An error return value signaling success or any particular failure. Anything below 0 is an error signal. - int UpdateScripts(ThreadScriptsToRun scriptsToRun) override; + int UpdateScripts() override; /// /// Updates this Attachable. Supposed to be done every frame. diff --git a/Entities/HDFirearm.cpp b/Entities/HDFirearm.cpp index 572dcd59a..86da06efc 100644 --- a/Entities/HDFirearm.cpp +++ b/Entities/HDFirearm.cpp @@ -1115,6 +1115,14 @@ void HDFirearm::Update() } m_FiredLastFrame = m_FireFrame; + + // Set the screen flash effect to draw at the final post processing stage + if (m_FireFrame && m_pFlash && m_pFlash->GetScreenEffect()) { + Vector muzzlePos = m_Pos + RotateOffset(m_MuzzleOff + Vector(m_pFlash->GetSpriteWidth() * 0.3F, 0)); + if (m_EffectAlwaysShows || !g_SceneMan.ObscuredPoint(muzzlePos)) { + g_PostProcessMan.RegisterPostEffect(muzzlePos, m_pFlash->GetScreenEffect(), m_pFlash->GetScreenEffectHash(), RandomNum(m_pFlash->GetEffectStopStrength(), m_pFlash->GetEffectStartStrength()), m_pFlash->GetEffectRotAngle()); + } + } } @@ -1144,14 +1152,6 @@ void HDFirearm::Draw(BITMAP *pTargetBitmap, const Vector &targetPos, DrawMode mo if (m_pFlash && m_FireFrame && m_pFlash->IsDrawnAfterParent() && mode == g_DrawColor && !onlyPhysical) { m_pFlash->Draw(pTargetBitmap, targetPos, mode, onlyPhysical); } - - // Set the screen flash effect to draw at the final post processing stage - if (m_FireFrame && m_pFlash && m_pFlash->GetScreenEffect() && mode == g_DrawColor && !onlyPhysical) { - Vector muzzlePos = m_Pos + RotateOffset(m_MuzzleOff + Vector(m_pFlash->GetSpriteWidth() * 0.3F, 0)); - if (!g_SceneMan.ObscuredPoint(muzzlePos)) { - g_PostProcessMan.RegisterPostEffect(muzzlePos, m_pFlash->GetScreenEffect(), m_pFlash->GetScreenEffectHash(), RandomNum(m_pFlash->GetEffectStopStrength(), m_pFlash->GetEffectStartStrength()), m_pFlash->GetEffectRotAngle()); - } - } } diff --git a/Entities/MOPixel.cpp b/Entities/MOPixel.cpp index d16796469..f82613816 100644 --- a/Entities/MOPixel.cpp +++ b/Entities/MOPixel.cpp @@ -221,6 +221,10 @@ namespace RTE { } } } + + if (m_pScreenEffect) { + SetPostScreenEffectToDraw(); + } } ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// @@ -261,10 +265,6 @@ namespace RTE { } g_SceneMan.RegisterDrawing(targetBitmap, mode == g_DrawNoMOID ? g_NoMOID : m_MOID, pixelPos, 1.0F); - - if (mode == g_DrawColor && m_pScreenEffect && !onlyPhysical) { - SetPostScreenEffectToDraw(); - } } ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// diff --git a/Entities/MOSParticle.cpp b/Entities/MOSParticle.cpp index e239a5d61..7ab18f68a 100644 --- a/Entities/MOSParticle.cpp +++ b/Entities/MOSParticle.cpp @@ -149,6 +149,14 @@ namespace RTE { } } +///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + void MOSParticle::Update() { + MOSprite::Update(); + + if (m_pScreenEffect) { SetPostScreenEffectToDraw(); } + } + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void MOSParticle::Draw(BITMAP *targetBitmap, const Vector &targetPos, DrawMode mode, bool onlyPhysical) const { @@ -221,8 +229,6 @@ namespace RTE { g_SceneMan.RegisterDrawing(targetBitmap, mode == g_DrawNoMOID ? g_NoMOID : m_MOID, spriteX, spriteY, spriteX + m_aSprite[m_Frame]->w, spriteY + m_aSprite[m_Frame]->h); } - - if (m_pScreenEffect && mode == g_DrawColor && !onlyPhysical) { SetPostScreenEffectToDraw(); } } ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// diff --git a/Entities/MOSParticle.h b/Entities/MOSParticle.h index 5b8f8d35c..3d40c7dbb 100644 --- a/Entities/MOSParticle.h +++ b/Entities/MOSParticle.h @@ -130,6 +130,11 @@ namespace RTE { /// Whether the MOSParticle should immediately halt any travel going on after this sinkage. bool OnSink(HitData &hd) override { return false; } + /// + /// Updates this MOParticle. Supposed to be done every frame. + /// + void Update() override; + /// /// Draws this MOSParticle's current graphical representation to a BITMAP of choice. /// diff --git a/Entities/MOSRotating.cpp b/Entities/MOSRotating.cpp index a8ad545e1..22e18c433 100644 --- a/Entities/MOSRotating.cpp +++ b/Entities/MOSRotating.cpp @@ -326,7 +326,7 @@ int MOSRotating::ReadProperty(const std::string_view &propName, Reader &reader) MatchProperty("SpecialBehaviour_ClearAllAttachables", { // This special property is used to make Attachables work with our limited serialization system, when saving the game. Note that we discard the property value here, because all that matters is whether or not we have the property. reader.ReadPropValue(); - for (std::list::iterator attachableIterator = m_Attachables.begin(); attachableIterator != m_Attachables.end(); ) { + for (auto attachableIterator = m_Attachables.begin(); attachableIterator != m_Attachables.end(); ) { Attachable *attachable = *attachableIterator; ++attachableIterator; delete RemoveAttachable(attachable); @@ -404,7 +404,7 @@ int MOSRotating::Save(Writer &writer) const writer << (*aItr); } */ - for (std::list::const_iterator gItr = m_Gibs.begin(); gItr != m_Gibs.end(); ++gItr) + for (auto gItr = m_Gibs.begin(); gItr != m_Gibs.end(); ++gItr) { writer.NewProperty("AddGib"); writer << (*gItr); @@ -582,11 +582,11 @@ float MOSRotating::RemoveWounds(int numberOfWoundsToRemove, bool includePositive } void MOSRotating::DestroyScriptState() { - for (std::list::const_iterator itr = m_Wounds.begin(); itr != m_Wounds.end(); ++itr) { + for (auto itr = m_Wounds.begin(); itr != m_Wounds.end(); ++itr) { (*itr)->DestroyScriptState(); } - for (std::list::const_iterator itr = m_Attachables.begin(); itr != m_Attachables.end(); ++itr) { + for (auto itr = m_Attachables.begin(); itr != m_Attachables.end(); ++itr) { (*itr)->DestroyScriptState(); } @@ -605,11 +605,11 @@ void MOSRotating::Destroy(bool notInherited) delete m_pAtomGroup; delete m_pDeepGroup; - for (std::list::iterator itr = m_Wounds.begin(); itr != m_Wounds.end(); ++itr) { + for (auto itr = m_Wounds.begin(); itr != m_Wounds.end(); ++itr) { delete (*itr); } - for (std::list::iterator aItr = m_Attachables.begin(); aItr != m_Attachables.end(); ++aItr) { + for (auto aItr = m_Attachables.begin(); aItr != m_Attachables.end(); ++aItr) { if (m_HardcodedAttachableUniqueIDsAndRemovers.find((*aItr)->GetUniqueID()) == m_HardcodedAttachableUniqueIDsAndRemovers.end()) { delete (*aItr); } @@ -1261,10 +1261,10 @@ void MOSRotating::ResetAllTimers() { MovableObject::ResetAllTimers(); - for (std::list::iterator emitter = m_Wounds.begin(); emitter != m_Wounds.end(); ++emitter) + for (auto emitter = m_Wounds.begin(); emitter != m_Wounds.end(); ++emitter) (*emitter)->ResetAllTimers(); - for (std::list::iterator attachable = m_Attachables.begin(); attachable != m_Attachables.end(); ++attachable) + for (auto attachable = m_Attachables.begin(); attachable != m_Attachables.end(); ++attachable) (*attachable)->ResetAllTimers(); } @@ -1543,7 +1543,7 @@ void MOSRotating::PostTravel() Attachable *attachable; - for (std::list::iterator attachableIterator = m_Attachables.begin(); attachableIterator != m_Attachables.end(); ) { + for (auto attachableIterator = m_Attachables.begin(); attachableIterator != m_Attachables.end(); ) { attachable = *attachableIterator; RTEAssert(attachable, "Broken Attachable in PostTravel!"); ++attachableIterator; @@ -1614,11 +1614,11 @@ void MOSRotating::Update() { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void MOSRotating::PostUpdate() { - for (std::list::const_iterator itr = m_Wounds.begin(); itr != m_Wounds.end(); ++itr) { + for (auto itr = m_Wounds.begin(); itr != m_Wounds.end(); ++itr) { (*itr)->PostUpdate(); } - for (std::list::const_iterator itr = m_Attachables.begin(); itr != m_Attachables.end(); ++itr) { + for (auto itr = m_Attachables.begin(); itr != m_Attachables.end(); ++itr) { (*itr)->PostUpdate(); } @@ -1774,7 +1774,7 @@ void MOSRotating::RemoveAndDeleteAttachable(Attachable *attachable) { void MOSRotating::RemoveOrDestroyAllAttachables(bool destroy) { Attachable *attachable; - for (std::list::iterator attachableIterator = m_Attachables.begin(); attachableIterator != m_Attachables.end(); ) { + for (auto attachableIterator = m_Attachables.begin(); attachableIterator != m_Attachables.end(); ) { attachable = *attachableIterator; RTEAssert(attachable, "Broken Attachable!"); ++attachableIterator; diff --git a/Entities/MovableObject.cpp b/Entities/MovableObject.cpp index c6df5f591..2098edba4 100644 --- a/Entities/MovableObject.cpp +++ b/Entities/MovableObject.cpp @@ -23,6 +23,7 @@ #include "Actor.h" #include "SLTerrain.h" +#include "Base64/base64.h" #include "tracy/Tracy.hpp" namespace RTE { @@ -89,6 +90,7 @@ void MovableObject::Clear() m_NumberValueMap.clear(); m_ObjectValueMap.clear(); m_ThreadedLuaState = nullptr; + m_ForceIntoMasterLuaState = false; m_ScriptObjectName.clear(); m_ScreenEffectFile.Reset(); m_pScreenEffect = 0; @@ -127,18 +129,17 @@ void MovableObject::Clear() } LuaStateWrapper & MovableObject::GetAndLockStateForScript(const std::string &scriptPath, const LuaFunction *function) { - // Initialize our threaded state if required - if ((function && function->m_ScriptIsMultithreaded) || g_LuaMan.IsScriptMultithreaded(scriptPath)) { - if (m_ThreadedLuaState == nullptr) { - m_ThreadedLuaState = g_LuaMan.GetAndLockFreeScriptState(); - } else { - m_ThreadedLuaState->GetMutex().lock(); - } - return *m_ThreadedLuaState; + if (m_ForceIntoMasterLuaState) { + m_ThreadedLuaState = &g_LuaMan.GetMasterScriptState(); + } + + if (m_ThreadedLuaState == nullptr) { + m_ThreadedLuaState = g_LuaMan.GetAndLockFreeScriptState(); + } else { + m_ThreadedLuaState->GetMutex().lock(); } - g_LuaMan.GetMasterScriptState().GetMutex().lock(); - return g_LuaMan.GetMasterScriptState(); + return *m_ThreadedLuaState; } ////////////////////////////////////////////////////////////////////////////////////////// @@ -247,6 +248,7 @@ int MovableObject::Create(const MovableObject &reference) m_CanBeSquished = reference.m_CanBeSquished; m_HUDVisible = reference.m_HUDVisible; + m_ForceIntoMasterLuaState = reference.m_ForceIntoMasterLuaState; for (auto &[scriptPath, scriptEnabled] : reference.m_AllLoadedScripts) { LoadScript(scriptPath, scriptEnabled); } @@ -403,6 +405,7 @@ int MovableObject::ReadProperty(const std::string_view &propName, Reader &reader MatchProperty("IgnoreTerrain", { reader >> m_IgnoreTerrain; }); MatchProperty("SimUpdatesBetweenScriptedUpdates", { reader >> m_SimUpdatesBetweenScriptedUpdates; }); MatchProperty("AddCustomValue", { ReadCustomValueProperty(reader); }); + MatchProperty("ForceIntoMasterLuaState", { reader >> m_ForceIntoMasterLuaState; }); EndPropertyList; } @@ -521,6 +524,9 @@ int MovableObject::Save(Writer &writer) const writer.NewPropertyWithValue(key, value); } + writer.NewProperty("ForceIntoMasterLuaState"); + writer << m_ForceIntoMasterLuaState; + return 0; } @@ -536,12 +542,6 @@ void MovableObject::DestroyScriptState() { m_ThreadedLuaState->UnregisterMO(this); m_ThreadedLuaState = nullptr; } - - { - std::lock_guard lock(g_LuaMan.GetMasterScriptState().GetMutex()); - g_LuaMan.GetMasterScriptState().RunScriptString(m_ScriptObjectName + " = nil;"); - g_LuaMan.GetMasterScriptState().UnregisterMO(this); - } } } @@ -582,25 +582,13 @@ int MovableObject::LoadScript(const std::string &scriptPath, bool loadAsEnabledS return -4; } - bool scriptMultithreaded = g_LuaMan.IsScriptMultithreaded(scriptPath); for (const auto &[functionName, functionObject] : scriptFileFunctions) { LuaFunction& luaFunction = m_FunctionsAndScripts.at(functionName).emplace_back(); luaFunction.m_ScriptIsEnabled = loadAsEnabledScript; - luaFunction.m_ScriptIsMultithreaded = scriptMultithreaded; luaFunction.m_LuaFunction = std::unique_ptr(functionObject); } if (ObjectScriptsInitialized()) { - if (scriptMultithreaded && m_ThreadedLuaState == nullptr) { - m_ThreadedLuaState = &usedState; - std::lock_guard lock(m_ThreadedLuaState->GetMutex()); - m_ThreadedLuaState->RegisterMO(this); - m_ThreadedLuaState->SetTempEntity(this); - if (m_ThreadedLuaState->RunScriptString("_ScriptedObjects = _ScriptedObjects or {}; " + m_ScriptObjectName + " = To" + GetClassName() + "(LuaMan.TempEntity); ") < 0) { - RTEAbort("Failed to initialize object scripts for " + GetModuleAndPresetName() + ". Please report this to a developer."); - } - } - if (RunFunctionOfScript(scriptPath, "Create") < 0) { return -5; } @@ -643,25 +631,12 @@ int MovableObject::ReloadScripts() { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// int MovableObject::InitializeObjectScripts() { - auto createScriptedObjectInState = [&](LuaStateWrapper &luaState) { - luaState.SetTempEntity(this); - if (luaState.RunScriptString("_ScriptedObjects = _ScriptedObjects or {}; " + m_ScriptObjectName + " = To" + GetClassName() + "(LuaMan.TempEntity); ") < 0) { - RTEAbort("Failed to initialize object scripts for " + GetModuleAndPresetName() + ". Please report this to a developer."); - } - }; - + std::lock_guard lock(m_ThreadedLuaState->GetMutex()); m_ScriptObjectName = "_ScriptedObjects[\"" + std::to_string(m_UniqueID) + "\"]"; - - { - std::lock_guard lock(g_LuaMan.GetMasterScriptState().GetMutex()); - g_LuaMan.GetMasterScriptState().RegisterMO(this); - createScriptedObjectInState(g_LuaMan.GetMasterScriptState()); - } - - if (m_ThreadedLuaState) { - std::lock_guard lock(m_ThreadedLuaState->GetMutex()); - m_ThreadedLuaState->RegisterMO(this); - createScriptedObjectInState(*m_ThreadedLuaState); + m_ThreadedLuaState->RegisterMO(this); + m_ThreadedLuaState->SetTempEntity(this); + if (m_ThreadedLuaState->RunScriptString("_ScriptedObjects = _ScriptedObjects or {}; " + m_ScriptObjectName + " = To" + GetClassName() + "(LuaMan.TempEntity); ") < 0) { + RTEAbort("Failed to initialize object scripts for " + GetModuleAndPresetName() + ". Please report this to a developer."); } if (!m_FunctionsAndScripts.at("Create").empty() && RunScriptedFunctionInAppropriateScripts("Create", false, true) < 0) { @@ -712,7 +687,7 @@ void MovableObject::EnableOrDisableAllScripts(bool enableScripts) { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// -int MovableObject::RunScriptedFunctionInAppropriateScripts(const std::string &functionName, bool runOnDisabledScripts, bool stopOnError, const std::vector &functionEntityArguments, const std::vector &functionLiteralArguments, const std::vector &functionObjectArguments, ThreadScriptsToRun scriptsToRun) { +int MovableObject::RunScriptedFunctionInAppropriateScripts(const std::string &functionName, bool runOnDisabledScripts, bool stopOnError, const std::vector &functionEntityArguments, const std::vector &functionLiteralArguments, const std::vector &functionObjectArguments) { int status = 0; auto itr = m_FunctionsAndScripts.find(functionName); @@ -728,13 +703,6 @@ int MovableObject::RunScriptedFunctionInAppropriateScripts(const std::string &fu ZoneScoped; ZoneText(functionName.c_str(), functionName.length()); for (const LuaFunction &luaFunction : itr->second) { - bool scriptIsSuitableForThread = scriptsToRun == ThreadScriptsToRun::SingleThreaded ? !luaFunction.m_ScriptIsMultithreaded : - scriptsToRun == ThreadScriptsToRun::MultiThreaded ? luaFunction.m_ScriptIsMultithreaded : - true; - if (!scriptIsSuitableForThread) { - continue; - } - const LuabindObjectWrapper *luabindObjectWrapper = luaFunction.m_LuaFunction.get(); if (runOnDisabledScripts || luaFunction.m_ScriptIsEnabled) { LuaStateWrapper& usedState = GetAndLockStateForScript(luabindObjectWrapper->GetFilePath(), &luaFunction); @@ -1063,7 +1031,7 @@ void MovableObject::Draw(BITMAP* targetBitmap, const Vector& targetPos, DrawMode ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// -int MovableObject::UpdateScripts(ThreadScriptsToRun scriptsToRun) { +int MovableObject::UpdateScripts() { m_SimUpdatesSinceLastScriptedUpdate++; if (m_AllLoadedScripts.empty()) { @@ -1082,7 +1050,7 @@ int MovableObject::UpdateScripts(ThreadScriptsToRun scriptsToRun) { m_SimUpdatesSinceLastScriptedUpdate = 0; if (status >= 0) { - status = RunScriptedFunctionInAppropriateScripts("Update", false, true, {}, {}, {}, scriptsToRun); + status = RunScriptedFunctionInAppropriateScripts("Update", false, true, {}, {}, {}); } return status; @@ -1102,6 +1070,18 @@ const std::string & MovableObject::GetStringValue(const std::string &key) const ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// +std::string MovableObject::GetEncodedStringValue(const std::string &key) const +{ + auto itr = m_StringValueMap.find(key); + if (itr == m_StringValueMap.end()) { + return ms_EmptyString; + } + + return base64_decode(itr->second); +} + +///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + double MovableObject::GetNumberValue(const std::string& key) const { auto itr = m_NumberValueMap.find(key); @@ -1133,6 +1113,13 @@ void MovableObject::SetStringValue(const std::string &key, const std::string &va ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// +void MovableObject::SetEncodedStringValue(const std::string &key, const std::string &value) +{ + m_StringValueMap[key] = base64_encode(value, true); +} + +///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + void MovableObject::SetNumberValue(const std::string &key, double value) { m_NumberValueMap[key] = value; diff --git a/Entities/MovableObject.h b/Entities/MovableObject.h index 95d78acdd..09d73642c 100644 --- a/Entities/MovableObject.h +++ b/Entities/MovableObject.h @@ -212,7 +212,7 @@ enum MOType /// Optional vector of entity pointers that should be passed into the Lua function. Their internal Lua states will not be accessible. Defaults to empty. /// Optional vector of strings, that should be passed into the Lua function. Entries must be surrounded with escaped quotes (i.e.`\"`) they'll be passed in as-is, allowing them to act as booleans, etc.. Defaults to empty. /// An error return value signaling success or any particular failure. Anything below 0 is an error signal. - int RunScriptedFunctionInAppropriateScripts(const std::string &functionName, bool runOnDisabledScripts = false, bool stopOnError = false, const std::vector &functionEntityArguments = std::vector(), const std::vector &functionLiteralArguments = std::vector(), const std::vector &functionObjectArguments = std::vector(), ThreadScriptsToRun scriptsToRun = ThreadScriptsToRun::Both); + int RunScriptedFunctionInAppropriateScripts(const std::string &functionName, bool runOnDisabledScripts = false, bool stopOnError = false, const std::vector &functionEntityArguments = std::vector(), const std::vector &functionLiteralArguments = std::vector(), const std::vector &functionObjectArguments = std::vector()); /// /// Cleans up and destroys the script state of this object, calling the Destroy callback in lua @@ -1517,9 +1517,8 @@ enum MOType /// /// Updates this MovableObject's Lua scripts. /// - /// Whether to run this objects single-threaded or multi-threaded scripts. /// An error return value signaling success or any particular failure. Anything below 0 is an error signal. - virtual int UpdateScripts(ThreadScriptsToRun scriptsToRun); + virtual int UpdateScripts(); /// /// Gets a const reference to this MOSRotating's map of string values. @@ -1540,6 +1539,13 @@ enum MOType /// The value associated with the key. const std::string & GetStringValue(const std::string &key) const; + /// + /// Returns an encoded string value associated with the specified key or "" if it does not exist. + /// + /// Key to retrieve value. + /// The value associated with the key. + std::string GetEncodedStringValue(const std::string &key) const; + /// /// Returns the number value associated with the specified key or 0 if it does not exist. /// @@ -1561,6 +1567,13 @@ enum MOType /// The new value to be associated with the key. void SetStringValue(const std::string &key, const std::string &value); + /// + /// Sets the string value associated with the specified key. + /// + /// Key to retrieve value. + /// The new value to be associated with the key. + void SetEncodedStringValue(const std::string &key, const std::string &value); + /// /// Sets the number value associated with the specified key. /// @@ -1885,9 +1898,9 @@ enum MOType bool DrawToTerrain(SLTerrain *terrain); /// - /// Used to get the Lua state that handles our multithread-safe scripts. + /// Used to get the Lua state that handles our scripts. /// - /// Our lua state. Can potentially be nullptr. + /// Our lua state. Can potentially be nullptr if we're not setup yet. LuaStateWrapper* GetLuaState() { return m_ThreadedLuaState; } /// @@ -2057,10 +2070,10 @@ enum MOType bool m_IsTraveling; //!< Prevents self-intersection while traveling. - LuaStateWrapper *m_ThreadedLuaState; //!< The lua state that will runs our multithreaded lua scripts. + LuaStateWrapper *m_ThreadedLuaState; //!< The lua state that will runs our lua scripts. + bool m_ForceIntoMasterLuaState; //!< This is awful, and only exists for automovers because they mangle global state all over the place. TODO - change automovers to use messages. struct LuaFunction { - bool m_ScriptIsMultithreaded; //!< Whether this function is in a script with the --[[MULTITHREAD]]- thread safety tag. bool m_ScriptIsEnabled; //!< Whether this function is in an enabled script. std::unique_ptr m_LuaFunction; //!< The lua function itself. }; @@ -2069,7 +2082,7 @@ enum MOType std::unordered_map m_AllLoadedScripts; //!< A map of script paths to the enabled state of the given script. std::unordered_map> m_FunctionsAndScripts; //!< A map of function names to vectors of Lua functions. Used to maintain script execution order and avoid extraneous Lua calls. - volatile bool m_RequestedSyncedUpdate; //!< For optimisation purposes, multithreaded scripts explicitly request a synced update if they want one. + volatile bool m_RequestedSyncedUpdate; //!< For optimisation purposes, scripts explicitly request a synced update if they want one. std::unordered_map m_StringValueMap; // m_NumberValueMap; //w, m_MainBitmap->h); @@ -217,15 +216,11 @@ namespace RTE { RTEAbort(std::string("Failed to save SceneLayerImpl bitmap to path and name: " + bitmapPath)); } destroy_bitmap(bitmapToSave); - if (doAsyncSaves) { - g_ActivityMan.DecrementSavingThreadCount(); - } }; m_BitmapFile.SetDataPath(bitmapPath); if (doAsyncSaves) { - std::thread saveThread(saveLayerBitmap, outputBitmap); - saveThread.detach(); + g_ActivityMan.GetSaveGameTask().push_back( g_ThreadMan.GetBackgroundThreadPool().submit(saveLayerBitmap, outputBitmap) ); } else { saveLayerBitmap(outputBitmap); } @@ -293,8 +288,8 @@ namespace RTE { void SceneLayerImpl::ClearBitmap(ColorKeys clearTo) { RTEAssert(m_MainBitmapOwned, "Bitmap not owned! We shouldn't be clearing this!"); - if (m_BitmapClearThread.joinable()) { - m_BitmapClearThread.join(); + if (m_BitmapClearTask.valid()) { + m_BitmapClearTask.wait(); } if (m_LastClearColor != clearTo) { @@ -306,12 +301,10 @@ namespace RTE { std::swap(m_MainBitmap, m_BackBitmap); // Start a new thread to clear the backbuffer bitmap asynchronously. - m_BitmapClearThread = std::thread([this, clearTo](BITMAP *bitmap, std::vector drawings) { + m_BitmapClearTask = g_ThreadMan.GetPriorityThreadPool().submit([this, clearTo](BITMAP *bitmap, std::vector drawings) { ClearDrawings(bitmap, drawings, clearTo); }, m_BackBitmap, m_Drawings); - m_BitmapClearThread.detach(); - m_Drawings.clear(); // This was copied into the new thread, so can be safely deleted. } diff --git a/Entities/SceneLayer.h b/Entities/SceneLayer.h index fba9b39d2..9b91cabe9 100644 --- a/Entities/SceneLayer.h +++ b/Entities/SceneLayer.h @@ -282,7 +282,7 @@ namespace RTE { BITMAP *m_BackBitmap; //!< The backbuffer BITMAP of this SceneLayer. // We use two bitmaps, as a backbuffer. While the main bitmap is being used, the secondary bitmap will be cleared on a separate thread. This is because we tend to want to clear some scene layers every frame and that is costly. - std::thread m_BitmapClearThread; //!< Thread for clearing BITMAP in background. + std::future m_BitmapClearTask; //!< Task for clearing BITMAP async in background. ColorKeys m_LastClearColor; //!< The last color we cleared this SceneLayer to. std::vector m_Drawings; //!< All the areas drawn within on this SceneLayer since the last clear. diff --git a/Lua/LuaAdapterDefinitions.h b/Lua/LuaAdapterDefinitions.h index 15ead6519..f037d6c9c 100644 --- a/Lua/LuaAdapterDefinitions.h +++ b/Lua/LuaAdapterDefinitions.h @@ -489,6 +489,23 @@ namespace RTE { /// The class name of the Entity to reload. /// Whether or not the Entity was reloaded. static bool ReloadEntityPreset2(PresetMan &presetMan, const std::string &presetName, const std::string &className); + + /// + /// Gets a list all previously read in (defined) Entities which are associated with a specific group. + /// + /// The group to look for. "All" will look in all. + /// The name of the least common denominator type of the Entities you want. "All" will look at all types. + /// Whether to only get those of one specific DataModule (0-n), or all (-1). + /// The list of all Entities with the given group and type in the module. + static std::list * GetAllEntitiesOfGroup(PresetMan &presetMan, const std::string &group, const std::string &type, int whichModule); + static std::list * GetAllEntitiesOfGroup2(PresetMan &presetMan, const std::string &group, const std::string &type) { return GetAllEntitiesOfGroup(presetMan, group, type, -1); } + static std::list * GetAllEntitiesOfGroup3(PresetMan &presetMan, const std::string &group) { return GetAllEntitiesOfGroup2(presetMan, group, "All"); } + + /// + /// Gets a list all previously read in (defined) Entities. + /// + /// The list of all Entities. + static std::list * GetAllEntities(PresetMan &presetMan) { return GetAllEntitiesOfGroup3(presetMan, "All"); } }; #pragma endregion diff --git a/Lua/LuaAdapters.cpp b/Lua/LuaAdapters.cpp index 0c0548ff5..61bf620d2 100644 --- a/Lua/LuaAdapters.cpp +++ b/Lua/LuaAdapters.cpp @@ -623,6 +623,14 @@ namespace RTE { return ReloadEntityPreset1(presetMan, presetName, className, ""); } +///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + std::list * LuaAdaptersPresetMan::GetAllEntitiesOfGroup(PresetMan &presetMan, const std::string &group, const std::string &type, int whichModule) { + std::list *entityList = new std::list(); + presetMan.GetAllOfGroup(*entityList, group, type, whichModule); + return entityList; + } + ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// const std::list * LuaAdaptersSceneMan::WrapBoxes(SceneMan &sceneMan, const Box &boxToWrap) { diff --git a/Lua/LuaBindingsEntities.cpp b/Lua/LuaBindingsEntities.cpp index 882395a71..4f13f0cd6 100644 --- a/Lua/LuaBindingsEntities.cpp +++ b/Lua/LuaBindingsEntities.cpp @@ -1038,9 +1038,11 @@ namespace RTE { .def("DisableScript", &LuaAdaptersMovableObject::DisableScript) .def("EnableOrDisableAllScripts", &MovableObject::EnableOrDisableAllScripts) .def("GetStringValue", &MovableObject::GetStringValue) + .def("GetEncodedStringValue", &MovableObject::GetEncodedStringValue) .def("GetNumberValue", &MovableObject::GetNumberValue) .def("GetObjectValue", &MovableObject::GetObjectValue) .def("SetStringValue", &MovableObject::SetStringValue) + .def("SetEncodedStringValue", &MovableObject::SetEncodedStringValue) .def("SetNumberValue", &MovableObject::SetNumberValue) .def("SetObjectValue", &MovableObject::SetObjectValue) .def("RemoveStringValue", &MovableObject::RemoveStringValue) diff --git a/Lua/LuaBindingsManagers.cpp b/Lua/LuaBindingsManagers.cpp index f75cbfd9d..11b0424f8 100644 --- a/Lua/LuaBindingsManagers.cpp +++ b/Lua/LuaBindingsManagers.cpp @@ -203,6 +203,10 @@ namespace RTE { .def("ReadReflectedPreset", &PresetMan::ReadReflectedPreset) .def("ReloadEntityPreset", &LuaAdaptersPresetMan::ReloadEntityPreset1) .def("ReloadEntityPreset", &LuaAdaptersPresetMan::ReloadEntityPreset2) + .def("GetAllEntities", &LuaAdaptersPresetMan::GetAllEntities, luabind::adopt(luabind::result) + luabind::return_stl_iterator) + .def("GetAllEntitiesOfGroup", &LuaAdaptersPresetMan::GetAllEntitiesOfGroup, luabind::adopt(luabind::result) + luabind::return_stl_iterator) + .def("GetAllEntitiesOfGroup", &LuaAdaptersPresetMan::GetAllEntitiesOfGroup2, luabind::adopt(luabind::result) + luabind::return_stl_iterator) + .def("GetAllEntitiesOfGroup", &LuaAdaptersPresetMan::GetAllEntitiesOfGroup3, luabind::adopt(luabind::result) + luabind::return_stl_iterator) .def("ReloadAllScripts", &PresetMan::ReloadAllScripts) .def("IsModuleOfficial", &PresetMan::IsModuleOfficial) .def("IsModuleUserdata", &PresetMan::IsModuleUserdata) diff --git a/Main.cpp b/Main.cpp index 93038d237..7a0709752 100644 --- a/Main.cpp +++ b/Main.cpp @@ -46,6 +46,7 @@ #include "CameraMan.h" #include "ActivityMan.h" #include "PrimitiveMan.h" +#include "ThreadMan.h" #include "tracy/Tracy.hpp" @@ -61,6 +62,7 @@ namespace RTE { /// Initializes all the essential managers. /// void InitializeManagers() { + ThreadMan::Construct(); TimerMan::Construct(); PresetMan::Construct(); SettingsMan::Construct(); @@ -84,6 +86,7 @@ namespace RTE { ActivityMan::Construct(); LoadingScreen::Construct(); + g_ThreadMan.Initialize(); g_SettingsMan.Initialize(); g_WindowMan.Initialize(); diff --git a/Managers/ActivityMan.cpp b/Managers/ActivityMan.cpp index d2dc37637..fd08699e5 100644 --- a/Managers/ActivityMan.cpp +++ b/Managers/ActivityMan.cpp @@ -11,6 +11,7 @@ #include "PerformanceMan.h" #include "PostProcessMan.h" #include "MetaMan.h" +#include "ThreadMan.h" #include "GAScripted.h" #include "SLTerrain.h" @@ -26,6 +27,7 @@ #include "MultiplayerServerLobby.h" #include "MultiplayerGame.h" + namespace RTE { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// @@ -35,8 +37,7 @@ namespace RTE { m_DefaultActivityName = "Tutorial Mission"; m_Activity = nullptr; m_StartActivity = nullptr; - m_ActiveSavingThreadCount = 0; - m_IsLoading = false; + m_SaveGameTask = BS::multi_future(); m_InActivity = false; m_ActivityNeedsRestart = false; m_ActivityNeedsResume = false; @@ -74,10 +75,8 @@ namespace RTE { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// bool ActivityMan::SaveCurrentGame(const std::string &fileName) { - if (IsSaving() || m_IsLoading) { - RTEError::ShowMessageBox("Cannot Save Game\nA game is currently being saved/loaded, try again shortly."); - return false; - } + m_SaveGameTask.wait(); + m_SaveGameTask = BS::multi_future(); Scene *scene = g_SceneMan.GetScene(); GAScripted *activity = dynamic_cast(GetActivity()); @@ -96,8 +95,6 @@ namespace RTE { return false; } - IncrementSavingThreadCount(); - // We need a copy of our scene, because we have to do some fixup to remove PLACEONLOAD items and only keep the current MovableMan state. std::unique_ptr modifiableScene(dynamic_cast(scene->Clone())); @@ -131,15 +128,13 @@ namespace RTE { writer->NewPropertyWithValue("PlaceUnitsIfSceneIsRestarted", g_SceneMan.GetPlaceUnitsOnLoad()); writer->NewPropertyWithValue("Scene", modifiableScene.get()); - auto saveWriterData = [this](std::unique_ptr writerToSave) { - // Explicitly flush to disk. This'll happen anyways at the end of this scope, but otherwise this lambda looks rather empty :) + auto saveWriterData = [](Writer* writerToSave) { writerToSave->EndWrite(); - DecrementSavingThreadCount(); + delete writerToSave; }; - // Make a thread to flush the data to the disk, and detach it so it can run concurrently with the game simulation. - std::thread saveThread(saveWriterData, std::move(writer)); - saveThread.detach(); + // For some reason I can't std::move a unique ptr in, so just releasing and deleting manually... + m_SaveGameTask.push_back( g_ThreadMan.GetBackgroundThreadPool().submit(saveWriterData, writer.release()) ); // We didn't transfer ownership, so we must be very careful that sceneAltered's deletion doesn't touch the stuff we got from MovableMan. modifiableScene->ClearPlacedObjectSet(Scene::PlacedObjectSets::PLACEONLOAD, false); @@ -150,11 +145,8 @@ namespace RTE { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// - bool ActivityMan::LoadAndLaunchGame(const std::string &fileName) { - if (IsSaving() || m_IsLoading) { - RTEError::ShowMessageBox("Cannot Load Game\nA game is currently being saved/loaded, try again shortly."); - return false; - } + bool ActivityMan::LoadAndLaunchGame(const std::string &fileName) { + m_SaveGameTask.wait(); std::string saveFilePath = g_PresetMan.GetFullModulePath(c_UserScriptedSavesModuleName) + "/" + fileName + "/Save.ini"; @@ -164,7 +156,6 @@ namespace RTE { } Reader reader(saveFilePath, true, nullptr, false); - m_IsLoading = true; std::unique_ptr scene(std::make_unique()); std::unique_ptr activity(std::make_unique()); @@ -198,8 +189,6 @@ namespace RTE { g_SceneMan.SetSceneToLoad(originalScenePresetName, placeObjectsIfSceneIsRestarted, placeUnitsIfSceneIsRestarted); g_ConsoleMan.PrintString("SYSTEM: Game \"" + fileName + "\" loaded!"); - - m_IsLoading = false; return true; } diff --git a/Managers/ActivityMan.h b/Managers/ActivityMan.h index fe018a101..d8f4da50a 100644 --- a/Managers/ActivityMan.h +++ b/Managers/ActivityMan.h @@ -4,6 +4,8 @@ #include "Singleton.h" #include "Activity.h" +#include "BS_thread_pool.hpp" + #define g_ActivityMan ActivityMan::Instance() namespace RTE { @@ -49,20 +51,10 @@ namespace RTE { Activity * GetActivity() const { return m_Activity.get(); } /// - /// Gets whether or not there is a game currently being saved. - /// - /// Whether or not there is a game currently being saved. - bool IsSaving() const { return m_ActiveSavingThreadCount > 0; } - - /// - /// Increments the saving thread count. - /// - void IncrementSavingThreadCount() { m_ActiveSavingThreadCount++; } - - /// - /// Decrements the saving thread count. + /// Gets the async save game task. /// - void DecrementSavingThreadCount() { m_ActiveSavingThreadCount--; } + /// The savegame task. + BS::multi_future& GetSaveGameTask() { return m_SaveGameTask; } /// /// Indicates whether the game is currently running or not (not editing, over or paused). @@ -288,8 +280,7 @@ namespace RTE { std::unique_ptr m_Activity; //!< The currently active Activity. std::unique_ptr m_StartActivity; //!< The starting condition of the next Activity to be (re)started. - std::atomic m_ActiveSavingThreadCount; //!< The number of threads currently saving. - bool m_IsLoading; //! Whether or not a game is loading. + BS::multi_future m_SaveGameTask; //!< The current save game task. bool m_InActivity; //!< Whether we are currently in game (as in, not in the main menu or any other out-of-game menus), regardless of its state. bool m_ActivityNeedsRestart; //!< Whether the current Activity needs to be restarted. diff --git a/Managers/FrameMan.cpp b/Managers/FrameMan.cpp index cffbf6a8c..bdeb5e43e 100644 --- a/Managers/FrameMan.cpp +++ b/Managers/FrameMan.cpp @@ -22,6 +22,8 @@ #include "GLCheck.h" #include "glad/gl.h" +#include "tracy/Tracy.hpp" + namespace RTE { void BitmapDeleter::operator()(BITMAP *bitmap) const { destroy_bitmap(bitmap); } @@ -801,6 +803,8 @@ namespace RTE { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void FrameMan::Draw() { + ZoneScopedN("Draw"); + // Count how many split screens we'll need int screenCount = (m_HSplit ? 2 : 1) * (m_VSplit ? 2 : 1); RTEAssert(screenCount <= 1 || m_PlayerScreen, "Splitscreen surface not ready when needed!"); diff --git a/Managers/LuaMan.cpp b/Managers/LuaMan.cpp index cf2fbbf25..ba47dbd64 100644 --- a/Managers/LuaMan.cpp +++ b/Managers/LuaMan.cpp @@ -1,6 +1,9 @@ #include "LuaMan.h" + #include "LuabindObjectWrapper.h" #include "LuaBindingRegisterDefinitions.h" +#include "ThreadMan.h" + #include "tracy/Tracy.hpp" #include "tracy/TracyLua.hpp" @@ -286,6 +289,8 @@ namespace RTE { void LuaMan::Initialize() { m_MasterScriptState.Initialize(); + + m_ScriptStates = std::vector(std::thread::hardware_concurrency()); for (LuaStateWrapper &luaState : m_ScriptStates) { luaState.Initialize(); } @@ -345,7 +350,7 @@ namespace RTE { return &(*itr);*/ int ourState = m_LastAssignedLuaState; - m_LastAssignedLuaState = (m_LastAssignedLuaState + 1) % c_NumThreadedLuaStates; + m_LastAssignedLuaState = (m_LastAssignedLuaState + 1) % m_ScriptStates.size(); bool success = m_ScriptStates[ourState].GetMutex().try_lock(); RTEAssert(success, "Script mutex was already locked while in a non-multithreaded environment!"); @@ -353,47 +358,10 @@ namespace RTE { return &m_ScriptStates[ourState]; } -///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// - - bool LuaMan::IsScriptMultithreaded(const std::string &scriptPath) { - // First check our cache - auto itr = m_ScriptMultithreadedtyMap.find(scriptPath); - if (itr != m_ScriptMultithreadedtyMap.end()) { - return itr->second; - } - - // Actually open the file and check if it has the multithread-safe mark - std::ifstream scriptFile = std::ifstream(scriptPath.c_str()); - if (!scriptFile.good()) { - m_ScriptMultithreadedtyMap.insert({ scriptPath, false }); - return false; - } - - std::string::size_type commentPos; - bool inBlockComment = false; - while (!scriptFile.eof()) { - char rawLine[512]; - scriptFile.getline(rawLine, 512); - std::string line = rawLine; - - if (line.find("--[[MULTITHREAD]]--", 0) != std::string::npos) { - m_ScriptMultithreadedtyMap.insert({scriptPath, true}); - return true; - } - } - - m_ScriptMultithreadedtyMap.insert({scriptPath, false}); - return false; - } - ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void LuaMan::ClearUserModuleCache() { - if (m_GCThread.joinable()) { - m_GCThread.join(); - } - - m_ScriptMultithreadedtyMap.clear(); + m_GarbageCollectionTask.wait(); m_MasterScriptState.ClearLuaScriptCache(); for (LuaStateWrapper& luaState : m_ScriptStates) { @@ -1098,9 +1066,7 @@ namespace RTE { } // Make sure a GC run isn't happening while we try to apply deletions - if (m_GCThread.joinable()) { - m_GCThread.join(); - } + m_GarbageCollectionTask.wait(); // Apply all deletions queued from lua LuabindObjectWrapper::ApplyQueuedDeletions(); @@ -1111,25 +1077,25 @@ namespace RTE { void LuaMan::StartAsyncGarbageCollection() { ZoneScoped; - // Start a new thread to perform the GC run. - m_GCThread = std::thread([this]() { - std::vector allStates; - allStates.reserve(m_ScriptStates.size() + 1); - - allStates.push_back(&m_MasterScriptState); - for (LuaStateWrapper& wrapper : m_ScriptStates) { - allStates.push_back(&wrapper); - } + std::vector allStates; + allStates.reserve(m_ScriptStates.size() + 1); - std::for_each(std::execution::par, allStates.begin(), allStates.end(), - [&](LuaStateWrapper* luaState) { + allStates.push_back(&m_MasterScriptState); + for (LuaStateWrapper& wrapper : m_ScriptStates) { + allStates.push_back(&wrapper); + } + + m_GarbageCollectionTask = BS::multi_future(); + for (LuaStateWrapper* luaState : allStates) { + m_GarbageCollectionTask.push_back( + g_ThreadMan.GetPriorityThreadPool().submit([luaState]() { ZoneScopedN("Lua Garbage Collection"); std::lock_guard lock(luaState->GetMutex()); lua_gc(luaState->GetLuaState(), LUA_GCSTEP, 100); lua_gc(luaState->GetLuaState(), LUA_GCSTOP, 0); - } + }) ); - }); + } } ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// diff --git a/Managers/LuaMan.h b/Managers/LuaMan.h index 5958ccf36..f16aae02d 100644 --- a/Managers/LuaMan.h +++ b/Managers/LuaMan.h @@ -6,6 +6,8 @@ #include "RTETools.h" #include "PerformanceMan.h" +#include "BS_thread_pool.hpp" + #define g_LuaMan LuaMan::Instance() struct lua_State; @@ -311,8 +313,7 @@ namespace RTE { RandomGenerator m_RandomGenerator; //!< The random number generator used for this lua state. }; - static constexpr int c_NumThreadedLuaStates = 16; - typedef std::array LuaStatesArray; + typedef std::vector LuaStatesArray; /// /// The singleton manager of each Lua state. @@ -386,12 +387,6 @@ namespace RTE { /// A script state. LuaStateWrapper * GetAndLockFreeScriptState(); - /// - /// Returns whether a script is safe to run in a multithreaded manner. - /// - /// Whether the script is thread-safe. - bool IsScriptMultithreaded(const std::string &scriptPath); - /// /// Clears internal Lua package tables from all user-defined modules. Those must be reloaded with ReloadAllScripts(). /// @@ -508,14 +503,12 @@ namespace RTE { LuaStateWrapper m_MasterScriptState; LuaStatesArray m_ScriptStates; - std::unordered_map m_ScriptMultithreadedtyMap; - std::vector> m_ScriptCallbacks; //!< A list of callback functions we'll trigger before processing lua scripts. This allows other threads (i.e pathing requests) to safely trigger callbacks in lua std::mutex m_ScriptCallbacksMutex; //!< Mutex to ensure multiple threads aren't modifying the script callback vector at the same time. int m_LastAssignedLuaState = 0; - std::thread m_GCThread; + BS::multi_future m_GarbageCollectionTask; /// /// Clears all the member variables of this LuaMan, effectively resetting the members of this abstraction level only. diff --git a/Managers/MovableMan.cpp b/Managers/MovableMan.cpp index 8bb5c2ae4..1e35ca30f 100644 --- a/Managers/MovableMan.cpp +++ b/Managers/MovableMan.cpp @@ -33,6 +33,7 @@ #include "SceneMan.h" #include "SettingsMan.h" #include "LuaMan.h" +#include "ThreadMan.h" #include "tracy/Tracy.hpp" @@ -880,7 +881,7 @@ void MovableMan::AddActor(Actor *actorToAdd) { if (actorToAdd->IsStatus(Actor::INACTIVE)) { actorToAdd->SetStatus(Actor::STABLE); } actorToAdd->NotResting(); actorToAdd->NewFrame(); - actorToAdd->SetAge(0); + actorToAdd->SetAge(g_TimerMan.GetDeltaTimeMS() * -1.0f); } { @@ -908,7 +909,7 @@ void MovableMan::AddItem(HeldDevice *itemToAdd) { if (!itemToAdd->IsSetToDelete()) { itemToAdd->MoveOutOfTerrain(g_MaterialGrass); } itemToAdd->NotResting(); itemToAdd->NewFrame(); - itemToAdd->SetAge(0); + itemToAdd->SetAge(g_TimerMan.GetDeltaTimeMS() * -1.0f); } std::lock_guard lock(m_AddedItemsMutex); @@ -931,7 +932,7 @@ void MovableMan::AddParticle(MovableObject *particleToAdd){ //TODO consider moving particles out of grass. It's old code that was removed because it's slow to do this for every particle. particleToAdd->NotResting(); particleToAdd->NewFrame(); - particleToAdd->SetAge(0); + particleToAdd->SetAge(g_TimerMan.GetDeltaTimeMS() * -1.0f); } if (particleToAdd->IsDevice()) { std::lock_guard lock(m_AddedItemsMutex); @@ -1562,55 +1563,55 @@ void MovableMan::RedrawOverlappingMOIDs(MovableObject *pOverlapsThis) ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// -void callLuaFunctionOnMORecursive(MovableObject* mo, const std::string& functionName, const std::vector& functionEntityArguments, const std::vector& functionLiteralArguments, const std::vector& functionObjectArguments, ThreadScriptsToRun scriptsToRun) { +void callLuaFunctionOnMORecursive(MovableObject* mo, const std::string& functionName, const std::vector& functionEntityArguments, const std::vector& functionLiteralArguments, const std::vector& functionObjectArguments) { if (MOSRotating* mosr = dynamic_cast(mo)) { for (auto attachablrItr = mosr->GetAttachableList().begin(); attachablrItr != mosr->GetAttachableList().end(); ) { Attachable* attachable = *attachablrItr; ++attachablrItr; - attachable->RunScriptedFunctionInAppropriateScripts(functionName, false, false, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); - callLuaFunctionOnMORecursive(attachable, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + attachable->RunScriptedFunctionInAppropriateScripts(functionName, false, false, functionEntityArguments, functionLiteralArguments, functionObjectArguments); + callLuaFunctionOnMORecursive(attachable, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments); } for (auto woundItr = mosr->GetWoundList().begin(); woundItr != mosr->GetWoundList().end(); ) { AEmitter* wound = *woundItr; ++woundItr; - wound->RunScriptedFunctionInAppropriateScripts(functionName, false, false, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); - callLuaFunctionOnMORecursive(wound, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + wound->RunScriptedFunctionInAppropriateScripts(functionName, false, false, functionEntityArguments, functionLiteralArguments, functionObjectArguments); + callLuaFunctionOnMORecursive(wound, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments); } } - mo->RunScriptedFunctionInAppropriateScripts(functionName, false, false, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + mo->RunScriptedFunctionInAppropriateScripts(functionName, false, false, functionEntityArguments, functionLiteralArguments, functionObjectArguments); }; ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// -void MovableMan::RunLuaFunctionOnAllMOs(const std::string &functionName, bool includeAdded, const std::vector &functionEntityArguments, const std::vector &functionLiteralArguments, const std::vector &functionObjectArguments, ThreadScriptsToRun scriptsToRun) { +void MovableMan::RunLuaFunctionOnAllMOs(const std::string &functionName, bool includeAdded, const std::vector &functionEntityArguments, const std::vector &functionLiteralArguments, const std::vector &functionObjectArguments) { if (includeAdded) { for (Actor* actor : m_AddedActors) { - callLuaFunctionOnMORecursive(actor, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + callLuaFunctionOnMORecursive(actor, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments); } for (MovableObject *item : m_AddedItems) { - callLuaFunctionOnMORecursive(item, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + callLuaFunctionOnMORecursive(item, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments); } for (MovableObject* particle : m_AddedParticles) { - callLuaFunctionOnMORecursive(particle, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + callLuaFunctionOnMORecursive(particle, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments); } } for (Actor *actor : m_Actors) { - callLuaFunctionOnMORecursive(actor, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + callLuaFunctionOnMORecursive(actor, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments); } for (MovableObject *item : m_Items) { - callLuaFunctionOnMORecursive(item, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + callLuaFunctionOnMORecursive(item, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments); } for (MovableObject* particle : m_Particles) { - callLuaFunctionOnMORecursive(particle, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments, scriptsToRun); + callLuaFunctionOnMORecursive(particle, functionName, functionEntityArguments, functionLiteralArguments, functionObjectArguments); } } @@ -1738,16 +1739,18 @@ void MovableMan::Update() const std::string threadedUpdate = "ThreadedUpdate"; // avoid string reconstruction LuaStatesArray& luaStates = g_LuaMan.GetThreadedScriptStates(); - std::for_each(std::execution::par, luaStates.begin(), luaStates.end(), - [&](LuaStateWrapper& luaState) { + g_ThreadMan.GetPriorityThreadPool().parallelize_loop(luaStates.size(), + [&](int start, int end) { + RTEAssert(start + 1 == end, "Threaded script state being updated across multiple threads!"); + LuaStateWrapper& luaState = luaStates[start]; g_LuaMan.SetThreadLuaStateOverride(&luaState); for (MovableObject *mo : luaState.GetRegisteredMOs()) { - mo->RunScriptedFunctionInAppropriateScripts(threadedUpdate, false, false, {}, {}, {}, ThreadScriptsToRun::MultiThreaded); + mo->RunScriptedFunctionInAppropriateScripts(threadedUpdate, false, false, {}, {}, {}); } g_LuaMan.SetThreadLuaStateOverride(nullptr); - }); + }).wait(); } { @@ -1760,7 +1763,7 @@ void MovableMan::Update() for (MovableObject* mo : luaState.GetRegisteredMOs()) { if (mo->HasRequestedSyncedUpdate()) { - mo->RunScriptedFunctionInAppropriateScripts(syncedUpdate, false, false, {}, {}, {}, ThreadScriptsToRun::MultiThreaded); + mo->RunScriptedFunctionInAppropriateScripts(syncedUpdate, false, false, {}, {}, {}); mo->ResetRequestedSyncedUpdateFlag(); } } @@ -1779,7 +1782,7 @@ void MovableMan::Update() actor->Update(); g_PerformanceMan.StartPerformanceMeasurement(PerformanceMan::ScriptsUpdate); - actor->UpdateScripts(ThreadScriptsToRun::Both); + actor->UpdateScripts(); g_PerformanceMan.StopPerformanceMeasurement(PerformanceMan::ScriptsUpdate); actor->ApplyImpulses(); @@ -1796,7 +1799,7 @@ void MovableMan::Update() (*iIt)->Update(); g_PerformanceMan.StartPerformanceMeasurement(PerformanceMan::ScriptsUpdate); - (*iIt)->UpdateScripts(ThreadScriptsToRun::Both); + (*iIt)->UpdateScripts(); g_PerformanceMan.StopPerformanceMeasurement(PerformanceMan::ScriptsUpdate); (*iIt)->ApplyImpulses(); @@ -1814,7 +1817,7 @@ void MovableMan::Update() particle->Update(); g_PerformanceMan.StartPerformanceMeasurement(PerformanceMan::ScriptsUpdate); - particle->UpdateScripts(ThreadScriptsToRun::Both); + particle->UpdateScripts(); g_PerformanceMan.StopPerformanceMeasurement(PerformanceMan::ScriptsUpdate); particle->ApplyImpulses(); @@ -2052,27 +2055,9 @@ void MovableMan::Update() //////////////////////////////////////////////////////////////////////// // Draw the MO matter and IDs to their layers for next frame - - UpdateDrawMOIDs(g_SceneMan.GetMOIDBitmap()); - - // COUNT MOID USAGE PER TEAM ////////////////////////////////////////////////// - { - int team = Activity::NoTeam; - - for (team = Activity::TeamOne; team < Activity::MaxTeamCount; team++) - m_TeamMOIDCount[team] = 0; - - for (std::vector::iterator itr = m_MOIDIndex.begin(); itr != m_MOIDIndex.end(); ++itr) - { - if (*itr) - { - team = (*itr)->GetTeam(); - - if (team > Activity::NoTeam && team < Activity::MaxTeamCount) - m_TeamMOIDCount[team]++; - } - } - } + m_DrawMOIDsTask = g_ThreadMan.GetPriorityThreadPool().submit([this]() { + UpdateDrawMOIDs(g_SceneMan.GetMOIDBitmap()); + }); //////////////////////////////////////////////////////////////////// @@ -2100,8 +2085,14 @@ void MovableMan::Travel() { ZoneScoped; + if (m_DrawMOIDsTask.valid()) { + m_DrawMOIDsTask.wait(); + } + // Travel Actors { + ZoneScopedN("Actors Travel"); + g_PerformanceMan.StartPerformanceMeasurement(PerformanceMan::ActorsTravel); for (auto aIt = m_Actors.begin(); aIt != m_Actors.end(); ++aIt) { @@ -2119,6 +2110,8 @@ void MovableMan::Travel() // Travel items { + ZoneScopedN("Items Travel"); + for (auto iIt = m_Items.begin(); iIt != m_Items.end(); ++iIt) { if (!((*iIt)->IsUpdated())) @@ -2134,6 +2127,8 @@ void MovableMan::Travel() // Travel particles { + ZoneScopedN("Particles Travel"); + g_PerformanceMan.StartPerformanceMeasurement(PerformanceMan::ParticlesTravel); for (auto parIt = m_Particles.begin(); parIt != m_Particles.end(); ++parIt) { @@ -2163,19 +2158,23 @@ void MovableMan::UpdateControllers() } LuaStatesArray& luaStates = g_LuaMan.GetThreadedScriptStates(); - std::for_each(std::execution::par, luaStates.begin(), luaStates.end(), - [&](LuaStateWrapper &luaState) { + g_ThreadMan.GetPriorityThreadPool().parallelize_loop(luaStates.size(), + [&](int start, int end) { + RTEAssert(start + 1 == end, "Threaded script state being updated across multiple threads!"); + LuaStateWrapper& luaState = luaStates[start]; g_LuaMan.SetThreadLuaStateOverride(&luaState); for (Actor *actor : m_Actors) { - if (actor->GetLuaState() == &luaState) { - actor->GetController()->UpdateAI(ThreadScriptsToRun::MultiThreaded); + if (actor->GetLuaState() == &luaState && actor->GetController()->ShouldUpdateAIThisFrame()) { + actor->RunScriptedFunctionInAppropriateScripts("ThreadedUpdateAI", false, true, {}, {}, {}); } } g_LuaMan.SetThreadLuaStateOverride(nullptr); - }); + }).wait(); for (Actor* actor : m_Actors) { - actor->GetController()->UpdateAI(ThreadScriptsToRun::SingleThreaded); + if (actor->GetController()->ShouldUpdateAIThisFrame()) { + actor->RunScriptedFunctionInAppropriateScripts("UpdateAI", false, true, {}, {}, {}); + } } } g_PerformanceMan.StopPerformanceMeasurement(PerformanceMan::ActorsAI); @@ -2310,6 +2309,20 @@ void MovableMan::UpdateDrawMOIDs(BITMAP *pTargetBitmap) particle->SetAsNoID(); } } + + // COUNT MOID USAGE PER TEAM ////////////////////////////////////////////////// + for (int team = Activity::TeamOne; team < Activity::MaxTeamCount; team++) { + m_TeamMOIDCount[team] = 0; + } + + for (auto itr = m_MOIDIndex.begin(); itr != m_MOIDIndex.end(); ++itr) { + if (*itr) { + int team = (*itr)->GetTeam(); + if (team > Activity::NoTeam && team < Activity::MaxTeamCount) { + m_TeamMOIDCount[team]++; + } + } + } } @@ -2324,14 +2337,30 @@ void MovableMan::Draw(BITMAP *pTargetBitmap, const Vector &targetPos) ZoneScoped; // Draw objects to accumulation bitmap, in reverse order so actors appear on top. - for (std::deque::iterator parIt = m_Particles.begin(); parIt != m_Particles.end(); ++parIt) - (*parIt)->Draw(pTargetBitmap, targetPos); - for (std::deque::reverse_iterator itmIt = m_Items.rbegin(); itmIt != m_Items.rend(); ++itmIt) - (*itmIt)->Draw(pTargetBitmap, targetPos); + { + ZoneScopedN("Particles Draw"); - for (std::deque::reverse_iterator aIt = m_Actors.rbegin(); aIt != m_Actors.rend(); ++aIt) - (*aIt)->Draw(pTargetBitmap, targetPos); + for (std::deque::iterator parIt = m_Particles.begin(); parIt != m_Particles.end(); ++parIt) { + (*parIt)->Draw(pTargetBitmap, targetPos); + } + } + + { + ZoneScopedN("Items Draw"); + + for (std::deque::reverse_iterator itmIt = m_Items.rbegin(); itmIt != m_Items.rend(); ++itmIt) { + (*itmIt)->Draw(pTargetBitmap, targetPos); + } + } + + { + ZoneScopedN("Actors Draw"); + + for (std::deque::reverse_iterator aIt = m_Actors.rbegin(); aIt != m_Actors.rend(); ++aIt) { + (*aIt)->Draw(pTargetBitmap, targetPos); + } + } } diff --git a/Managers/MovableMan.h b/Managers/MovableMan.h index 5b6f3c813..29445a27b 100644 --- a/Managers/MovableMan.h +++ b/Managers/MovableMan.h @@ -944,7 +944,7 @@ class MovableMan : public Singleton, public Serializable { /// /// Runs a lua function on all MOs in the simulation, including owned child MOs. /// - void RunLuaFunctionOnAllMOs(const std::string& functionName, bool includeAdded, const std::vector& functionEntityArguments = std::vector(), const std::vector& functionLiteralArguments = std::vector(), const std::vector& functionObjectArguments = std::vector(), ThreadScriptsToRun scriptsToRun = ThreadScriptsToRun::Both); + void RunLuaFunctionOnAllMOs(const std::string& functionName, bool includeAdded, const std::vector& functionEntityArguments = std::vector(), const std::vector& functionLiteralArguments = std::vector(), const std::vector& functionObjectArguments = std::vector()); /// /// Clears all cached lua functions on all MOs, including owned child MOs. @@ -994,6 +994,9 @@ class MovableMan : public Singleton, public Serializable { // Mutex to ensure actors don't change team roster from seperate threads at the same time std::mutex m_ActorRosterMutex; + // Async to draw MOIDs while rendering + std::future m_DrawMOIDsTask; + // Roster of each team's actors, sorted by their X positions in the scene. Actors not owned here std::list m_ActorRoster[Activity::MaxTeamCount]; // Whether to draw HUD lines between the actors of a specific team diff --git a/Managers/PerformanceMan.cpp b/Managers/PerformanceMan.cpp index c6d5503f6..0f59cc16f 100644 --- a/Managers/PerformanceMan.cpp +++ b/Managers/PerformanceMan.cpp @@ -155,8 +155,6 @@ namespace RTE { std::snprintf(str, sizeof(str), "Sim Updates Since Last Drawn: %i", g_TimerMan.SimUpdatesSinceDrawn()); guiFont->DrawAligned(&drawBitmap, c_StatsOffsetX, c_StatsHeight + 90, str, GUIFont::Left); - if (g_TimerMan.IsOneSimUpdatePerFrame()) { guiFont->DrawAligned(&drawBitmap, c_StatsOffsetX, c_StatsHeight + 100, "ONE Sim Update Per Frame!", GUIFont::Left); } - if (int totalPlayingChannelCount = 0, realPlayingChannelCount = 0; g_AudioMan.GetPlayingChannelCount(&totalPlayingChannelCount, &realPlayingChannelCount)) { std::snprintf(str, sizeof(str), "Sound Channels: %d / %d Real | %d / %d Virtual", realPlayingChannelCount, g_AudioMan.GetTotalRealChannelCount(), totalPlayingChannelCount - realPlayingChannelCount, g_AudioMan.GetTotalVirtualChannelCount()); } diff --git a/Managers/PresetMan.cpp b/Managers/PresetMan.cpp index 86ca9d14c..e788bb2ae 100644 --- a/Managers/PresetMan.cpp +++ b/Managers/PresetMan.cpp @@ -209,7 +209,7 @@ bool PresetMan::LoadAllDataModules() { const DataModule * PresetMan::GetDataModule(int whichModule) { - RTEAssert(whichModule < (int)m_pDataModules.size(), "Tried to access an out of bounds data module number!"); + RTEAssert(whichModule >= 0 && whichModule < (int)m_pDataModules.size(), "Tried to access an out of bounds data module number!"); return m_pDataModules[whichModule]; } @@ -220,12 +220,10 @@ const DataModule * PresetMan::GetDataModule(int whichModule) const std::string PresetMan::GetDataModuleName(int whichModule) { - RTEAssert(whichModule < (int)m_pDataModules.size(), "Tried to access an out of bounds data module number!"); + RTEAssert(whichModule >= 0 && whichModule < (int)m_pDataModules.size(), "Tried to access an out of bounds data module number!"); return m_pDataModules[whichModule]->GetFileName(); } - - ////////////////////////////////////////////////////////////////////////////////////////// // Method: GetModuleID ////////////////////////////////////////////////////////////////////////////////////////// @@ -362,7 +360,7 @@ std::string PresetMan::GetFullModulePath(const std::string &modulePath) const { bool PresetMan::AddEntityPreset(Entity *pEntToAdd, int whichModule, bool overwriteSame, std::string readFromFile) { - RTEAssert(whichModule >= 0 && whichModule < m_pDataModules.size(), "Tried to access an out of bounds data module number!"); + RTEAssert(whichModule >= 0 && whichModule < (int)m_pDataModules.size(), "Tried to access an out of bounds data module number!"); return m_pDataModules[whichModule]->AddEntityPreset(pEntToAdd, overwriteSame, readFromFile); } @@ -429,7 +427,7 @@ const Entity * PresetMan::GetEntityPreset(Reader &reader) { // The reader is aware of which DataModule it is reading within int whichModule = reader.GetReadModuleID(); - RTEAssert(whichModule >= 0 && whichModule < m_pDataModules.size(), "Reader has an out of bounds module number!"); + RTEAssert(whichModule >= 0 && whichModule < (int)m_pDataModules.size(), "Reader has an out of bounds module number!"); std::string ClassName; const Entity::ClassInfo *pClass = 0; @@ -490,7 +488,7 @@ Entity * PresetMan::ReadReflectedPreset(Reader &reader) { // The reader is aware of which DataModule it's reading within int whichModule = reader.GetReadModuleID(); - RTEAssert(whichModule >= 0 && whichModule < m_pDataModules.size(), "Reader has an out of bounds module number!"); + RTEAssert(whichModule >= 0 && whichModule < (int)m_pDataModules.size(), "Reader has an out of bounds module number!"); std::string ClassName; const Entity::ClassInfo *pClass = 0; @@ -548,7 +546,7 @@ bool PresetMan::GetAllOfType(std::list &entityList, std::string type, // Specific module else { - RTEAssert(whichModule < m_pDataModules.size(), "Trying to get from an out of bounds DataModule ID!"); + RTEAssert(whichModule < (int)m_pDataModules.size(), "Trying to get from an out of bounds DataModule ID!"); foundAny = m_pDataModules[whichModule]->GetAllOfType(entityList, type); } @@ -597,7 +595,7 @@ bool PresetMan::GetAllOfGroups(std::list &entityList, const std::vecto foundAny = dataModule->GetAllOfGroups(entityList, groups, type) || foundAny; } } else { - RTEAssert(whichModule < m_pDataModules.size(), "Trying to get from an out of bounds DataModule ID in PresetMan::GetAllOfGroups!"); + RTEAssert(whichModule < (int)m_pDataModules.size(), "Trying to get from an out of bounds DataModule ID in PresetMan::GetAllOfGroups!"); foundAny = m_pDataModules[whichModule]->GetAllOfGroups(entityList, groups, type); } return foundAny; @@ -619,7 +617,7 @@ bool PresetMan::GetAllNotOfGroups(std::list &entityList, const std::ve foundAny = dataModule->GetAllNotOfGroups(entityList, groups, type) || foundAny; } } else { - RTEAssert(whichModule < m_pDataModules.size(), "Trying to get from an out of bounds DataModule ID in PresetMan::GetAllNotOfGroups!"); + RTEAssert(whichModule < (int)m_pDataModules.size(), "Trying to get from an out of bounds DataModule ID in PresetMan::GetAllNotOfGroups!"); foundAny = m_pDataModules[whichModule]->GetAllNotOfGroups(entityList, groups, type); } return foundAny; diff --git a/Managers/PrimitiveMan.cpp b/Managers/PrimitiveMan.cpp index 7244b425d..6fe5b3f69 100644 --- a/Managers/PrimitiveMan.cpp +++ b/Managers/PrimitiveMan.cpp @@ -5,6 +5,8 @@ #include "ConsoleMan.h" #include "MOSprite.h" +#include "tracy/Tracy.hpp" + namespace RTE { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// @@ -306,9 +308,12 @@ namespace RTE { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void PrimitiveMan::DrawPrimitives(int player, BITMAP *targetBitmap, const Vector &targetPos) const { + ZoneScoped; + if (m_ScheduledPrimitives.empty()) { return; } + int lastDrawMode = DRAW_MODE_SOLID; DrawBlendMode lastBlendMode = DrawBlendMode::NoBlend; std::array lastBlendAmounts = { BlendAmountLimits::MinBlend, BlendAmountLimits::MinBlend, BlendAmountLimits::MinBlend, BlendAmountLimits::MinBlend }; diff --git a/Managers/SceneMan.cpp b/Managers/SceneMan.cpp index f353c76e9..3f80b9993 100644 --- a/Managers/SceneMan.cpp +++ b/Managers/SceneMan.cpp @@ -36,6 +36,8 @@ // Temp #include "Controller.h" +#include "tracy/Tracy.hpp" + namespace RTE { @@ -2759,7 +2761,9 @@ bool SceneMan::AddSceneObject(SceneObject *sceneObject) { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void SceneMan::Update(int screenId) { - if (!m_pCurrentScene) { + ZoneScoped; + + if (!m_pCurrentScene) { return; } @@ -2805,7 +2809,9 @@ void SceneMan::Update(int screenId) { ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// void SceneMan::Draw(BITMAP *targetBitmap, BITMAP *targetGUIBitmap, const Vector &targetPos, bool skipBackgroundLayers, bool skipTerrain) { - if (!m_pCurrentScene) { + ZoneScoped; + + if (!m_pCurrentScene) { return; } SLTerrain *terrain = m_pCurrentScene->GetTerrain(); diff --git a/Managers/ThreadMan.cpp b/Managers/ThreadMan.cpp index 73984c924..5b20b6988 100644 --- a/Managers/ThreadMan.cpp +++ b/Managers/ThreadMan.cpp @@ -19,11 +19,6 @@ using namespace std; namespace RTE { -#define DELTABUFFERSIZE 30 - -const string ThreadMan::m_ClassName = "ThreadMan"; - - ////////////////////////////////////////////////////////////////////////////////////////// // Method: Clear ////////////////////////////////////////////////////////////////////////////////////////// @@ -32,6 +27,8 @@ const string ThreadMan::m_ClassName = "ThreadMan"; void ThreadMan::Clear() { + m_PriorityThreadPool.reset(); + m_BackgroundThreadPool.reset(std::thread::hardware_concurrency() / 2); } @@ -42,7 +39,6 @@ void ThreadMan::Clear() int ThreadMan::Create() { - return 0; } @@ -54,8 +50,6 @@ int ThreadMan::Create() void ThreadMan::Destroy() { - - Clear(); } diff --git a/Managers/ThreadMan.h b/Managers/ThreadMan.h index 3b5599c65..7cf623a66 100644 --- a/Managers/ThreadMan.h +++ b/Managers/ThreadMan.h @@ -14,14 +14,11 @@ ////////////////////////////////////////////////////////////////////////////////////////// // Inclusions of header files -#include "global_types.h" - - -//#include - #include "Singleton.h" #define g_ThreadMan ThreadMan::Instance() +#include "BS_thread_pool.hpp" + namespace RTE { @@ -53,6 +50,11 @@ class ThreadMan: ThreadMan() { Clear(); Create(); } + /// + /// Makes the TimerMan object ready for use. + /// + void Initialize() { }; + ////////////////////////////////////////////////////////////////////////////////////////// // Destructor: ~ThreadMan @@ -96,23 +98,15 @@ class ThreadMan: void Destroy(); -////////////////////////////////////////////////////////////////////////////////////////// -// Virtual method: GetClassName -////////////////////////////////////////////////////////////////////////////////////////// -// Description: Gets the class name of this Entity. -// Arguments: None. -// Return value: A string with the friendly-formatted type name of this object. + BS::thread_pool& GetPriorityThreadPool() { return m_PriorityThreadPool; } - virtual const std::string & GetClassName() const { return m_ClassName; } + BS::thread_pool& GetBackgroundThreadPool() { return m_BackgroundThreadPool; } ////////////////////////////////////////////////////////////////////////////////////////// // Protected member variable and method declarations protected: - // Member variables - static const std::string m_ClassName; - ////////////////////////////////////////////////////////////////////////////////////////// @@ -134,6 +128,11 @@ class ThreadMan: ThreadMan(const ThreadMan &reference); ThreadMan & operator=(const ThreadMan &rhs); + // For tasks that we want to be performed ASAP, i.e needs to be complete this frame at some point + BS::thread_pool m_PriorityThreadPool; + + // For background tasks that we can just let happen whenever over multiple frames + BS::thread_pool m_BackgroundThreadPool; }; } // namespace RTE diff --git a/Managers/TimerMan.cpp b/Managers/TimerMan.cpp index bfddb36b1..758331abe 100644 --- a/Managers/TimerMan.cpp +++ b/Managers/TimerMan.cpp @@ -29,8 +29,7 @@ namespace RTE { m_SimSpeed = 1.0F; m_TimeScale = 1.0F; m_SimPaused = false; - // This gets dynamically turned on for short periods when sim gets heavy (explosions) and slow-mo effect is appropriate. - m_OneSimUpdatePerFrame = false; + m_OneSimUpdatePerFrame = true; m_SimSpeedLimited = true; } diff --git a/Managers/meson.build b/Managers/meson.build index 9c509a2e6..e59e1544d 100644 --- a/Managers/meson.build +++ b/Managers/meson.build @@ -17,6 +17,7 @@ sources += files( 'PrimitiveMan.cpp', 'SceneMan.cpp', 'SettingsMan.cpp', +'ThreadMan.cpp', 'TimerMan.cpp', 'UInputMan.cpp', 'WindowMan.cpp' @@ -34,4 +35,4 @@ error('Unsupported compiler') endif luaman = static_library('LuaMan', 'LuaMan.cpp', dependencies:deps + luabind_dep, include_directories: [source_inc_dirs, external_inc_dirs], cpp_args: luaman_cpp_args, cpp_pch:pch, override_options: defaults_override) -source_libs += luaman \ No newline at end of file +source_libs += luaman diff --git a/RTEA.vcxproj b/RTEA.vcxproj index ee3f70adf..669864936 100644 --- a/RTEA.vcxproj +++ b/RTEA.vcxproj @@ -160,7 +160,7 @@ /bigobj /Zm300 %(AdditionalOptions) /bigobj %(AdditionalOptions) Disabled - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;DEBUG_BUILD;DEBUGMODE;TARGET_MACHINE_X86;%(PreprocessorDefinitions) false EnableFastChecks @@ -212,7 +212,7 @@ /bigobj /Zm300 %(AdditionalOptions) /bigobj %(AdditionalOptions) MaxSpeed - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;TRACY_ENABLE;TRACY_ON_DEMAND;DEBUG_BUILD;DEBUGMODE;%(PreprocessorDefinitions) false EnableFastChecks @@ -264,7 +264,7 @@ /bigobj /Zm300 %(AdditionalOptions) /bigobj %(AdditionalOptions) Disabled - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;MIN_DEBUG_BUILD;DEBUGMODE;TARGET_MACHINE_X86;%(PreprocessorDefinitions) false EnableFastChecks @@ -316,7 +316,7 @@ /bigobj /Zm300 %(AdditionalOptions) /bigobj %(AdditionalOptions) Disabled - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;MIN_DEBUG_BUILD;DEBUGMODE;%(PreprocessorDefinitions) false EnableFastChecks @@ -371,7 +371,7 @@ true Speed false - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;DEBUG_RELEASE_BUILD;NDEBUG;TRACY_ENABLE;TRACY_ON_DEMAND;TARGET_MACHINE_X86;%(PreprocessorDefinitions) Sync MultiThreadedDLL @@ -432,7 +432,7 @@ true Speed false - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;TRACY_ENABLE;TRACY_ON_DEMAND;PROFILING_BUILD;NDEBUG;TARGET_MACHINE_X86;%(PreprocessorDefinitions) Sync MultiThreadedDLL @@ -493,7 +493,7 @@ true Speed false - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;DEBUG_RELEASE_BUILD;NDEBUG;TRACY_ENABLE;TRACY_ON_DEMAND;%(PreprocessorDefinitions) Sync MultiThreadedDLL @@ -554,7 +554,7 @@ true Speed true - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;TRACY_ENABLE;TRACY_ON_DEMAND;PROFILING_BUILD;NDEBUG;%(PreprocessorDefinitions) Sync MultiThreadedDLL @@ -615,7 +615,7 @@ true Speed true - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;RELEASE_BUILD;NDEBUG;LUABIND_NO_ERROR_CHECKING;TRACY_ENABLE;TRACY_ON_DEMAND;TRACY_ONLY_LOCALHOST;TRACY_NO_BROADCAST;TARGET_MACHINE_X86;%(PreprocessorDefinitions) Sync MultiThreadedDLL @@ -674,7 +674,7 @@ true Speed true - $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public + $(ProjectDir);$(ProjectDir)System;$(ProjectDir)Lua;$(ProjectDir)Entities;$(ProjectDir)Activities;$(ProjectDir)Managers;$(ProjectDir)Menus;$(ProjectDir)GUI;$(ProjectDir)GUI\Wrappers;$(ProjectDir)external\include\win;$(ProjectDir)external\include\common\;$(ProjectDir)external\include\common\boost_1_75;$(ProjectDir)external\include\win\LZ4;$(ProjectDir)external\include\common\fmod;$(ProjectDir)external\include\common\thread-pool-3.5.0\include;$(ProjectDir)external\include\common\hopscotch-map-2.3.1\include;$(ProjectDir)external\sources\libpng-1.6.40\include;$(ProjectDir)external\sources\zlib-ng-2.1.3\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\include;$(ProjectDir)external\sources\allegro 4.4.3.1-custom\addons\loadpng;$(ProjectDir)external\sources\SDL2-2.26.3\include;$(ProjectDir)external\sources\LuaJIT-2.1\src;$(ProjectDir)external\sources\luabind-0.7.1\;$(ProjectDir)external\sources\luabind-0.7.1\luabind;$(ProjectDir)external\sources\RakNet\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\include;$(ProjectDir)external\sources\minizip-ng-4.0.0\src;$(ProjectDir)external\sources\tracy\public _ITERATOR_DEBUG_LEVEL=0;WIN32_LEAN_AND_MEAN;ALLEGRO_STATICLINK;ALLEGRO_NO_STD_HEADERS;ALLEGRO_NO_MAGIC_MAIN;_CRT_SECURE_NO_DEPRECATE;ZLIB_WINAPI;ZLIB_COMPAT;_WINDOWS;WIN32;RELEASE_BUILD;NDEBUG;LUABIND_NO_ERROR_CHECKING;TRACY_ENABLE;TRACY_ON_DEMAND;TRACY_ONLY_LOCALHOST;TRACY_NO_BROADCAST;%(PreprocessorDefinitions) Sync MultiThreadedDLL @@ -755,6 +755,7 @@ + @@ -998,6 +999,7 @@ + diff --git a/RTEA.vcxproj.filters b/RTEA.vcxproj.filters index 99d81cb88..ba0737859 100644 --- a/RTEA.vcxproj.filters +++ b/RTEA.vcxproj.filters @@ -621,6 +621,9 @@ System + + Managers + @@ -1194,6 +1197,9 @@ System + + Managers + diff --git a/System/Constants.h b/System/Constants.h index 2bbf2732f..6d528fbf9 100644 --- a/System/Constants.h +++ b/System/Constants.h @@ -332,15 +332,6 @@ namespace RTE { {Directions::Left, c_PI}, {Directions::Right, 0.0F} }; - - /// - /// Enum of thread script types to run. - /// - enum class ThreadScriptsToRun { - SingleThreaded, - MultiThreaded, - Both - }; #pragma endregion #pragma region Un-Definitions diff --git a/System/Controller.cpp b/System/Controller.cpp index f51cd53dc..fad7b95f3 100644 --- a/System/Controller.cpp +++ b/System/Controller.cpp @@ -214,7 +214,7 @@ namespace RTE { bool Controller::ShouldUpdateAIThisFrame() const { - if (IsDisabled()) { + if (IsDisabled() || m_InputMode != InputMode::CIM_AI) { return false; } @@ -228,19 +228,6 @@ namespace RTE { return true; } -///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// - - void Controller::UpdateAI(ThreadScriptsToRun scriptsToRun) { - if (m_InputMode != InputMode::CIM_AI || !ShouldUpdateAIThisFrame()) { - return; - } - - // Run the scripted AI for the controlled Actor - if (m_ControlledActor && m_ControlledActor->ObjectScriptsInitialized()) { - m_ControlledActor->UpdateAIScripted(scriptsToRun); - } - } - ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// Controller & Controller::operator=(const Controller &rhs) { diff --git a/System/Controller.h b/System/Controller.h index 66baea305..18c510b9e 100644 --- a/System/Controller.h +++ b/System/Controller.h @@ -315,15 +315,15 @@ namespace RTE { /// /// A pointer to a an Actor which is being controlled by this. Ownership is NOT transferred! void SetControlledActor(Actor *controlledActor = nullptr) { m_ControlledActor = controlledActor; } -#pragma endregion -#pragma region Virtual Override Methods /// - /// Updates this Controller. Supposed to be done every frame. + /// Returns whether the AI should be updated this frame. /// - /// Whether to update single-threaded, multi-threaded, or both types of AI scripts. - void UpdateAI(ThreadScriptsToRun scriptsToRun); + /// Whether the AI should be updated this frame. + bool ShouldUpdateAIThisFrame() const; +#pragma endregion +#pragma region Virtual Override Methods /// /// Updates this Controller. Supposed to be done every frame. /// @@ -417,8 +417,6 @@ namespace RTE { /// Requests and applies input from the player. /// void GetInputFromPlayer(); - - bool ShouldUpdateAIThisFrame() const; #pragma endregion /// diff --git a/System/GenericSavedData.cpp b/System/GenericSavedData.cpp index 0206de3eb..e9b22abff 100644 --- a/System/GenericSavedData.cpp +++ b/System/GenericSavedData.cpp @@ -83,7 +83,7 @@ namespace RTE { int GenericSavedData::GenericSavedEncodedStrings::ReadProperty(const std::string_view &propName, Reader &reader) { std::string value = reader.ReadPropValue(); - m_Data[std::string(propName)] = base64_decode(value);; // until we get P0919R2. + m_Data[std::string(propName)] = base64_decode(value); // until we get P0919R2. return 0; } diff --git a/System/PathFinder.cpp b/System/PathFinder.cpp index d75c0543b..f21ed39ae 100644 --- a/System/PathFinder.cpp +++ b/System/PathFinder.cpp @@ -3,6 +3,7 @@ #include "Material.h" #include "Scene.h" #include "SceneMan.h" +#include "ThreadMan.h" namespace RTE { @@ -207,7 +208,7 @@ namespace RTE { const_cast(pathRequest->startPos) = start; const_cast(pathRequest->targetPos) = end; - std::thread pathThread([this, start, end, digStrength, callback](std::shared_ptr volRequest) { + g_ThreadMan.GetBackgroundThreadPool().push_task([this, start, end, digStrength, callback](std::shared_ptr volRequest) { // Cast away the volatile-ness - only matters outside (and complicates the API otherwise) PathRequest &request = const_cast(*volRequest); @@ -225,7 +226,6 @@ namespace RTE { request.complete = true; }, pathRequest); - pathThread.detach(); return pathRequest; } diff --git a/System/SpatialPartitionGrid.cpp b/System/SpatialPartitionGrid.cpp index 1f82d6ad5..51abbfb40 100644 --- a/System/SpatialPartitionGrid.cpp +++ b/System/SpatialPartitionGrid.cpp @@ -87,7 +87,6 @@ namespace RTE { for (int x = topLeftCellX; x <= bottomRightCellX; x++) { for (int y = topLeftCellY; y <= bottomRightCellY; y++) { int cellId = GetCellIdForCellCoords(x, y); - m_UsedCellIds.insert(cellId); m_Cells[team + 1][cellId].push_back(mo.GetID()); if (mo.GetsHitByMOs()) { @@ -96,6 +95,12 @@ namespace RTE { } } } + + for (int x = topLeftCellX; x <= bottomRightCellX; x++) { + for (int y = topLeftCellY; y <= bottomRightCellY; y++) { + m_UsedCellIds.insert(GetCellIdForCellCoords(x, y)); + } + } } ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// diff --git a/System/SpatialPartitionGrid.h b/System/SpatialPartitionGrid.h index 5f37282c4..9cadcf772 100644 --- a/System/SpatialPartitionGrid.h +++ b/System/SpatialPartitionGrid.h @@ -6,6 +6,8 @@ #include "Constants.h" +#include "tsl/hopscotch_set.h" + namespace RTE { class Box; @@ -98,7 +100,7 @@ namespace RTE { std::array>, Activity::MaxTeamCount + 1> m_Cells; //!< Array of cells for each team. The outside-vector is the vector of cells for the team, and each inside-vector entry contains all MOIDs in the cell's space that can collide with that team. std::array>, Activity::MaxTeamCount + 1> m_PhysicsCells; //!< Same as m_Cells, but includes only objects that are GetsHitByMOs. - std::unordered_set m_UsedCellIds; //!< Set of used cell Ids, maintained to avoid wasting time looping through and clearing unused cells. + tsl::hopscotch_set m_UsedCellIds; //!< Set of used cell Ids, maintained to avoid wasting time looping through and clearing unused cells. /// /// Gets the Id of the cell at the given SpatialPartitionGrid coordinates, automatically accounting for wrapping. diff --git a/System/StandardIncludes.h b/System/StandardIncludes.h index a83770356..e31b8f8a2 100644 --- a/System/StandardIncludes.h +++ b/System/StandardIncludes.h @@ -85,6 +85,7 @@ #include #include #include +#include namespace std { diff --git a/external/include/common/hopscotch-map-2.3.1/.codecov.yml b/external/include/common/hopscotch-map-2.3.1/.codecov.yml new file mode 100644 index 000000000..92a0a4a07 --- /dev/null +++ b/external/include/common/hopscotch-map-2.3.1/.codecov.yml @@ -0,0 +1,5 @@ +comment: off +coverage: + status: + project: off + patch: off diff --git a/external/include/common/hopscotch-map-2.3.1/.github/workflows/ci.yml b/external/include/common/hopscotch-map-2.3.1/.github/workflows/ci.yml new file mode 100644 index 000000000..593b05d68 --- /dev/null +++ b/external/include/common/hopscotch-map-2.3.1/.github/workflows/ci.yml @@ -0,0 +1,136 @@ +name: CI + +on: [push, pull_request, release] + +jobs: + build: + strategy: + fail-fast: false + matrix: + config: + - { + name: linux-x64-gcc, + os: ubuntu-latest, + cxx: g++, + cmake-build-type: Release + } + - { + name: linux-x64-gcc-no-exceptions, + os: ubuntu-latest, + cxx: g++, + cxx-flags: -fno-exceptions, + cmake-build-type: Release + } + - { + name: linux-x64-clang, + os: ubuntu-latest, + cxx: clang++, + cmake-build-type: Release + } + - { + name: macos-x64-gcc, + os: macos-latest, + cxx: g++, + cmake-build-type: Release + } + - { + name: macos-x64-clang, + os: macos-latest, + cxx: clang++, + cmake-build-type: Release + } + - { + name: linux-x64-clang-sanitize, + os: ubuntu-latest, + cxx: clang++, + cxx-flags: "-fsanitize=address,undefined", + cmake-build-type: Release + } + - { + name: linux-x64-gcc-coverage, + os: ubuntu-latest, + cxx: g++, + cxx-flags: --coverage, + gcov-tool: gcov, + cmake-build-type: Debug + } + - { + name: windows-x64-vs-2019, + os: windows-2019, + cmake-build-type: Release, + cmake-generator: Visual Studio 16 2019, + cmake-platform: x64, + vcpkg-triplet: x64-windows-static-md + } + - { + name: windows-x86-vs-2019, + os: windows-2019, + cmake-build-type: Release, + cmake-generator: Visual Studio 16 2019, + cmake-platform: Win32, + vcpkg-triplet: x86-windows-static-md + } + - { + name: windows-x64-vs-2022, + os: windows-2022, + cmake-build-type: Release, + cmake-generator: Visual Studio 17 2022, + cmake-platform: x64, + vcpkg-triplet: x64-windows-static-md + } + - { + name: windows-x86-vs-2022, + os: windows-2022, + cmake-build-type: Release, + cmake-generator: Visual Studio 17 2022, + cmake-platform: Win32, + vcpkg-triplet: x86-windows-static-md + } + name: ${{matrix.config.name}} + runs-on: ${{matrix.config.os}} + steps: + - uses: actions/checkout@v2 + + # Windows + - name: Install boost (Windows) + run: vcpkg install boost-test:${{matrix.config.vcpkg-triplet}} + if: runner.os == 'Windows' + + - name: Configure CMake (Windows) + run: cmake -G "${{matrix.config.cmake-generator}}" -A ${{matrix.config.cmake-platform}} -DCMAKE_BUILD_TYPE=${{matrix.config.cmake-build-type}} -DCMAKE_TOOLCHAIN_FILE="$env:VCPKG_INSTALLATION_ROOT/scripts/buildsystems/vcpkg.cmake" -DVCPKG_TARGET_TRIPLET=${{matrix.config.vcpkg-triplet}} -S ${{github.workspace}}/tests -B ${{github.workspace}}/build + if: runner.os == 'Windows' + + - name: Build (Windows) + run: cmake --build ${{github.workspace}}/build --config ${{matrix.config.cmake-build-type}} --verbose + if: runner.os == 'Windows' + + - name: Test (Windows) + run: ${{github.workspace}}/build/${{matrix.config.cmake-build-type}}/tsl_hopscotch_map_tests.exe + if: runner.os == 'Windows' + + # Linux or macOS + - name: Install boost (Linux or macOS) + run: vcpkg install boost-test + if: runner.os == 'Linux' || runner.os == 'macOS' + + - name: Configure CMake (Linux or macOS) + run: cmake -DCMAKE_BUILD_TYPE=${{matrix.config.cmake-build-type}} -DCMAKE_TOOLCHAIN_FILE="$VCPKG_INSTALLATION_ROOT/scripts/buildsystems/vcpkg.cmake" -S ${{github.workspace}}/tests -B ${{github.workspace}}/build + env: + CXX: ${{matrix.config.cxx}} + CXXFLAGS: ${{matrix.config.cxx-flags}} + if: runner.os == 'Linux' || runner.os == 'macOS' + + - name: Build (Linux or macOS) + run: cmake --build ${{github.workspace}}/build --verbose + if: runner.os == 'Linux' || runner.os == 'macOS' + + - name: Test (Linux or macOS) + run: ${{github.workspace}}/build/tsl_hopscotch_map_tests + if: runner.os == 'Linux' || runner.os == 'macOS' + + - name: Coverage + run: | + sudo apt-get install -y lcov + lcov -c -b ${{github.workspace}}/include -d ${{github.workspace}}/build -o ${{github.workspace}}/coverage.info --no-external --gcov-tool ${{matrix.config.gcov-tool}} + bash <(curl -s https://codecov.io/bash) -f ${{github.workspace}}/coverage.info + if: ${{matrix.config.name == 'linux-x64-gcc-coverage'}} diff --git a/external/include/common/hopscotch-map-2.3.1/CMakeLists.txt b/external/include/common/hopscotch-map-2.3.1/CMakeLists.txt new file mode 100644 index 000000000..6fea27993 --- /dev/null +++ b/external/include/common/hopscotch-map-2.3.1/CMakeLists.txt @@ -0,0 +1,80 @@ +cmake_minimum_required(VERSION 3.1) +include(GNUInstallDirs) + + +project(tsl-hopscotch-map VERSION 2.3.1) + +add_library(hopscotch_map INTERFACE) +# Use tsl::hopscotch_map as target, more consistent with other libraries conventions (Boost, Qt, ...) +add_library(tsl::hopscotch_map ALIAS hopscotch_map) + +target_include_directories(hopscotch_map INTERFACE + "$" + "$") + +list(APPEND headers "${CMAKE_CURRENT_SOURCE_DIR}/include/tsl/bhopscotch_map.h" + "${CMAKE_CURRENT_SOURCE_DIR}/include/tsl/bhopscotch_set.h" + "${CMAKE_CURRENT_SOURCE_DIR}/include/tsl/hopscotch_growth_policy.h" + "${CMAKE_CURRENT_SOURCE_DIR}/include/tsl/hopscotch_hash.h" + "${CMAKE_CURRENT_SOURCE_DIR}/include/tsl/hopscotch_map.h" + "${CMAKE_CURRENT_SOURCE_DIR}/include/tsl/hopscotch_set.h") +target_sources(hopscotch_map INTERFACE "$") + +if(MSVC) + target_sources(hopscotch_map INTERFACE + "$" + "$") +endif() + + + + +# Installation (only compatible with CMake version >= 3.3) +if(${CMAKE_VERSION} VERSION_GREATER "3.2") + include(CMakePackageConfigHelpers) + + ## Install include directory and potential natvis file + install(DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}/include/tsl" + DESTINATION "${CMAKE_INSTALL_INCLUDEDIR}") + + if(MSVC) + install(FILES "${CMAKE_CURRENT_SOURCE_DIR}/tsl-hopscotch-map.natvis" + DESTINATION "${CMAKE_INSTALL_DATAROOTDIR}") + endif() + + + + ## Create and install tsl-hopscotch-mapConfig.cmake + configure_package_config_file("${CMAKE_CURRENT_SOURCE_DIR}/cmake/tsl-hopscotch-mapConfig.cmake.in" + "${CMAKE_CURRENT_BINARY_DIR}/tsl-hopscotch-mapConfig.cmake" + INSTALL_DESTINATION "${CMAKE_INSTALL_DATAROOTDIR}/cmake/tsl-hopscotch-map") + + install(FILES "${CMAKE_CURRENT_BINARY_DIR}/tsl-hopscotch-mapConfig.cmake" + DESTINATION "${CMAKE_INSTALL_DATAROOTDIR}/cmake/tsl-hopscotch-map") + + + + ## Create and install tsl-hopscotch-mapTargets.cmake + install(TARGETS hopscotch_map + EXPORT tsl-hopscotch-mapTargets) + + install(EXPORT tsl-hopscotch-mapTargets + NAMESPACE tsl:: + DESTINATION "${CMAKE_INSTALL_DATAROOTDIR}/cmake/tsl-hopscotch-map") + + + + ## Create and install tsl-hopscotch-mapConfigVersion.cmake + # tsl-hopscotch-map is header-only and does not depend on the architecture. + # Remove CMAKE_SIZEOF_VOID_P from tsl-hopscotch-mapConfigVersion.cmake so that a + # tsl-hopscotch-mapConfig.cmake generated for a 64 bit target can be used for 32 bit + # targets and vice versa. + set(CMAKE_SIZEOF_VOID_P_BACKUP ${CMAKE_SIZEOF_VOID_P}) + unset(CMAKE_SIZEOF_VOID_P) + write_basic_package_version_file("${CMAKE_CURRENT_BINARY_DIR}/tsl-hopscotch-mapConfigVersion.cmake" + COMPATIBILITY SameMajorVersion) + set(CMAKE_SIZEOF_VOID_P ${CMAKE_SIZEOF_VOID_P_BACKUP}) + + install(FILES "${CMAKE_CURRENT_BINARY_DIR}/tsl-hopscotch-mapConfigVersion.cmake" + DESTINATION "${CMAKE_INSTALL_DATAROOTDIR}/cmake/tsl-hopscotch-map") +endif() diff --git a/external/include/common/hopscotch-map-2.3.1/LICENSE b/external/include/common/hopscotch-map-2.3.1/LICENSE new file mode 100644 index 000000000..b360b6daa --- /dev/null +++ b/external/include/common/hopscotch-map-2.3.1/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2016 Thibaut Goetghebuer-Planchon + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/external/include/common/hopscotch-map-2.3.1/README.md b/external/include/common/hopscotch-map-2.3.1/README.md new file mode 100644 index 000000000..026822141 --- /dev/null +++ b/external/include/common/hopscotch-map-2.3.1/README.md @@ -0,0 +1,342 @@ +[![CI](https://github.com/Tessil/hopscotch-map/actions/workflows/ci.yml/badge.svg?branch=master)](https://github.com/Tessil/hopscotch-map/actions/workflows/ci.yml) + +## A C++ implementation of a fast hash map and hash set using hopscotch hashing + +The hopscotch-map library is a C++ implementation of a fast hash map and hash set using open-addressing and hopscotch hashing to resolve collisions. It is a cache-friendly data structure offering better performances than `std::unordered_map` in most cases and is closely similar to `google::dense_hash_map` while using less memory and providing more functionalities. + +The library provides the following main classes: `tsl::hopscotch_map`, `tsl::hopscotch_set`, `tsl::hopscotch_pg_map` and `tsl::hopscotch_pg_set`. The first two are faster and use a power of two growth policy, the last two use a prime growth policy instead and are able to cope better with a poor hash function. Use the prime version if there is a chance of repeating patterns in the lower bits of your hash (e.g. you are storing pointers with an identity hash function). See [GrowthPolicy](#growth-policy) for details. + + +In addition to these classes the library also provides `tsl::bhopscotch_map`, `tsl::bhopscotch_set`, `tsl::bhopscotch_pg_map` and `tsl::bhopscotch_pg_set`. These classes have an additional requirement for the key, it must be `LessThanComparable`, but they provide a better asymptotic upper bound, see [details](#deny-of-service-dos-attack) in example. Nonetheless if you don't have specific requirements (risk of hash DoS attacks), `tsl::hopscotch_map` and `tsl::hopscotch_set` should be sufficient in most cases and should be your default pick as they perform better in general. + + +An overview of hopscotch hashing and some implementation details can be found [here](https://tessil.github.io/2016/08/29/hopscotch-hashing.html). + +A **benchmark** of `tsl::hopscotch_map` against other hash maps may be found [here](https://tessil.github.io/2016/08/29/benchmark-hopscotch-map.html). This page also gives some advices on which hash table structure you should try for your use case (useful if you are a bit lost with the multiple hash tables implementations in the `tsl` namespace). + +### Key features +- Header-only library, just add the [include](include/) directory to your include path and you are ready to go. If you use CMake, you can also use the `tsl::hopscotch_map` exported target from the [CMakeLists.txt](CMakeLists.txt). +- Fast hash table, see [benchmark](https://tessil.github.io/2016/08/29/benchmark-hopscotch-map.html) for some numbers. +- Support for move-only and non-default constructible key/value. +- Support for heterogeneous lookups allowing the usage of `find` with a type different than `Key` (e.g. if you have a map that uses `std::unique_ptr` as key, you can use a `foo*` or a `std::uintptr_t` as key parameter to `find` without constructing a `std::unique_ptr`, see [example](#heterogeneous-lookups)). +- No need to reserve any sentinel value from the keys. +- Possibility to store the hash value on insert for faster rehash and lookup if the hash or the key equal functions are expensive to compute (see the [StoreHash](https://tessil.github.io/hopscotch-map/classtsl_1_1hopscotch__map.html#details) template parameter). +- If the hash is known before a lookup, it is possible to pass it as parameter to speed-up the lookup (see `precalculated_hash` parameter in [API](https://tessil.github.io/hopscotch-map/classtsl_1_1hopscotch__map.html#a74d83c67c50bc8385bb11f78142eaa86)). +- The `tsl::bhopscotch_map` and `tsl::bhopscotch_set` provide a worst-case of O(log n) on lookups and deletions making these classes resistant to hash table Deny of Service (DoS) attacks (see [details](#deny-of-service-dos-attack) in example). +- The library can be used with exceptions disabled (through `-fno-exceptions` option on Clang and GCC, without an `/EH` option on MSVC or simply by defining `TSL_NO_EXCEPTIONS`). `std::terminate` is used in replacement of the `throw` instruction when exceptions are disabled. +- API closely similar to `std::unordered_map` and `std::unordered_set`. + +### Differences compared to `std::unordered_map` +`tsl::hopscotch_map` tries to have an interface similar to `std::unordered_map`, but some differences exist. +- Iterator invalidation on insert doesn't behave in the same way. In general any operation modifying the hash table, except `erase`, invalidate all the iterators (see [API](https://tessil.github.io/hopscotch-map/classtsl_1_1hopscotch__map.html#details) for details). +- References and pointers to keys or values in the map are invalidated in the same way as iterators to these keys-values on insert. +- For iterators, `operator*()` and `operator->()` return a reference and a pointer to `const std::pair` instead of `std::pair`, making the value `T` not modifiable. To modify the value you have to call the `value()` method of the iterator to get a mutable reference. Example: +```c++ +tsl::hopscotch_map map = {{1, 1}, {2, 1}, {3, 1}}; +for(auto it = map.begin(); it != map.end(); ++it) { + //it->second = 2; // Illegal + it.value() = 2; // Ok +} +``` +- Move-only types must have a nothrow move constructor (with open addressing, it is not possible to keep the strong exception guarantee on rehash if the move constructor may throw). +- No support for some buckets related methods (like `bucket_size`, `bucket`, ...). + +These differences also apply between `std::unordered_set` and `tsl::hopscotch_set`. + +Thread-safety and exceptions guarantees are the same as `std::unordered_map/set` (i.e. possible to have multiple readers with no writer). + +### Growth policy + +The library supports multiple growth policies through the `GrowthPolicy` template parameter. Three policies are provided by the library but you can easily implement your own if needed. + +* **[tsl::hh::power_of_two_growth_policy.](https://tessil.github.io/hopscotch-map/classtsl_1_1hh_1_1power__of__two__growth__policy.html)** Default policy used by `tsl::(b)hopscotch_map/set`. This policy keeps the size of the bucket array of the hash table to a power of two. This constraint allows the policy to avoid the usage of the slow modulo operation to map a hash to a bucket, instead of hash % 2n, it uses hash & (2n - 1) (see [fast modulo](https://en.wikipedia.org/wiki/Modulo_operation#Performance_issues)). Fast but this may cause a lot of collisions with a poor hash function as the modulo with a power of two only masks the most significant bits in the end. +* **[tsl::hh::prime_growth_policy.](https://tessil.github.io/hopscotch-map/classtsl_1_1hh_1_1prime__growth__policy.html)** Default policy used by `tsl::(b)hopscotch_pg_map/set`. The policy keeps the size of the bucket array of the hash table to a prime number. When mapping a hash to a bucket, using a prime number as modulo will result in a better distribution of the hashes across the buckets even with a poor hash function. To allow the compiler to optimize the modulo operation, the policy use a lookup table with constant primes modulos (see [API](https://tessil.github.io/hopscotch-map/classtsl_1_1hh_1_1prime__growth__policy.html#details) for details). Slower than `tsl::hh::power_of_two_growth_policy` but more secure. +* **[tsl::hh::mod_growth_policy.](https://tessil.github.io/hopscotch-map/classtsl_1_1hh_1_1mod__growth__policy.html)** The policy grows the map by a customizable growth factor passed in parameter. It then just use the modulo operator to map a hash to a bucket. Slower but more flexible. + +If you encounter poor performances check the `overflow_size()`, if it is not zero you may have a lot of hash collisions. Either change the hash function for something more uniform or try another growth policy (mainly `tsl::hh::prime_growth_policy`). Unfortunately it is sometimes difficult to guard yourself against collisions (e.g. DoS attack on the hash map). If needed, check also `tsl::bhopscotch_map/set` which offer a worst-case scenario of O(log n) on lookups instead of O(n), see [details](#deny-of-service-dos-attack) in example. + +To implement your own policy, you have to implement the following interface. + +```c++ +struct custom_policy { + // Called on hash table construction and rehash, min_bucket_count_in_out is the minimum buckets + // that the hash table needs. The policy can change it to a higher number of buckets if needed + // and the hash table will use this value as bucket count. If 0 bucket is asked, then the value + // must stay at 0. + explicit custom_policy(std::size_t& min_bucket_count_in_out); + + // Return the bucket [0, bucket_count()) to which the hash belongs. + // If bucket_count() is 0, it must always return 0. + std::size_t bucket_for_hash(std::size_t hash) const noexcept; + + // Return the number of buckets that should be used on next growth + std::size_t next_bucket_count() const; + + // Maximum number of buckets supported by the policy + std::size_t max_bucket_count() const; + + // Reset the growth policy as if the policy was created with a bucket count of 0. + // After a clear, the policy must always return 0 when bucket_for_hash() is called. + void clear() noexcept; +} +``` +### Installation +To use hopscotch-map, just add the [include](include/) directory to your include path. It is a **header-only** library. + +If you use CMake, you can also use the `tsl::hopscotch_map` exported target from the [CMakeLists.txt](CMakeLists.txt) with `target_link_libraries`. +```cmake +# Example where the hopscotch-map project is stored in a third-party directory +add_subdirectory(third-party/hopscotch-map) +target_link_libraries(your_target PRIVATE tsl::hopscotch_map) +``` + +If the project has been installed through `make install`, you can also use `find_package(tsl-hopscotch-map REQUIRED)` instead of `add_subdirectory`. + +The code should work with any C++11 standard-compliant compiler and has been tested with GCC 4.8.4, Clang 3.5.0 and Visual Studio 2015. + +To run the tests you will need the Boost Test library and CMake. + +```bash +git clone https://github.com/Tessil/hopscotch-map.git +cd hopscotch-map/tests +mkdir build +cd build +cmake .. +cmake --build . +./tsl_hopscotch_map_tests +``` + + +### Usage +The API can be found [here](https://tessil.github.io/hopscotch-map/). + +All methods are not documented yet, but they replicate the behaviour of the ones in `std::unordered_map` and `std::unordered_set`, except if specified otherwise. + +### Example + +```c++ +#include +#include +#include +#include +#include + +int main() { + tsl::hopscotch_map map = {{"a", 1}, {"b", 2}}; + map["c"] = 3; + map["d"] = 4; + + map.insert({"e", 5}); + map.erase("b"); + + for(auto it = map.begin(); it != map.end(); ++it) { + //it->second += 2; // Not valid. + it.value() += 2; + } + + // {d, 6} {a, 3} {e, 7} {c, 5} + for(const auto& key_value : map) { + std::cout << "{" << key_value.first << ", " << key_value.second << "}" << std::endl; + } + + + if(map.find("a") != map.end()) { + std::cout << "Found \"a\"." << std::endl; + } + + const std::size_t precalculated_hash = std::hash()("a"); + // If we already know the hash beforehand, we can pass it in parameter to speed-up lookups. + if(map.find("a", precalculated_hash) != map.end()) { + std::cout << "Found \"a\" with hash " << precalculated_hash << "." << std::endl; + } + + + /* + * Calculating the hash and comparing two std::string may be slow. + * We can store the hash of each std::string in the hash map to make + * the inserts and lookups faster by setting StoreHash to true. + */ + tsl::hopscotch_map, + std::equal_to, + std::allocator>, + 30, true> map2; + + map2["a"] = 1; + map2["b"] = 2; + + // {a, 1} {b, 2} + for(const auto& key_value : map2) { + std::cout << "{" << key_value.first << ", " << key_value.second << "}" << std::endl; + } + + + + + tsl::hopscotch_set set; + set.insert({1, 9, 0}); + set.insert({2, -1, 9}); + + // {0} {1} {2} {9} {-1} + for(const auto& key : set) { + std::cout << "{" << key << "}" << std::endl; + } +} +``` + +#### Heterogeneous lookups + +Heterogeneous overloads allow the usage of other types than `Key` for lookup and erase operations as long as the used types are hashable and comparable to `Key`. + +To activate the heterogeneous overloads in `tsl::hopscotch_map/set`, the qualified-id `KeyEqual::is_transparent` must be valid. It works the same way as for [`std::map::find`](http://en.cppreference.com/w/cpp/container/map/find). You can either use [`std::equal_to<>`](http://en.cppreference.com/w/cpp/utility/functional/equal_to_void) or define your own function object. + +Both `KeyEqual` and `Hash` will need to be able to deal with the different types. + +```c++ +#include +#include +#include +#include + + +struct employee { + employee(int id, std::string name) : m_id(id), m_name(std::move(name)) { + } + + // Either we include the comparators in the class and we use `std::equal_to<>`... + friend bool operator==(const employee& empl, int empl_id) { + return empl.m_id == empl_id; + } + + friend bool operator==(int empl_id, const employee& empl) { + return empl_id == empl.m_id; + } + + friend bool operator==(const employee& empl1, const employee& empl2) { + return empl1.m_id == empl2.m_id; + } + + + int m_id; + std::string m_name; +}; + +// ... or we implement a separate class to compare employees. +struct equal_employee { + using is_transparent = void; + + bool operator()(const employee& empl, int empl_id) const { + return empl.m_id == empl_id; + } + + bool operator()(int empl_id, const employee& empl) const { + return empl_id == empl.m_id; + } + + bool operator()(const employee& empl1, const employee& empl2) const { + return empl1.m_id == empl2.m_id; + } +}; + +struct hash_employee { + std::size_t operator()(const employee& empl) const { + return std::hash()(empl.m_id); + } + + std::size_t operator()(int id) const { + return std::hash()(id); + } +}; + + +int main() { + // Use std::equal_to<> which will automatically deduce and forward the parameters + tsl::hopscotch_map> map; + map.insert({employee(1, "John Doe"), 2001}); + map.insert({employee(2, "Jane Doe"), 2002}); + map.insert({employee(3, "John Smith"), 2003}); + + // John Smith 2003 + auto it = map.find(3); + if(it != map.end()) { + std::cout << it->first.m_name << " " << it->second << std::endl; + } + + map.erase(1); + + + + // Use a custom KeyEqual which has an is_transparent member type + tsl::hopscotch_map map2; + map2.insert({employee(4, "Johnny Doe"), 2004}); + + // 2004 + std::cout << map2.at(4) << std::endl; +} +``` + +#### Deny of Service (DoS) attack +In addition to `tsl::hopscotch_map` and `tsl::hopscotch_set`, the library provides two more "secure" options: `tsl::bhopscotch_map` and `tsl::bhopscotch_set` (all with their `pg` counterparts). + +These two additions have a worst-case asymptotic complexity of O(log n) for lookups and deletions and an amortized worst case of O(log n) for insertions (amortized due to the possibility of rehash which would be in O(n)). Even if the hash function maps all the elements to the same bucket, the O(log n) would still hold. + +This provides a security against hash table Deny of Service (DoS) attacks. + +To achieve this, the *secure* versions use a binary search tree for the overflown elements (see [implementation details](https://tessil.github.io/2016/08/29/hopscotch-hashing.html)) and thus need the elements to be `LessThanComparable`. An additional `Compare` template parameter is needed. + +```c++ +#include +#include +#include +#include +#include + +/* + * Poor hash function which always returns 1 to simulate + * a Deny of Service attack. + */ +struct dos_attack_simulation_hash { + std::size_t operator()(int id) const { + return 1; + } +}; + +int main() { + /* + * Slow due to the hash function, insertions are done in O(n). + */ + tsl::hopscotch_map map; + + auto start = std::chrono::high_resolution_clock::now(); + for(int i=0; i < 10000; i++) { + map.insert({i, 0}); + } + auto end = std::chrono::high_resolution_clock::now(); + + // 110 ms + auto duration = std::chrono::duration_cast(end-start); + std::cout << duration.count() << " ms" << std::endl; + + + + + /* + * Faster. Even with the poor hash function, insertions end-up to + * be O(log n) in average (and O(n) when a rehash occurs). + */ + tsl::bhopscotch_map map_secure; + + start = std::chrono::high_resolution_clock::now(); + for(int i=0; i < 10000; i++) { + map_secure.insert({i, 0}); + } + end = std::chrono::high_resolution_clock::now(); + + // 2 ms + duration = std::chrono::duration_cast(end-start); + std::cout << duration.count() << " ms" << std::endl; +} +``` + +### License + +The code is licensed under the MIT license, see the [LICENSE file](LICENSE) for details. diff --git a/external/include/common/hopscotch-map-2.3.1/cmake/tsl-hopscotch-mapConfig.cmake.in b/external/include/common/hopscotch-map-2.3.1/cmake/tsl-hopscotch-mapConfig.cmake.in new file mode 100644 index 000000000..16226759b --- /dev/null +++ b/external/include/common/hopscotch-map-2.3.1/cmake/tsl-hopscotch-mapConfig.cmake.in @@ -0,0 +1,9 @@ +# This module sets the following variables: +# * tsl-hopscotch-map_FOUND - true if tsl-hopscotch-map found on the system +# * tsl-hopscotch-map_INCLUDE_DIRS - the directory containing tsl-hopscotch-map headers +@PACKAGE_INIT@ + +if(NOT TARGET tsl::hopscotch_map) + include("${CMAKE_CURRENT_LIST_DIR}/tsl-hopscotch-mapTargets.cmake") + get_target_property(tsl-hopscotch-map_INCLUDE_DIRS tsl::hopscotch_map INTERFACE_INCLUDE_DIRECTORIES) +endif() diff --git a/external/include/common/hopscotch-map-2.3.1/doxygen.conf b/external/include/common/hopscotch-map-2.3.1/doxygen.conf new file mode 100644 index 000000000..5cd56fbc9 --- /dev/null +++ b/external/include/common/hopscotch-map-2.3.1/doxygen.conf @@ -0,0 +1,2486 @@ +# Doxyfile 1.8.11 + +# This file describes the settings to be used by the documentation system +# doxygen (www.doxygen.org) for a project. +# +# All text after a double hash (##) is considered a comment and is placed in +# front of the TAG it is preceding. +# +# All text after a single hash (#) is considered a comment and will be ignored. +# The format is: +# TAG = value [value, ...] +# For lists, items can also be appended using: +# TAG += value [value, ...] +# Values that contain spaces should be placed between quotes (\" \"). + +#--------------------------------------------------------------------------- +# Project related configuration options +#--------------------------------------------------------------------------- + +# This tag specifies the encoding used for all characters in the config file +# that follow. The default is UTF-8 which is also the encoding used for all text +# before the first occurrence of this tag. Doxygen uses libiconv (or the iconv +# built into libc) for the transcoding. See http://www.gnu.org/software/libiconv +# for the list of possible encodings. +# The default value is: UTF-8. + +DOXYFILE_ENCODING = UTF-8 + +# The PROJECT_NAME tag is a single word (or a sequence of words surrounded by +# double-quotes, unless you are using Doxywizard) that should identify the +# project for which the documentation is generated. This name is used in the +# title of most generated pages and in a few other places. +# The default value is: My Project. + +PROJECT_NAME = hopscotch-map + +# The PROJECT_NUMBER tag can be used to enter a project or revision number. This +# could be handy for archiving the generated documentation or if some version +# control system is used. + +PROJECT_NUMBER = + +# Using the PROJECT_BRIEF tag one can provide an optional one line description +# for a project that appears at the top of each page and should give viewer a +# quick idea about the purpose of the project. Keep the description short. + +PROJECT_BRIEF = + +# With the PROJECT_LOGO tag one can specify a logo or an icon that is included +# in the documentation. The maximum height of the logo should not exceed 55 +# pixels and the maximum width should not exceed 200 pixels. Doxygen will copy +# the logo to the output directory. + +PROJECT_LOGO = + +# The OUTPUT_DIRECTORY tag is used to specify the (relative or absolute) path +# into which the generated documentation will be written. If a relative path is +# entered, it will be relative to the location where doxygen was started. If +# left blank the current directory will be used. + +OUTPUT_DIRECTORY = doc/ + +# If the CREATE_SUBDIRS tag is set to YES then doxygen will create 4096 sub- +# directories (in 2 levels) under the output directory of each output format and +# will distribute the generated files over these directories. Enabling this +# option can be useful when feeding doxygen a huge amount of source files, where +# putting all generated files in the same directory would otherwise causes +# performance problems for the file system. +# The default value is: NO. + +CREATE_SUBDIRS = NO + +# If the ALLOW_UNICODE_NAMES tag is set to YES, doxygen will allow non-ASCII +# characters to appear in the names of generated files. If set to NO, non-ASCII +# characters will be escaped, for example _xE3_x81_x84 will be used for Unicode +# U+3044. +# The default value is: NO. + +ALLOW_UNICODE_NAMES = NO + +# The OUTPUT_LANGUAGE tag is used to specify the language in which all +# documentation generated by doxygen is written. Doxygen will use this +# information to generate all constant output in the proper language. +# Possible values are: Afrikaans, Arabic, Armenian, Brazilian, Catalan, Chinese, +# Chinese-Traditional, Croatian, Czech, Danish, Dutch, English (United States), +# Esperanto, Farsi (Persian), Finnish, French, German, Greek, Hungarian, +# Indonesian, Italian, Japanese, Japanese-en (Japanese with English messages), +# Korean, Korean-en (Korean with English messages), Latvian, Lithuanian, +# Macedonian, Norwegian, Persian (Farsi), Polish, Portuguese, Romanian, Russian, +# Serbian, Serbian-Cyrillic, Slovak, Slovene, Spanish, Swedish, Turkish, +# Ukrainian and Vietnamese. +# The default value is: English. + +OUTPUT_LANGUAGE = English + +# If the BRIEF_MEMBER_DESC tag is set to YES, doxygen will include brief member +# descriptions after the members that are listed in the file and class +# documentation (similar to Javadoc). Set to NO to disable this. +# The default value is: YES. + +BRIEF_MEMBER_DESC = YES + +# If the REPEAT_BRIEF tag is set to YES, doxygen will prepend the brief +# description of a member or function before the detailed description +# +# Note: If both HIDE_UNDOC_MEMBERS and BRIEF_MEMBER_DESC are set to NO, the +# brief descriptions will be completely suppressed. +# The default value is: YES. + +REPEAT_BRIEF = YES + +# This tag implements a quasi-intelligent brief description abbreviator that is +# used to form the text in various listings. Each string in this list, if found +# as the leading text of the brief description, will be stripped from the text +# and the result, after processing the whole list, is used as the annotated +# text. Otherwise, the brief description is used as-is. If left blank, the +# following values are used ($name is automatically replaced with the name of +# the entity):The $name class, The $name widget, The $name file, is, provides, +# specifies, contains, represents, a, an and the. + +ABBREVIATE_BRIEF = "The $name class" \ + "The $name widget" \ + "The $name file" \ + is \ + provides \ + specifies \ + contains \ + represents \ + a \ + an \ + the + +# If the ALWAYS_DETAILED_SEC and REPEAT_BRIEF tags are both set to YES then +# doxygen will generate a detailed section even if there is only a brief +# description. +# The default value is: NO. + +ALWAYS_DETAILED_SEC = NO + +# If the INLINE_INHERITED_MEMB tag is set to YES, doxygen will show all +# inherited members of a class in the documentation of that class as if those +# members were ordinary class members. Constructors, destructors and assignment +# operators of the base classes will not be shown. +# The default value is: NO. + +INLINE_INHERITED_MEMB = NO + +# If the FULL_PATH_NAMES tag is set to YES, doxygen will prepend the full path +# before files name in the file list and in the header files. If set to NO the +# shortest path that makes the file name unique will be used +# The default value is: YES. + +FULL_PATH_NAMES = YES + +# The STRIP_FROM_PATH tag can be used to strip a user-defined part of the path. +# Stripping is only done if one of the specified strings matches the left-hand +# part of the path. The tag can be used to show relative paths in the file list. +# If left blank the directory from which doxygen is run is used as the path to +# strip. +# +# Note that you can specify absolute paths here, but also relative paths, which +# will be relative from the directory where doxygen is started. +# This tag requires that the tag FULL_PATH_NAMES is set to YES. + +STRIP_FROM_PATH = + +# The STRIP_FROM_INC_PATH tag can be used to strip a user-defined part of the +# path mentioned in the documentation of a class, which tells the reader which +# header file to include in order to use a class. If left blank only the name of +# the header file containing the class definition is used. Otherwise one should +# specify the list of include paths that are normally passed to the compiler +# using the -I flag. + +STRIP_FROM_INC_PATH = + +# If the SHORT_NAMES tag is set to YES, doxygen will generate much shorter (but +# less readable) file names. This can be useful is your file systems doesn't +# support long names like on DOS, Mac, or CD-ROM. +# The default value is: NO. + +SHORT_NAMES = NO + +# If the JAVADOC_AUTOBRIEF tag is set to YES then doxygen will interpret the +# first line (until the first dot) of a Javadoc-style comment as the brief +# description. If set to NO, the Javadoc-style will behave just like regular Qt- +# style comments (thus requiring an explicit @brief command for a brief +# description.) +# The default value is: NO. + +JAVADOC_AUTOBRIEF = NO + +# If the QT_AUTOBRIEF tag is set to YES then doxygen will interpret the first +# line (until the first dot) of a Qt-style comment as the brief description. If +# set to NO, the Qt-style will behave just like regular Qt-style comments (thus +# requiring an explicit \brief command for a brief description.) +# The default value is: NO. + +QT_AUTOBRIEF = NO + +# The MULTILINE_CPP_IS_BRIEF tag can be set to YES to make doxygen treat a +# multi-line C++ special comment block (i.e. a block of //! or /// comments) as +# a brief description. This used to be the default behavior. The new default is +# to treat a multi-line C++ comment block as a detailed description. Set this +# tag to YES if you prefer the old behavior instead. +# +# Note that setting this tag to YES also means that rational rose comments are +# not recognized any more. +# The default value is: NO. + +MULTILINE_CPP_IS_BRIEF = NO + +# If the INHERIT_DOCS tag is set to YES then an undocumented member inherits the +# documentation from any documented member that it re-implements. +# The default value is: YES. + +INHERIT_DOCS = YES + +# If the SEPARATE_MEMBER_PAGES tag is set to YES then doxygen will produce a new +# page for each member. If set to NO, the documentation of a member will be part +# of the file/class/namespace that contains it. +# The default value is: NO. + +SEPARATE_MEMBER_PAGES = NO + +# The TAB_SIZE tag can be used to set the number of spaces in a tab. Doxygen +# uses this value to replace tabs by spaces in code fragments. +# Minimum value: 1, maximum value: 16, default value: 4. + +TAB_SIZE = 4 + +# This tag can be used to specify a number of aliases that act as commands in +# the documentation. An alias has the form: +# name=value +# For example adding +# "sideeffect=@par Side Effects:\n" +# will allow you to put the command \sideeffect (or @sideeffect) in the +# documentation, which will result in a user-defined paragraph with heading +# "Side Effects:". You can put \n's in the value part of an alias to insert +# newlines. + +ALIASES = + +# This tag can be used to specify a number of word-keyword mappings (TCL only). +# A mapping has the form "name=value". For example adding "class=itcl::class" +# will allow you to use the command class in the itcl::class meaning. + +TCL_SUBST = + +# Set the OPTIMIZE_OUTPUT_FOR_C tag to YES if your project consists of C sources +# only. Doxygen will then generate output that is more tailored for C. For +# instance, some of the names that are used will be different. The list of all +# members will be omitted, etc. +# The default value is: NO. + +OPTIMIZE_OUTPUT_FOR_C = NO + +# Set the OPTIMIZE_OUTPUT_JAVA tag to YES if your project consists of Java or +# Python sources only. Doxygen will then generate output that is more tailored +# for that language. For instance, namespaces will be presented as packages, +# qualified scopes will look different, etc. +# The default value is: NO. + +OPTIMIZE_OUTPUT_JAVA = NO + +# Set the OPTIMIZE_FOR_FORTRAN tag to YES if your project consists of Fortran +# sources. Doxygen will then generate output that is tailored for Fortran. +# The default value is: NO. + +OPTIMIZE_FOR_FORTRAN = NO + +# Set the OPTIMIZE_OUTPUT_VHDL tag to YES if your project consists of VHDL +# sources. Doxygen will then generate output that is tailored for VHDL. +# The default value is: NO. + +OPTIMIZE_OUTPUT_VHDL = NO + +# Doxygen selects the parser to use depending on the extension of the files it +# parses. With this tag you can assign which parser to use for a given +# extension. Doxygen has a built-in mapping, but you can override or extend it +# using this tag. The format is ext=language, where ext is a file extension, and +# language is one of the parsers supported by doxygen: IDL, Java, Javascript, +# C#, C, C++, D, PHP, Objective-C, Python, Fortran (fixed format Fortran: +# FortranFixed, free formatted Fortran: FortranFree, unknown formatted Fortran: +# Fortran. In the later case the parser tries to guess whether the code is fixed +# or free formatted code, this is the default for Fortran type files), VHDL. For +# instance to make doxygen treat .inc files as Fortran files (default is PHP), +# and .f files as C (default is Fortran), use: inc=Fortran f=C. +# +# Note: For files without extension you can use no_extension as a placeholder. +# +# Note that for custom extensions you also need to set FILE_PATTERNS otherwise +# the files are not read by doxygen. + +EXTENSION_MAPPING = + +# If the MARKDOWN_SUPPORT tag is enabled then doxygen pre-processes all comments +# according to the Markdown format, which allows for more readable +# documentation. See http://daringfireball.net/projects/markdown/ for details. +# The output of markdown processing is further processed by doxygen, so you can +# mix doxygen, HTML, and XML commands with Markdown formatting. Disable only in +# case of backward compatibilities issues. +# The default value is: YES. + +MARKDOWN_SUPPORT = YES + +# When enabled doxygen tries to link words that correspond to documented +# classes, or namespaces to their corresponding documentation. Such a link can +# be prevented in individual cases by putting a % sign in front of the word or +# globally by setting AUTOLINK_SUPPORT to NO. +# The default value is: YES. + +AUTOLINK_SUPPORT = YES + +# If you use STL classes (i.e. std::string, std::vector, etc.) but do not want +# to include (a tag file for) the STL sources as input, then you should set this +# tag to YES in order to let doxygen match functions declarations and +# definitions whose arguments contain STL classes (e.g. func(std::string); +# versus func(std::string) {}). This also make the inheritance and collaboration +# diagrams that involve STL classes more complete and accurate. +# The default value is: NO. + +BUILTIN_STL_SUPPORT = NO + +# If you use Microsoft's C++/CLI language, you should set this option to YES to +# enable parsing support. +# The default value is: NO. + +CPP_CLI_SUPPORT = NO + +# Set the SIP_SUPPORT tag to YES if your project consists of sip (see: +# http://www.riverbankcomputing.co.uk/software/sip/intro) sources only. Doxygen +# will parse them like normal C++ but will assume all classes use public instead +# of private inheritance when no explicit protection keyword is present. +# The default value is: NO. + +SIP_SUPPORT = NO + +# For Microsoft's IDL there are propget and propput attributes to indicate +# getter and setter methods for a property. Setting this option to YES will make +# doxygen to replace the get and set methods by a property in the documentation. +# This will only work if the methods are indeed getting or setting a simple +# type. If this is not the case, or you want to show the methods anyway, you +# should set this option to NO. +# The default value is: YES. + +IDL_PROPERTY_SUPPORT = YES + +# If member grouping is used in the documentation and the DISTRIBUTE_GROUP_DOC +# tag is set to YES then doxygen will reuse the documentation of the first +# member in the group (if any) for the other members of the group. By default +# all members of a group must be documented explicitly. +# The default value is: NO. + +DISTRIBUTE_GROUP_DOC = NO + +# If one adds a struct or class to a group and this option is enabled, then also +# any nested class or struct is added to the same group. By default this option +# is disabled and one has to add nested compounds explicitly via \ingroup. +# The default value is: NO. + +GROUP_NESTED_COMPOUNDS = NO + +# Set the SUBGROUPING tag to YES to allow class member groups of the same type +# (for instance a group of public functions) to be put as a subgroup of that +# type (e.g. under the Public Functions section). Set it to NO to prevent +# subgrouping. Alternatively, this can be done per class using the +# \nosubgrouping command. +# The default value is: YES. + +SUBGROUPING = YES + +# When the INLINE_GROUPED_CLASSES tag is set to YES, classes, structs and unions +# are shown inside the group in which they are included (e.g. using \ingroup) +# instead of on a separate page (for HTML and Man pages) or section (for LaTeX +# and RTF). +# +# Note that this feature does not work in combination with +# SEPARATE_MEMBER_PAGES. +# The default value is: NO. + +INLINE_GROUPED_CLASSES = NO + +# When the INLINE_SIMPLE_STRUCTS tag is set to YES, structs, classes, and unions +# with only public data fields or simple typedef fields will be shown inline in +# the documentation of the scope in which they are defined (i.e. file, +# namespace, or group documentation), provided this scope is documented. If set +# to NO, structs, classes, and unions are shown on a separate page (for HTML and +# Man pages) or section (for LaTeX and RTF). +# The default value is: NO. + +INLINE_SIMPLE_STRUCTS = NO + +# When TYPEDEF_HIDES_STRUCT tag is enabled, a typedef of a struct, union, or +# enum is documented as struct, union, or enum with the name of the typedef. So +# typedef struct TypeS {} TypeT, will appear in the documentation as a struct +# with name TypeT. When disabled the typedef will appear as a member of a file, +# namespace, or class. And the struct will be named TypeS. This can typically be +# useful for C code in case the coding convention dictates that all compound +# types are typedef'ed and only the typedef is referenced, never the tag name. +# The default value is: NO. + +TYPEDEF_HIDES_STRUCT = NO + +# The size of the symbol lookup cache can be set using LOOKUP_CACHE_SIZE. This +# cache is used to resolve symbols given their name and scope. Since this can be +# an expensive process and often the same symbol appears multiple times in the +# code, doxygen keeps a cache of pre-resolved symbols. If the cache is too small +# doxygen will become slower. If the cache is too large, memory is wasted. The +# cache size is given by this formula: 2^(16+LOOKUP_CACHE_SIZE). The valid range +# is 0..9, the default is 0, corresponding to a cache size of 2^16=65536 +# symbols. At the end of a run doxygen will report the cache usage and suggest +# the optimal cache size from a speed point of view. +# Minimum value: 0, maximum value: 9, default value: 0. + +LOOKUP_CACHE_SIZE = 0 + +#--------------------------------------------------------------------------- +# Build related configuration options +#--------------------------------------------------------------------------- + +# If the EXTRACT_ALL tag is set to YES, doxygen will assume all entities in +# documentation are documented, even if no documentation was available. Private +# class members and static file members will be hidden unless the +# EXTRACT_PRIVATE respectively EXTRACT_STATIC tags are set to YES. +# Note: This will also disable the warnings about undocumented members that are +# normally produced when WARNINGS is set to YES. +# The default value is: NO. + +EXTRACT_ALL = YES + +# If the EXTRACT_PRIVATE tag is set to YES, all private members of a class will +# be included in the documentation. +# The default value is: NO. + +EXTRACT_PRIVATE = NO + +# If the EXTRACT_PACKAGE tag is set to YES, all members with package or internal +# scope will be included in the documentation. +# The default value is: NO. + +EXTRACT_PACKAGE = NO + +# If the EXTRACT_STATIC tag is set to YES, all static members of a file will be +# included in the documentation. +# The default value is: NO. + +EXTRACT_STATIC = NO + +# If the EXTRACT_LOCAL_CLASSES tag is set to YES, classes (and structs) defined +# locally in source files will be included in the documentation. If set to NO, +# only classes defined in header files are included. Does not have any effect +# for Java sources. +# The default value is: YES. + +EXTRACT_LOCAL_CLASSES = YES + +# This flag is only useful for Objective-C code. If set to YES, local methods, +# which are defined in the implementation section but not in the interface are +# included in the documentation. If set to NO, only methods in the interface are +# included. +# The default value is: NO. + +EXTRACT_LOCAL_METHODS = NO + +# If this flag is set to YES, the members of anonymous namespaces will be +# extracted and appear in the documentation as a namespace called +# 'anonymous_namespace{file}', where file will be replaced with the base name of +# the file that contains the anonymous namespace. By default anonymous namespace +# are hidden. +# The default value is: NO. + +EXTRACT_ANON_NSPACES = NO + +# If the HIDE_UNDOC_MEMBERS tag is set to YES, doxygen will hide all +# undocumented members inside documented classes or files. If set to NO these +# members will be included in the various overviews, but no documentation +# section is generated. This option has no effect if EXTRACT_ALL is enabled. +# The default value is: NO. + +HIDE_UNDOC_MEMBERS = NO + +# If the HIDE_UNDOC_CLASSES tag is set to YES, doxygen will hide all +# undocumented classes that are normally visible in the class hierarchy. If set +# to NO, these classes will be included in the various overviews. This option +# has no effect if EXTRACT_ALL is enabled. +# The default value is: NO. + +HIDE_UNDOC_CLASSES = NO + +# If the HIDE_FRIEND_COMPOUNDS tag is set to YES, doxygen will hide all friend +# (class|struct|union) declarations. If set to NO, these declarations will be +# included in the documentation. +# The default value is: NO. + +HIDE_FRIEND_COMPOUNDS = NO + +# If the HIDE_IN_BODY_DOCS tag is set to YES, doxygen will hide any +# documentation blocks found inside the body of a function. If set to NO, these +# blocks will be appended to the function's detailed documentation block. +# The default value is: NO. + +HIDE_IN_BODY_DOCS = NO + +# The INTERNAL_DOCS tag determines if documentation that is typed after a +# \internal command is included. If the tag is set to NO then the documentation +# will be excluded. Set it to YES to include the internal documentation. +# The default value is: NO. + +INTERNAL_DOCS = NO + +# If the CASE_SENSE_NAMES tag is set to NO then doxygen will only generate file +# names in lower-case letters. If set to YES, upper-case letters are also +# allowed. This is useful if you have classes or files whose names only differ +# in case and if your file system supports case sensitive file names. Windows +# and Mac users are advised to set this option to NO. +# The default value is: system dependent. + +CASE_SENSE_NAMES = NO + +# If the HIDE_SCOPE_NAMES tag is set to NO then doxygen will show members with +# their full class and namespace scopes in the documentation. If set to YES, the +# scope will be hidden. +# The default value is: NO. + +HIDE_SCOPE_NAMES = NO + +# If the HIDE_COMPOUND_REFERENCE tag is set to NO (default) then doxygen will +# append additional text to a page's title, such as Class Reference. If set to +# YES the compound reference will be hidden. +# The default value is: NO. + +HIDE_COMPOUND_REFERENCE= NO + +# If the SHOW_INCLUDE_FILES tag is set to YES then doxygen will put a list of +# the files that are included by a file in the documentation of that file. +# The default value is: YES. + +SHOW_INCLUDE_FILES = YES + +# If the SHOW_GROUPED_MEMB_INC tag is set to YES then Doxygen will add for each +# grouped member an include statement to the documentation, telling the reader +# which file to include in order to use the member. +# The default value is: NO. + +SHOW_GROUPED_MEMB_INC = NO + +# If the FORCE_LOCAL_INCLUDES tag is set to YES then doxygen will list include +# files with double quotes in the documentation rather than with sharp brackets. +# The default value is: NO. + +FORCE_LOCAL_INCLUDES = NO + +# If the INLINE_INFO tag is set to YES then a tag [inline] is inserted in the +# documentation for inline members. +# The default value is: YES. + +INLINE_INFO = YES + +# If the SORT_MEMBER_DOCS tag is set to YES then doxygen will sort the +# (detailed) documentation of file and class members alphabetically by member +# name. If set to NO, the members will appear in declaration order. +# The default value is: YES. + +SORT_MEMBER_DOCS = YES + +# If the SORT_BRIEF_DOCS tag is set to YES then doxygen will sort the brief +# descriptions of file, namespace and class members alphabetically by member +# name. If set to NO, the members will appear in declaration order. Note that +# this will also influence the order of the classes in the class list. +# The default value is: NO. + +SORT_BRIEF_DOCS = NO + +# If the SORT_MEMBERS_CTORS_1ST tag is set to YES then doxygen will sort the +# (brief and detailed) documentation of class members so that constructors and +# destructors are listed first. If set to NO the constructors will appear in the +# respective orders defined by SORT_BRIEF_DOCS and SORT_MEMBER_DOCS. +# Note: If SORT_BRIEF_DOCS is set to NO this option is ignored for sorting brief +# member documentation. +# Note: If SORT_MEMBER_DOCS is set to NO this option is ignored for sorting +# detailed member documentation. +# The default value is: NO. + +SORT_MEMBERS_CTORS_1ST = NO + +# If the SORT_GROUP_NAMES tag is set to YES then doxygen will sort the hierarchy +# of group names into alphabetical order. If set to NO the group names will +# appear in their defined order. +# The default value is: NO. + +SORT_GROUP_NAMES = NO + +# If the SORT_BY_SCOPE_NAME tag is set to YES, the class list will be sorted by +# fully-qualified names, including namespaces. If set to NO, the class list will +# be sorted only by class name, not including the namespace part. +# Note: This option is not very useful if HIDE_SCOPE_NAMES is set to YES. +# Note: This option applies only to the class list, not to the alphabetical +# list. +# The default value is: NO. + +SORT_BY_SCOPE_NAME = NO + +# If the STRICT_PROTO_MATCHING option is enabled and doxygen fails to do proper +# type resolution of all parameters of a function it will reject a match between +# the prototype and the implementation of a member function even if there is +# only one candidate or it is obvious which candidate to choose by doing a +# simple string match. By disabling STRICT_PROTO_MATCHING doxygen will still +# accept a match between prototype and implementation in such cases. +# The default value is: NO. + +STRICT_PROTO_MATCHING = NO + +# The GENERATE_TODOLIST tag can be used to enable (YES) or disable (NO) the todo +# list. This list is created by putting \todo commands in the documentation. +# The default value is: YES. + +GENERATE_TODOLIST = YES + +# The GENERATE_TESTLIST tag can be used to enable (YES) or disable (NO) the test +# list. This list is created by putting \test commands in the documentation. +# The default value is: YES. + +GENERATE_TESTLIST = YES + +# The GENERATE_BUGLIST tag can be used to enable (YES) or disable (NO) the bug +# list. This list is created by putting \bug commands in the documentation. +# The default value is: YES. + +GENERATE_BUGLIST = YES + +# The GENERATE_DEPRECATEDLIST tag can be used to enable (YES) or disable (NO) +# the deprecated list. This list is created by putting \deprecated commands in +# the documentation. +# The default value is: YES. + +GENERATE_DEPRECATEDLIST= YES + +# The ENABLED_SECTIONS tag can be used to enable conditional documentation +# sections, marked by \if ... \endif and \cond +# ... \endcond blocks. + +ENABLED_SECTIONS = + +# The MAX_INITIALIZER_LINES tag determines the maximum number of lines that the +# initial value of a variable or macro / define can have for it to appear in the +# documentation. If the initializer consists of more lines than specified here +# it will be hidden. Use a value of 0 to hide initializers completely. The +# appearance of the value of individual variables and macros / defines can be +# controlled using \showinitializer or \hideinitializer command in the +# documentation regardless of this setting. +# Minimum value: 0, maximum value: 10000, default value: 30. + +MAX_INITIALIZER_LINES = 30 + +# Set the SHOW_USED_FILES tag to NO to disable the list of files generated at +# the bottom of the documentation of classes and structs. If set to YES, the +# list will mention the files that were used to generate the documentation. +# The default value is: YES. + +SHOW_USED_FILES = YES + +# Set the SHOW_FILES tag to NO to disable the generation of the Files page. This +# will remove the Files entry from the Quick Index and from the Folder Tree View +# (if specified). +# The default value is: YES. + +SHOW_FILES = YES + +# Set the SHOW_NAMESPACES tag to NO to disable the generation of the Namespaces +# page. This will remove the Namespaces entry from the Quick Index and from the +# Folder Tree View (if specified). +# The default value is: YES. + +SHOW_NAMESPACES = YES + +# The FILE_VERSION_FILTER tag can be used to specify a program or script that +# doxygen should invoke to get the current version for each file (typically from +# the version control system). Doxygen will invoke the program by executing (via +# popen()) the command command input-file, where command is the value of the +# FILE_VERSION_FILTER tag, and input-file is the name of an input file provided +# by doxygen. Whatever the program writes to standard output is used as the file +# version. For an example see the documentation. + +FILE_VERSION_FILTER = + +# The LAYOUT_FILE tag can be used to specify a layout file which will be parsed +# by doxygen. The layout file controls the global structure of the generated +# output files in an output format independent way. To create the layout file +# that represents doxygen's defaults, run doxygen with the -l option. You can +# optionally specify a file name after the option, if omitted DoxygenLayout.xml +# will be used as the name of the layout file. +# +# Note that if you run doxygen from a directory containing a file called +# DoxygenLayout.xml, doxygen will parse it automatically even if the LAYOUT_FILE +# tag is left empty. + +LAYOUT_FILE = + +# The CITE_BIB_FILES tag can be used to specify one or more bib files containing +# the reference definitions. This must be a list of .bib files. The .bib +# extension is automatically appended if omitted. This requires the bibtex tool +# to be installed. See also http://en.wikipedia.org/wiki/BibTeX for more info. +# For LaTeX the style of the bibliography can be controlled using +# LATEX_BIB_STYLE. To use this feature you need bibtex and perl available in the +# search path. See also \cite for info how to create references. + +CITE_BIB_FILES = + +#--------------------------------------------------------------------------- +# Configuration options related to warning and progress messages +#--------------------------------------------------------------------------- + +# The QUIET tag can be used to turn on/off the messages that are generated to +# standard output by doxygen. If QUIET is set to YES this implies that the +# messages are off. +# The default value is: NO. + +QUIET = NO + +# The WARNINGS tag can be used to turn on/off the warning messages that are +# generated to standard error (stderr) by doxygen. If WARNINGS is set to YES +# this implies that the warnings are on. +# +# Tip: Turn warnings on while writing the documentation. +# The default value is: YES. + +WARNINGS = YES + +# If the WARN_IF_UNDOCUMENTED tag is set to YES then doxygen will generate +# warnings for undocumented members. If EXTRACT_ALL is set to YES then this flag +# will automatically be disabled. +# The default value is: YES. + +WARN_IF_UNDOCUMENTED = YES + +# If the WARN_IF_DOC_ERROR tag is set to YES, doxygen will generate warnings for +# potential errors in the documentation, such as not documenting some parameters +# in a documented function, or documenting parameters that don't exist or using +# markup commands wrongly. +# The default value is: YES. + +WARN_IF_DOC_ERROR = YES + +# This WARN_NO_PARAMDOC option can be enabled to get warnings for functions that +# are documented, but have no documentation for their parameters or return +# value. If set to NO, doxygen will only warn about wrong or incomplete +# parameter documentation, but not about the absence of documentation. +# The default value is: NO. + +WARN_NO_PARAMDOC = NO + +# If the WARN_AS_ERROR tag is set to YES then doxygen will immediately stop when +# a warning is encountered. +# The default value is: NO. + +WARN_AS_ERROR = NO + +# The WARN_FORMAT tag determines the format of the warning messages that doxygen +# can produce. The string should contain the $file, $line, and $text tags, which +# will be replaced by the file and line number from which the warning originated +# and the warning text. Optionally the format may contain $version, which will +# be replaced by the version of the file (if it could be obtained via +# FILE_VERSION_FILTER) +# The default value is: $file:$line: $text. + +WARN_FORMAT = "$file:$line: $text" + +# The WARN_LOGFILE tag can be used to specify a file to which warning and error +# messages should be written. If left blank the output is written to standard +# error (stderr). + +WARN_LOGFILE = + +#--------------------------------------------------------------------------- +# Configuration options related to the input files +#--------------------------------------------------------------------------- + +# The INPUT tag is used to specify the files and/or directories that contain +# documented source files. You may enter file names like myfile.cpp or +# directories like /usr/src/myproject. Separate the files or directories with +# spaces. See also FILE_PATTERNS and EXTENSION_MAPPING +# Note: If this tag is empty the current directory is searched. + +INPUT = include/tsl/ README.md + +# This tag can be used to specify the character encoding of the source files +# that doxygen parses. Internally doxygen uses the UTF-8 encoding. Doxygen uses +# libiconv (or the iconv built into libc) for the transcoding. See the libiconv +# documentation (see: http://www.gnu.org/software/libiconv) for the list of +# possible encodings. +# The default value is: UTF-8. + +INPUT_ENCODING = UTF-8 + +# If the value of the INPUT tag contains directories, you can use the +# FILE_PATTERNS tag to specify one or more wildcard patterns (like *.cpp and +# *.h) to filter out the source-files in the directories. +# +# Note that for custom extensions or not directly supported extensions you also +# need to set EXTENSION_MAPPING for the extension otherwise the files are not +# read by doxygen. +# +# If left blank the following patterns are tested:*.c, *.cc, *.cxx, *.cpp, +# *.c++, *.java, *.ii, *.ixx, *.ipp, *.i++, *.inl, *.idl, *.ddl, *.odl, *.h, +# *.hh, *.hxx, *.hpp, *.h++, *.cs, *.d, *.php, *.php4, *.php5, *.phtml, *.inc, +# *.m, *.markdown, *.md, *.mm, *.dox, *.py, *.pyw, *.f90, *.f, *.for, *.tcl, +# *.vhd, *.vhdl, *.ucf, *.qsf, *.as and *.js. + +FILE_PATTERNS = *.c \ + *.cc \ + *.cxx \ + *.cpp \ + *.c++ \ + *.java \ + *.ii \ + *.ixx \ + *.ipp \ + *.i++ \ + *.inl \ + *.idl \ + *.ddl \ + *.odl \ + *.h \ + *.hh \ + *.hxx \ + *.hpp \ + *.h++ \ + *.cs \ + *.d \ + *.php \ + *.php4 \ + *.php5 \ + *.phtml \ + *.inc \ + *.m \ + *.markdown \ + *.md \ + *.mm \ + *.dox \ + *.py \ + *.pyw \ + *.f90 \ + *.f \ + *.for \ + *.tcl \ + *.vhd \ + *.vhdl \ + *.ucf \ + *.qsf \ + *.as \ + *.js + +# The RECURSIVE tag can be used to specify whether or not subdirectories should +# be searched for input files as well. +# The default value is: NO. + +RECURSIVE = YES + +# The EXCLUDE tag can be used to specify files and/or directories that should be +# excluded from the INPUT source files. This way you can easily exclude a +# subdirectory from a directory tree whose root is specified with the INPUT tag. +# +# Note that relative paths are relative to the directory from which doxygen is +# run. + +EXCLUDE = + +# The EXCLUDE_SYMLINKS tag can be used to select whether or not files or +# directories that are symbolic links (a Unix file system feature) are excluded +# from the input. +# The default value is: NO. + +EXCLUDE_SYMLINKS = NO + +# If the value of the INPUT tag contains directories, you can use the +# EXCLUDE_PATTERNS tag to specify one or more wildcard patterns to exclude +# certain files from those directories. +# +# Note that the wildcards are matched against the file with absolute path, so to +# exclude all test directories for example use the pattern */test/* + +EXCLUDE_PATTERNS = + +# The EXCLUDE_SYMBOLS tag can be used to specify one or more symbol names +# (namespaces, classes, functions, etc.) that should be excluded from the +# output. The symbol name can be a fully qualified name, a word, or if the +# wildcard * is used, a substring. Examples: ANamespace, AClass, +# AClass::ANamespace, ANamespace::*Test +# +# Note that the wildcards are matched against the file with absolute path, so to +# exclude all test directories use the pattern */test/* + +EXCLUDE_SYMBOLS = \ +tsl::detail_hopscotch_hash::has_is_transparent* \ +tsl::detail_hopscotch_hash::has_key_compare* \ +tsl::detail_hopscotch_hash::make_void \ +tsl::detail_hopscotch_hash::is_power_of_two_policy* \ +tsl::detail_hopscotch_hash::smallest_type_for_min_bits \ +tsl::detail_hopscotch_hash::hopscotch_bucket_hash* \ +tsl::detail_hopscotch_hash::hopscotch_bucket + +# The EXAMPLE_PATH tag can be used to specify one or more files or directories +# that contain example code fragments that are included (see the \include +# command). + +EXAMPLE_PATH = + +# If the value of the EXAMPLE_PATH tag contains directories, you can use the +# EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp and +# *.h) to filter out the source-files in the directories. If left blank all +# files are included. + +EXAMPLE_PATTERNS = * + +# If the EXAMPLE_RECURSIVE tag is set to YES then subdirectories will be +# searched for input files to be used with the \include or \dontinclude commands +# irrespective of the value of the RECURSIVE tag. +# The default value is: NO. + +EXAMPLE_RECURSIVE = NO + +# The IMAGE_PATH tag can be used to specify one or more files or directories +# that contain images that are to be included in the documentation (see the +# \image command). + +IMAGE_PATH = + +# The INPUT_FILTER tag can be used to specify a program that doxygen should +# invoke to filter for each input file. Doxygen will invoke the filter program +# by executing (via popen()) the command: +# +# +# +# where is the value of the INPUT_FILTER tag, and is the +# name of an input file. Doxygen will then use the output that the filter +# program writes to standard output. If FILTER_PATTERNS is specified, this tag +# will be ignored. +# +# Note that the filter must not add or remove lines; it is applied before the +# code is scanned, but not when the output code is generated. If lines are added +# or removed, the anchors will not be placed correctly. +# +# Note that for custom extensions or not directly supported extensions you also +# need to set EXTENSION_MAPPING for the extension otherwise the files are not +# properly processed by doxygen. + +INPUT_FILTER = + +# The FILTER_PATTERNS tag can be used to specify filters on a per file pattern +# basis. Doxygen will compare the file name with each pattern and apply the +# filter if there is a match. The filters are a list of the form: pattern=filter +# (like *.cpp=my_cpp_filter). See INPUT_FILTER for further information on how +# filters are used. If the FILTER_PATTERNS tag is empty or if none of the +# patterns match the file name, INPUT_FILTER is applied. +# +# Note that for custom extensions or not directly supported extensions you also +# need to set EXTENSION_MAPPING for the extension otherwise the files are not +# properly processed by doxygen. + +FILTER_PATTERNS = + +# If the FILTER_SOURCE_FILES tag is set to YES, the input filter (if set using +# INPUT_FILTER) will also be used to filter the input files that are used for +# producing the source files to browse (i.e. when SOURCE_BROWSER is set to YES). +# The default value is: NO. + +FILTER_SOURCE_FILES = NO + +# The FILTER_SOURCE_PATTERNS tag can be used to specify source filters per file +# pattern. A pattern will override the setting for FILTER_PATTERN (if any) and +# it is also possible to disable source filtering for a specific pattern using +# *.ext= (so without naming a filter). +# This tag requires that the tag FILTER_SOURCE_FILES is set to YES. + +FILTER_SOURCE_PATTERNS = + +# If the USE_MDFILE_AS_MAINPAGE tag refers to the name of a markdown file that +# is part of the input, its contents will be placed on the main page +# (index.html). This can be useful if you have a project on for instance GitHub +# and want to reuse the introduction page also for the doxygen output. + +USE_MDFILE_AS_MAINPAGE = README.md + +#--------------------------------------------------------------------------- +# Configuration options related to source browsing +#--------------------------------------------------------------------------- + +# If the SOURCE_BROWSER tag is set to YES then a list of source files will be +# generated. Documented entities will be cross-referenced with these sources. +# +# Note: To get rid of all source code in the generated output, make sure that +# also VERBATIM_HEADERS is set to NO. +# The default value is: NO. + +SOURCE_BROWSER = NO + +# Setting the INLINE_SOURCES tag to YES will include the body of functions, +# classes and enums directly into the documentation. +# The default value is: NO. + +INLINE_SOURCES = NO + +# Setting the STRIP_CODE_COMMENTS tag to YES will instruct doxygen to hide any +# special comment blocks from generated source code fragments. Normal C, C++ and +# Fortran comments will always remain visible. +# The default value is: YES. + +STRIP_CODE_COMMENTS = YES + +# If the REFERENCED_BY_RELATION tag is set to YES then for each documented +# function all documented functions referencing it will be listed. +# The default value is: NO. + +REFERENCED_BY_RELATION = NO + +# If the REFERENCES_RELATION tag is set to YES then for each documented function +# all documented entities called/used by that function will be listed. +# The default value is: NO. + +REFERENCES_RELATION = NO + +# If the REFERENCES_LINK_SOURCE tag is set to YES and SOURCE_BROWSER tag is set +# to YES then the hyperlinks from functions in REFERENCES_RELATION and +# REFERENCED_BY_RELATION lists will link to the source code. Otherwise they will +# link to the documentation. +# The default value is: YES. + +REFERENCES_LINK_SOURCE = YES + +# If SOURCE_TOOLTIPS is enabled (the default) then hovering a hyperlink in the +# source code will show a tooltip with additional information such as prototype, +# brief description and links to the definition and documentation. Since this +# will make the HTML file larger and loading of large files a bit slower, you +# can opt to disable this feature. +# The default value is: YES. +# This tag requires that the tag SOURCE_BROWSER is set to YES. + +SOURCE_TOOLTIPS = YES + +# If the USE_HTAGS tag is set to YES then the references to source code will +# point to the HTML generated by the htags(1) tool instead of doxygen built-in +# source browser. The htags tool is part of GNU's global source tagging system +# (see http://www.gnu.org/software/global/global.html). You will need version +# 4.8.6 or higher. +# +# To use it do the following: +# - Install the latest version of global +# - Enable SOURCE_BROWSER and USE_HTAGS in the config file +# - Make sure the INPUT points to the root of the source tree +# - Run doxygen as normal +# +# Doxygen will invoke htags (and that will in turn invoke gtags), so these +# tools must be available from the command line (i.e. in the search path). +# +# The result: instead of the source browser generated by doxygen, the links to +# source code will now point to the output of htags. +# The default value is: NO. +# This tag requires that the tag SOURCE_BROWSER is set to YES. + +USE_HTAGS = NO + +# If the VERBATIM_HEADERS tag is set the YES then doxygen will generate a +# verbatim copy of the header file for each class for which an include is +# specified. Set to NO to disable this. +# See also: Section \class. +# The default value is: YES. + +VERBATIM_HEADERS = YES + +# If the CLANG_ASSISTED_PARSING tag is set to YES then doxygen will use the +# clang parser (see: http://clang.llvm.org/) for more accurate parsing at the +# cost of reduced performance. This can be particularly helpful with template +# rich C++ code for which doxygen's built-in parser lacks the necessary type +# information. +# Note: The availability of this option depends on whether or not doxygen was +# generated with the -Duse-libclang=ON option for CMake. +# The default value is: NO. + +CLANG_ASSISTED_PARSING = YES + +# If clang assisted parsing is enabled you can provide the compiler with command +# line options that you would normally use when invoking the compiler. Note that +# the include paths will already be set by doxygen for the files and directories +# specified with INPUT and INCLUDE_PATH. +# This tag requires that the tag CLANG_ASSISTED_PARSING is set to YES. + +CLANG_OPTIONS = + +#--------------------------------------------------------------------------- +# Configuration options related to the alphabetical class index +#--------------------------------------------------------------------------- + +# If the ALPHABETICAL_INDEX tag is set to YES, an alphabetical index of all +# compounds will be generated. Enable this if the project contains a lot of +# classes, structs, unions or interfaces. +# The default value is: YES. + +ALPHABETICAL_INDEX = YES + +# The COLS_IN_ALPHA_INDEX tag can be used to specify the number of columns in +# which the alphabetical index list will be split. +# Minimum value: 1, maximum value: 20, default value: 5. +# This tag requires that the tag ALPHABETICAL_INDEX is set to YES. + +COLS_IN_ALPHA_INDEX = 5 + +# In case all classes in a project start with a common prefix, all classes will +# be put under the same header in the alphabetical index. The IGNORE_PREFIX tag +# can be used to specify a prefix (or a list of prefixes) that should be ignored +# while generating the index headers. +# This tag requires that the tag ALPHABETICAL_INDEX is set to YES. + +IGNORE_PREFIX = + +#--------------------------------------------------------------------------- +# Configuration options related to the HTML output +#--------------------------------------------------------------------------- + +# If the GENERATE_HTML tag is set to YES, doxygen will generate HTML output +# The default value is: YES. + +GENERATE_HTML = YES + +# The HTML_OUTPUT tag is used to specify where the HTML docs will be put. If a +# relative path is entered the value of OUTPUT_DIRECTORY will be put in front of +# it. +# The default directory is: html. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_OUTPUT = html + +# The HTML_FILE_EXTENSION tag can be used to specify the file extension for each +# generated HTML page (for example: .htm, .php, .asp). +# The default value is: .html. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_FILE_EXTENSION = .html + +# The HTML_HEADER tag can be used to specify a user-defined HTML header file for +# each generated HTML page. If the tag is left blank doxygen will generate a +# standard header. +# +# To get valid HTML the header file that includes any scripts and style sheets +# that doxygen needs, which is dependent on the configuration options used (e.g. +# the setting GENERATE_TREEVIEW). It is highly recommended to start with a +# default header using +# doxygen -w html new_header.html new_footer.html new_stylesheet.css +# YourConfigFile +# and then modify the file new_header.html. See also section "Doxygen usage" +# for information on how to generate the default header that doxygen normally +# uses. +# Note: The header is subject to change so you typically have to regenerate the +# default header when upgrading to a newer version of doxygen. For a description +# of the possible markers and block names see the documentation. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_HEADER = + +# The HTML_FOOTER tag can be used to specify a user-defined HTML footer for each +# generated HTML page. If the tag is left blank doxygen will generate a standard +# footer. See HTML_HEADER for more information on how to generate a default +# footer and what special commands can be used inside the footer. See also +# section "Doxygen usage" for information on how to generate the default footer +# that doxygen normally uses. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_FOOTER = + +# The HTML_STYLESHEET tag can be used to specify a user-defined cascading style +# sheet that is used by each HTML page. It can be used to fine-tune the look of +# the HTML output. If left blank doxygen will generate a default style sheet. +# See also section "Doxygen usage" for information on how to generate the style +# sheet that doxygen normally uses. +# Note: It is recommended to use HTML_EXTRA_STYLESHEET instead of this tag, as +# it is more robust and this tag (HTML_STYLESHEET) will in the future become +# obsolete. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_STYLESHEET = + +# The HTML_EXTRA_STYLESHEET tag can be used to specify additional user-defined +# cascading style sheets that are included after the standard style sheets +# created by doxygen. Using this option one can overrule certain style aspects. +# This is preferred over using HTML_STYLESHEET since it does not replace the +# standard style sheet and is therefore more robust against future updates. +# Doxygen will copy the style sheet files to the output directory. +# Note: The order of the extra style sheet files is of importance (e.g. the last +# style sheet in the list overrules the setting of the previous ones in the +# list). For an example see the documentation. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_EXTRA_STYLESHEET = + +# The HTML_EXTRA_FILES tag can be used to specify one or more extra images or +# other source files which should be copied to the HTML output directory. Note +# that these files will be copied to the base HTML output directory. Use the +# $relpath^ marker in the HTML_HEADER and/or HTML_FOOTER files to load these +# files. In the HTML_STYLESHEET file, use the file name only. Also note that the +# files will be copied as-is; there are no commands or markers available. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_EXTRA_FILES = + +# The HTML_COLORSTYLE_HUE tag controls the color of the HTML output. Doxygen +# will adjust the colors in the style sheet and background images according to +# this color. Hue is specified as an angle on a colorwheel, see +# http://en.wikipedia.org/wiki/Hue for more information. For instance the value +# 0 represents red, 60 is yellow, 120 is green, 180 is cyan, 240 is blue, 300 +# purple, and 360 is red again. +# Minimum value: 0, maximum value: 359, default value: 220. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_COLORSTYLE_HUE = 220 + +# The HTML_COLORSTYLE_SAT tag controls the purity (or saturation) of the colors +# in the HTML output. For a value of 0 the output will use grayscales only. A +# value of 255 will produce the most vivid colors. +# Minimum value: 0, maximum value: 255, default value: 100. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_COLORSTYLE_SAT = 100 + +# The HTML_COLORSTYLE_GAMMA tag controls the gamma correction applied to the +# luminance component of the colors in the HTML output. Values below 100 +# gradually make the output lighter, whereas values above 100 make the output +# darker. The value divided by 100 is the actual gamma applied, so 80 represents +# a gamma of 0.8, The value 220 represents a gamma of 2.2, and 100 does not +# change the gamma. +# Minimum value: 40, maximum value: 240, default value: 80. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_COLORSTYLE_GAMMA = 80 + +# If the HTML_TIMESTAMP tag is set to YES then the footer of each generated HTML +# page will contain the date and time when the page was generated. Setting this +# to YES can help to show when doxygen was last run and thus if the +# documentation is up to date. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_TIMESTAMP = NO + +# If the HTML_DYNAMIC_SECTIONS tag is set to YES then the generated HTML +# documentation will contain sections that can be hidden and shown after the +# page has loaded. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_DYNAMIC_SECTIONS = NO + +# With HTML_INDEX_NUM_ENTRIES one can control the preferred number of entries +# shown in the various tree structured indices initially; the user can expand +# and collapse entries dynamically later on. Doxygen will expand the tree to +# such a level that at most the specified number of entries are visible (unless +# a fully collapsed tree already exceeds this amount). So setting the number of +# entries 1 will produce a full collapsed tree by default. 0 is a special value +# representing an infinite number of entries and will result in a full expanded +# tree by default. +# Minimum value: 0, maximum value: 9999, default value: 100. +# This tag requires that the tag GENERATE_HTML is set to YES. + +HTML_INDEX_NUM_ENTRIES = 100 + +# If the GENERATE_DOCSET tag is set to YES, additional index files will be +# generated that can be used as input for Apple's Xcode 3 integrated development +# environment (see: http://developer.apple.com/tools/xcode/), introduced with +# OSX 10.5 (Leopard). To create a documentation set, doxygen will generate a +# Makefile in the HTML output directory. Running make will produce the docset in +# that directory and running make install will install the docset in +# ~/Library/Developer/Shared/Documentation/DocSets so that Xcode will find it at +# startup. See http://developer.apple.com/tools/creatingdocsetswithdoxygen.html +# for more information. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +GENERATE_DOCSET = NO + +# This tag determines the name of the docset feed. A documentation feed provides +# an umbrella under which multiple documentation sets from a single provider +# (such as a company or product suite) can be grouped. +# The default value is: Doxygen generated docs. +# This tag requires that the tag GENERATE_DOCSET is set to YES. + +DOCSET_FEEDNAME = "Doxygen generated docs" + +# This tag specifies a string that should uniquely identify the documentation +# set bundle. This should be a reverse domain-name style string, e.g. +# com.mycompany.MyDocSet. Doxygen will append .docset to the name. +# The default value is: org.doxygen.Project. +# This tag requires that the tag GENERATE_DOCSET is set to YES. + +DOCSET_BUNDLE_ID = org.doxygen.Project + +# The DOCSET_PUBLISHER_ID tag specifies a string that should uniquely identify +# the documentation publisher. This should be a reverse domain-name style +# string, e.g. com.mycompany.MyDocSet.documentation. +# The default value is: org.doxygen.Publisher. +# This tag requires that the tag GENERATE_DOCSET is set to YES. + +DOCSET_PUBLISHER_ID = org.doxygen.Publisher + +# The DOCSET_PUBLISHER_NAME tag identifies the documentation publisher. +# The default value is: Publisher. +# This tag requires that the tag GENERATE_DOCSET is set to YES. + +DOCSET_PUBLISHER_NAME = Publisher + +# If the GENERATE_HTMLHELP tag is set to YES then doxygen generates three +# additional HTML index files: index.hhp, index.hhc, and index.hhk. The +# index.hhp is a project file that can be read by Microsoft's HTML Help Workshop +# (see: http://www.microsoft.com/en-us/download/details.aspx?id=21138) on +# Windows. +# +# The HTML Help Workshop contains a compiler that can convert all HTML output +# generated by doxygen into a single compiled HTML file (.chm). Compiled HTML +# files are now used as the Windows 98 help format, and will replace the old +# Windows help format (.hlp) on all Windows platforms in the future. Compressed +# HTML files also contain an index, a table of contents, and you can search for +# words in the documentation. The HTML workshop also contains a viewer for +# compressed HTML files. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +GENERATE_HTMLHELP = NO + +# The CHM_FILE tag can be used to specify the file name of the resulting .chm +# file. You can add a path in front of the file if the result should not be +# written to the html output directory. +# This tag requires that the tag GENERATE_HTMLHELP is set to YES. + +CHM_FILE = + +# The HHC_LOCATION tag can be used to specify the location (absolute path +# including file name) of the HTML help compiler (hhc.exe). If non-empty, +# doxygen will try to run the HTML help compiler on the generated index.hhp. +# The file has to be specified with full path. +# This tag requires that the tag GENERATE_HTMLHELP is set to YES. + +HHC_LOCATION = + +# The GENERATE_CHI flag controls if a separate .chi index file is generated +# (YES) or that it should be included in the master .chm file (NO). +# The default value is: NO. +# This tag requires that the tag GENERATE_HTMLHELP is set to YES. + +GENERATE_CHI = NO + +# The CHM_INDEX_ENCODING is used to encode HtmlHelp index (hhk), content (hhc) +# and project file content. +# This tag requires that the tag GENERATE_HTMLHELP is set to YES. + +CHM_INDEX_ENCODING = + +# The BINARY_TOC flag controls whether a binary table of contents is generated +# (YES) or a normal table of contents (NO) in the .chm file. Furthermore it +# enables the Previous and Next buttons. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTMLHELP is set to YES. + +BINARY_TOC = NO + +# The TOC_EXPAND flag can be set to YES to add extra items for group members to +# the table of contents of the HTML help documentation and to the tree view. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTMLHELP is set to YES. + +TOC_EXPAND = NO + +# If the GENERATE_QHP tag is set to YES and both QHP_NAMESPACE and +# QHP_VIRTUAL_FOLDER are set, an additional index file will be generated that +# can be used as input for Qt's qhelpgenerator to generate a Qt Compressed Help +# (.qch) of the generated HTML documentation. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +GENERATE_QHP = NO + +# If the QHG_LOCATION tag is specified, the QCH_FILE tag can be used to specify +# the file name of the resulting .qch file. The path specified is relative to +# the HTML output folder. +# This tag requires that the tag GENERATE_QHP is set to YES. + +QCH_FILE = + +# The QHP_NAMESPACE tag specifies the namespace to use when generating Qt Help +# Project output. For more information please see Qt Help Project / Namespace +# (see: http://qt-project.org/doc/qt-4.8/qthelpproject.html#namespace). +# The default value is: org.doxygen.Project. +# This tag requires that the tag GENERATE_QHP is set to YES. + +QHP_NAMESPACE = org.doxygen.Project + +# The QHP_VIRTUAL_FOLDER tag specifies the namespace to use when generating Qt +# Help Project output. For more information please see Qt Help Project / Virtual +# Folders (see: http://qt-project.org/doc/qt-4.8/qthelpproject.html#virtual- +# folders). +# The default value is: doc. +# This tag requires that the tag GENERATE_QHP is set to YES. + +QHP_VIRTUAL_FOLDER = doc + +# If the QHP_CUST_FILTER_NAME tag is set, it specifies the name of a custom +# filter to add. For more information please see Qt Help Project / Custom +# Filters (see: http://qt-project.org/doc/qt-4.8/qthelpproject.html#custom- +# filters). +# This tag requires that the tag GENERATE_QHP is set to YES. + +QHP_CUST_FILTER_NAME = + +# The QHP_CUST_FILTER_ATTRS tag specifies the list of the attributes of the +# custom filter to add. For more information please see Qt Help Project / Custom +# Filters (see: http://qt-project.org/doc/qt-4.8/qthelpproject.html#custom- +# filters). +# This tag requires that the tag GENERATE_QHP is set to YES. + +QHP_CUST_FILTER_ATTRS = + +# The QHP_SECT_FILTER_ATTRS tag specifies the list of the attributes this +# project's filter section matches. Qt Help Project / Filter Attributes (see: +# http://qt-project.org/doc/qt-4.8/qthelpproject.html#filter-attributes). +# This tag requires that the tag GENERATE_QHP is set to YES. + +QHP_SECT_FILTER_ATTRS = + +# The QHG_LOCATION tag can be used to specify the location of Qt's +# qhelpgenerator. If non-empty doxygen will try to run qhelpgenerator on the +# generated .qhp file. +# This tag requires that the tag GENERATE_QHP is set to YES. + +QHG_LOCATION = + +# If the GENERATE_ECLIPSEHELP tag is set to YES, additional index files will be +# generated, together with the HTML files, they form an Eclipse help plugin. To +# install this plugin and make it available under the help contents menu in +# Eclipse, the contents of the directory containing the HTML and XML files needs +# to be copied into the plugins directory of eclipse. The name of the directory +# within the plugins directory should be the same as the ECLIPSE_DOC_ID value. +# After copying Eclipse needs to be restarted before the help appears. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +GENERATE_ECLIPSEHELP = NO + +# A unique identifier for the Eclipse help plugin. When installing the plugin +# the directory name containing the HTML and XML files should also have this +# name. Each documentation set should have its own identifier. +# The default value is: org.doxygen.Project. +# This tag requires that the tag GENERATE_ECLIPSEHELP is set to YES. + +ECLIPSE_DOC_ID = org.doxygen.Project + +# If you want full control over the layout of the generated HTML pages it might +# be necessary to disable the index and replace it with your own. The +# DISABLE_INDEX tag can be used to turn on/off the condensed index (tabs) at top +# of each HTML page. A value of NO enables the index and the value YES disables +# it. Since the tabs in the index contain the same information as the navigation +# tree, you can set this option to YES if you also set GENERATE_TREEVIEW to YES. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +DISABLE_INDEX = NO + +# The GENERATE_TREEVIEW tag is used to specify whether a tree-like index +# structure should be generated to display hierarchical information. If the tag +# value is set to YES, a side panel will be generated containing a tree-like +# index structure (just like the one that is generated for HTML Help). For this +# to work a browser that supports JavaScript, DHTML, CSS and frames is required +# (i.e. any modern browser). Windows users are probably better off using the +# HTML help feature. Via custom style sheets (see HTML_EXTRA_STYLESHEET) one can +# further fine-tune the look of the index. As an example, the default style +# sheet generated by doxygen has an example that shows how to put an image at +# the root of the tree instead of the PROJECT_NAME. Since the tree basically has +# the same information as the tab index, you could consider setting +# DISABLE_INDEX to YES when enabling this option. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +GENERATE_TREEVIEW = NO + +# The ENUM_VALUES_PER_LINE tag can be used to set the number of enum values that +# doxygen will group on one line in the generated HTML documentation. +# +# Note that a value of 0 will completely suppress the enum values from appearing +# in the overview section. +# Minimum value: 0, maximum value: 20, default value: 4. +# This tag requires that the tag GENERATE_HTML is set to YES. + +ENUM_VALUES_PER_LINE = 4 + +# If the treeview is enabled (see GENERATE_TREEVIEW) then this tag can be used +# to set the initial width (in pixels) of the frame in which the tree is shown. +# Minimum value: 0, maximum value: 1500, default value: 250. +# This tag requires that the tag GENERATE_HTML is set to YES. + +TREEVIEW_WIDTH = 250 + +# If the EXT_LINKS_IN_WINDOW option is set to YES, doxygen will open links to +# external symbols imported via tag files in a separate window. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +EXT_LINKS_IN_WINDOW = NO + +# Use this tag to change the font size of LaTeX formulas included as images in +# the HTML documentation. When you change the font size after a successful +# doxygen run you need to manually remove any form_*.png images from the HTML +# output directory to force them to be regenerated. +# Minimum value: 8, maximum value: 50, default value: 10. +# This tag requires that the tag GENERATE_HTML is set to YES. + +FORMULA_FONTSIZE = 10 + +# Use the FORMULA_TRANPARENT tag to determine whether or not the images +# generated for formulas are transparent PNGs. Transparent PNGs are not +# supported properly for IE 6.0, but are supported on all modern browsers. +# +# Note that when changing this option you need to delete any form_*.png files in +# the HTML output directory before the changes have effect. +# The default value is: YES. +# This tag requires that the tag GENERATE_HTML is set to YES. + +FORMULA_TRANSPARENT = YES + +# Enable the USE_MATHJAX option to render LaTeX formulas using MathJax (see +# http://www.mathjax.org) which uses client side Javascript for the rendering +# instead of using pre-rendered bitmaps. Use this if you do not have LaTeX +# installed or if you want to formulas look prettier in the HTML output. When +# enabled you may also need to install MathJax separately and configure the path +# to it using the MATHJAX_RELPATH option. +# The default value is: NO. +# This tag requires that the tag GENERATE_HTML is set to YES. + +USE_MATHJAX = NO + +# When MathJax is enabled you can set the default output format to be used for +# the MathJax output. See the MathJax site (see: +# http://docs.mathjax.org/en/latest/output.html) for more details. +# Possible values are: HTML-CSS (which is slower, but has the best +# compatibility), NativeMML (i.e. MathML) and SVG. +# The default value is: HTML-CSS. +# This tag requires that the tag USE_MATHJAX is set to YES. + +MATHJAX_FORMAT = HTML-CSS + +# When MathJax is enabled you need to specify the location relative to the HTML +# output directory using the MATHJAX_RELPATH option. The destination directory +# should contain the MathJax.js script. For instance, if the mathjax directory +# is located at the same level as the HTML output directory, then +# MATHJAX_RELPATH should be ../mathjax. The default value points to the MathJax +# Content Delivery Network so you can quickly see the result without installing +# MathJax. However, it is strongly recommended to install a local copy of +# MathJax from http://www.mathjax.org before deployment. +# The default value is: http://cdn.mathjax.org/mathjax/latest. +# This tag requires that the tag USE_MATHJAX is set to YES. + +MATHJAX_RELPATH = http://cdn.mathjax.org/mathjax/latest + +# The MATHJAX_EXTENSIONS tag can be used to specify one or more MathJax +# extension names that should be enabled during MathJax rendering. For example +# MATHJAX_EXTENSIONS = TeX/AMSmath TeX/AMSsymbols +# This tag requires that the tag USE_MATHJAX is set to YES. + +MATHJAX_EXTENSIONS = + +# The MATHJAX_CODEFILE tag can be used to specify a file with javascript pieces +# of code that will be used on startup of the MathJax code. See the MathJax site +# (see: http://docs.mathjax.org/en/latest/output.html) for more details. For an +# example see the documentation. +# This tag requires that the tag USE_MATHJAX is set to YES. + +MATHJAX_CODEFILE = + +# When the SEARCHENGINE tag is enabled doxygen will generate a search box for +# the HTML output. The underlying search engine uses javascript and DHTML and +# should work on any modern browser. Note that when using HTML help +# (GENERATE_HTMLHELP), Qt help (GENERATE_QHP), or docsets (GENERATE_DOCSET) +# there is already a search function so this one should typically be disabled. +# For large projects the javascript based search engine can be slow, then +# enabling SERVER_BASED_SEARCH may provide a better solution. It is possible to +# search using the keyboard; to jump to the search box use + S +# (what the is depends on the OS and browser, but it is typically +# , /