Fixed race conditions and forgotten clear in Lua ref tracking. (#3530)
This fixes occasional crashes on plugin reload.
This commit is contained in:
parent
f62711f97c
commit
fb0fc07579
@ -218,7 +218,7 @@ void cLuaState::cTrackedRef::Clear(void)
|
|||||||
// Free the reference:
|
// Free the reference:
|
||||||
lua_State * luaState = nullptr;
|
lua_State * luaState = nullptr;
|
||||||
{
|
{
|
||||||
auto cs = m_CS;
|
auto cs = m_CS.load();
|
||||||
if (cs != nullptr)
|
if (cs != nullptr)
|
||||||
{
|
{
|
||||||
cCSLock Lock(*cs);
|
cCSLock Lock(*cs);
|
||||||
@ -246,7 +246,7 @@ void cLuaState::cTrackedRef::Clear(void)
|
|||||||
|
|
||||||
bool cLuaState::cTrackedRef::IsValid(void)
|
bool cLuaState::cTrackedRef::IsValid(void)
|
||||||
{
|
{
|
||||||
auto cs = m_CS;
|
auto cs = m_CS.load();
|
||||||
if (cs == nullptr)
|
if (cs == nullptr)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
@ -261,7 +261,7 @@ bool cLuaState::cTrackedRef::IsValid(void)
|
|||||||
|
|
||||||
bool cLuaState::cTrackedRef::IsSameLuaState(cLuaState & a_LuaState)
|
bool cLuaState::cTrackedRef::IsSameLuaState(cLuaState & a_LuaState)
|
||||||
{
|
{
|
||||||
auto cs = m_CS;
|
auto cs = m_CS.load();
|
||||||
if (cs == nullptr)
|
if (cs == nullptr)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
@ -285,7 +285,7 @@ bool cLuaState::cTrackedRef::IsSameLuaState(cLuaState & a_LuaState)
|
|||||||
|
|
||||||
void cLuaState::cTrackedRef::Invalidate(void)
|
void cLuaState::cTrackedRef::Invalidate(void)
|
||||||
{
|
{
|
||||||
auto cs = m_CS;
|
auto cs = m_CS.load();
|
||||||
if (cs == nullptr)
|
if (cs == nullptr)
|
||||||
{
|
{
|
||||||
// Already invalid
|
// Already invalid
|
||||||
@ -548,6 +548,7 @@ void cLuaState::Close(void)
|
|||||||
{
|
{
|
||||||
r->Invalidate();
|
r->Invalidate();
|
||||||
}
|
}
|
||||||
|
m_TrackedRefs.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
cCanonLuaStates::Remove(*this);
|
cCanonLuaStates::Remove(*this);
|
||||||
@ -2357,6 +2358,7 @@ void cLuaState::cRef::RefStack(cLuaState & a_LuaState, int a_StackPos)
|
|||||||
{
|
{
|
||||||
UnRef();
|
UnRef();
|
||||||
}
|
}
|
||||||
|
ASSERT(cCanonLuaStates::GetCanonState(a_LuaState)->m_CS.IsLockedByCurrentThread());
|
||||||
m_LuaState = a_LuaState;
|
m_LuaState = a_LuaState;
|
||||||
lua_pushvalue(a_LuaState, a_StackPos); // Push a copy of the value at a_StackPos onto the stack
|
lua_pushvalue(a_LuaState, a_StackPos); // Push a copy of the value at a_StackPos onto the stack
|
||||||
m_Ref = luaL_ref(a_LuaState, LUA_REGISTRYINDEX);
|
m_Ref = luaL_ref(a_LuaState, LUA_REGISTRYINDEX);
|
||||||
@ -2370,6 +2372,7 @@ void cLuaState::cRef::UnRef(void)
|
|||||||
{
|
{
|
||||||
if (IsValid())
|
if (IsValid())
|
||||||
{
|
{
|
||||||
|
ASSERT(cCanonLuaStates::GetCanonState(m_LuaState)->m_CS.IsLockedByCurrentThread());
|
||||||
luaL_unref(m_LuaState, LUA_REGISTRYINDEX, m_Ref);
|
luaL_unref(m_LuaState, LUA_REGISTRYINDEX, m_Ref);
|
||||||
}
|
}
|
||||||
m_LuaState = nullptr;
|
m_LuaState = nullptr;
|
||||||
|
@ -35,6 +35,7 @@ extern "C"
|
|||||||
#include "lua/src/lauxlib.h"
|
#include "lua/src/lauxlib.h"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#include <atomic>
|
||||||
#include "../Vector3.h"
|
#include "../Vector3.h"
|
||||||
#include "../Defines.h"
|
#include "../Defines.h"
|
||||||
#include "PluginManager.h"
|
#include "PluginManager.h"
|
||||||
@ -189,24 +190,29 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
/** Set the contained reference to the object at the specified Lua state's stack position.
|
/** Set the contained reference to the object at the specified Lua state's stack position.
|
||||||
If another reference has been previously contained, it is freed first. */
|
If another reference has been previously contained, it is Clear()-ed first. */
|
||||||
bool RefStack(cLuaState & a_LuaState, int a_StackPos);
|
bool RefStack(cLuaState & a_LuaState, int a_StackPos);
|
||||||
|
|
||||||
/** Frees the contained reference, if any. */
|
/** Frees the contained reference, if any.
|
||||||
|
Untracks the reference from its canon Lua state. */
|
||||||
void Clear(void);
|
void Clear(void);
|
||||||
|
|
||||||
/** Returns true if the contained reference is valid. */
|
/** Returns true if the contained reference is valid.
|
||||||
|
(Note that depending on this value is not thread-safe, another thread may invalidate the ref in the meantime. It is meant for quick ASSERTs only). */
|
||||||
bool IsValid(void);
|
bool IsValid(void);
|
||||||
|
|
||||||
/** Returns true if the reference resides in the specified Lua state.
|
/** Returns true if the reference resides in the specified Lua state.
|
||||||
Internally, compares the reference's canon Lua state. */
|
Internally, compares the reference's canon Lua state.
|
||||||
|
(Note that depending on this value is not thread-safe, another thread may modify the ref in the meantime. It is meant for quick ASSERTs only). */
|
||||||
bool IsSameLuaState(cLuaState & a_LuaState);
|
bool IsSameLuaState(cLuaState & a_LuaState);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
friend class cLuaState;
|
friend class cLuaState;
|
||||||
|
|
||||||
/** The mutex protecting m_Ref against multithreaded access */
|
/** The mutex protecting m_Ref against multithreaded access.
|
||||||
cCriticalSection * m_CS;
|
Actually points to the canon Lua state's m_CriticalSection.
|
||||||
|
Is nullptr when ref is empty (not bound). */
|
||||||
|
std::atomic<cCriticalSection *> m_CS;
|
||||||
|
|
||||||
/** Reference to the Lua callback */
|
/** Reference to the Lua callback */
|
||||||
cRef m_Ref;
|
cRef m_Ref;
|
||||||
@ -254,7 +260,7 @@ public:
|
|||||||
template <typename... Args>
|
template <typename... Args>
|
||||||
bool Call(Args &&... args)
|
bool Call(Args &&... args)
|
||||||
{
|
{
|
||||||
auto cs = m_CS;
|
auto cs = m_CS.load();
|
||||||
if (cs == nullptr)
|
if (cs == nullptr)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
@ -336,7 +342,7 @@ public:
|
|||||||
template <typename... Args>
|
template <typename... Args>
|
||||||
bool CallTableFn(const char * a_FnName, Args &&... args)
|
bool CallTableFn(const char * a_FnName, Args &&... args)
|
||||||
{
|
{
|
||||||
auto cs = m_CS;
|
auto cs = m_CS.load();
|
||||||
if (cs == nullptr)
|
if (cs == nullptr)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
@ -356,7 +362,7 @@ public:
|
|||||||
template <typename... Args>
|
template <typename... Args>
|
||||||
bool CallTableFnWithSelf(const char * a_FnName, Args &&... args)
|
bool CallTableFnWithSelf(const char * a_FnName, Args &&... args)
|
||||||
{
|
{
|
||||||
auto cs = m_CS;
|
auto cs = m_CS.load();
|
||||||
if (cs == nullptr)
|
if (cs == nullptr)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
|
@ -204,6 +204,7 @@ bool cPrefabPiecePool::LoadFromCubeset(const AString & a_Contents, const AString
|
|||||||
// Load the file in the Lua interpreter:
|
// Load the file in the Lua interpreter:
|
||||||
cLuaState Lua(Printf("LoadablePiecePool %s", a_FileName.c_str()));
|
cLuaState Lua(Printf("LoadablePiecePool %s", a_FileName.c_str()));
|
||||||
Lua.Create();
|
Lua.Create();
|
||||||
|
cLuaState::cLock lock(Lua);
|
||||||
if (!Lua.LoadString(a_Contents, a_FileName, a_LogWarnings))
|
if (!Lua.LoadString(a_Contents, a_FileName, a_LogWarnings))
|
||||||
{
|
{
|
||||||
// Reason for failure has already been logged in LoadFile()
|
// Reason for failure has already been logged in LoadFile()
|
||||||
|
Loading…
Reference in New Issue
Block a user