mirror of
https://github.com/SFML/SFML.git
synced 2024-11-28 22:31:09 +08:00
More sophisticated handling of sf::err() data race
This commit is contained in:
parent
cfdba14e35
commit
0203dae7f6
@ -26,10 +26,13 @@
|
|||||||
// Headers
|
// Headers
|
||||||
////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////
|
||||||
#include <SFML/System/Err.hpp>
|
#include <SFML/System/Err.hpp>
|
||||||
|
#include <SFML/System/Lock.hpp>
|
||||||
|
#include <SFML/System/Mutex.hpp>
|
||||||
#include <SFML/System/ThreadLocalPtr.hpp>
|
#include <SFML/System/ThreadLocalPtr.hpp>
|
||||||
#include <SFML/System/NonCopyable.hpp>
|
#include <SFML/System/NonCopyable.hpp>
|
||||||
#include <streambuf>
|
#include <streambuf>
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
@ -96,6 +99,17 @@ private:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// No-op buffer
|
||||||
|
class NullBuffer : public std::streambuf
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
|
||||||
|
int overflow(int character)
|
||||||
|
{
|
||||||
|
return character;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// Groups a std::ostream with its std::streambuf
|
// Groups a std::ostream with its std::streambuf
|
||||||
struct ThreadLocalErr : sf::NonCopyable
|
struct ThreadLocalErr : sf::NonCopyable
|
||||||
{
|
{
|
||||||
@ -109,8 +123,33 @@ struct ThreadLocalErr : sf::NonCopyable
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// This per-thread variable holds the current instance of DefaultErrStreamBuf
|
// Note: the below contraption makes thread-safe usage of sf::err() possible, however it relies
|
||||||
sf::ThreadLocalPtr<ThreadLocalErr> currentErr(NULL);
|
// on several global variables to be alive when used (i.e. initialized and not yet destroyed).
|
||||||
|
// If sf::err() is called during global init/exit, UB might occur. There is a check for invocations
|
||||||
|
// during global exit, but it's best-effort and will leak memory.
|
||||||
|
|
||||||
|
typedef sf::ThreadLocalPtr<ThreadLocalErr> ErrPtr;
|
||||||
|
|
||||||
|
ErrPtr currentErr(NULL); //< Per-thread variable holds the current instance of DefaultErrStreamBuf
|
||||||
|
std::vector<ThreadLocalErr*> activePtrs; //< Pointers to Err objects across all threads
|
||||||
|
bool isAtexitRegistered; //< Whether the global cleanup has been initialized
|
||||||
|
bool isAtexitExecuted; //< Whether the global cleanup has run
|
||||||
|
sf::Mutex mutex; //< Synchronize access to the above book-keeping
|
||||||
|
|
||||||
|
// Function called at program exit, destroys all active ThreadLocalErr instances
|
||||||
|
void destroyErrs()
|
||||||
|
{
|
||||||
|
// Note: since this function has been registered with atexit() *after* the above global variables
|
||||||
|
// have been initialized*, it should be invoked *before* they are destroyed, ensuring their liveness.
|
||||||
|
// ______
|
||||||
|
// * that is, unless err() is called during global init, in which case nothing is guaranteed (likely UB).
|
||||||
|
|
||||||
|
for (std::vector<ThreadLocalErr*>::iterator it = activePtrs.begin(); it != activePtrs.end(); ++it)
|
||||||
|
delete *it;
|
||||||
|
|
||||||
|
activePtrs.clear();
|
||||||
|
isAtexitExecuted = true;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -119,9 +158,32 @@ namespace sf
|
|||||||
////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////
|
||||||
std::ostream& err()
|
std::ostream& err()
|
||||||
{
|
{
|
||||||
|
// Note: access not synchronized, to not incur locking on every err() invocation.
|
||||||
|
if (isAtexitExecuted)
|
||||||
|
{
|
||||||
|
// Last resort if err() is called after the thread-local instance has been destroyed.
|
||||||
|
// If this happens, all that counts is avoiding UB and data races, so the stream
|
||||||
|
// will leak memory and won't be operational (standard facilities might anyway be in
|
||||||
|
// the process of destruction).
|
||||||
|
NullBuffer* buf = new NullBuffer();
|
||||||
|
return *new std::ostream(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Lazy-initialize thread-local Err provider
|
||||||
if (currentErr == NULL)
|
if (currentErr == NULL)
|
||||||
{
|
{
|
||||||
currentErr = new ThreadLocalErr();
|
sf::Lock lock(mutex);
|
||||||
|
|
||||||
|
ThreadLocalErr* newErr = new ThreadLocalErr();
|
||||||
|
|
||||||
|
currentErr = newErr;
|
||||||
|
activePtrs.push_back(newErr);
|
||||||
|
|
||||||
|
if (!isAtexitRegistered)
|
||||||
|
{
|
||||||
|
std::atexit(&destroyErrs);
|
||||||
|
isAtexitRegistered = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return currentErr->stream;
|
return currentErr->stream;
|
||||||
|
Loading…
Reference in New Issue
Block a user