From aa5abefafffdbd7f7660aba71c66ae577ac573d8 Mon Sep 17 00:00:00 2001 From: Leonardo de Moura Date: Wed, 17 Sep 2014 08:23:25 -0700 Subject: [PATCH] fix(util/thread_script_state): make sure system does not crash during finalization --- src/util/thread_script_state.cpp | 60 ++++++++++++++++++++------------ 1 file changed, 37 insertions(+), 23 deletions(-) diff --git a/src/util/thread_script_state.cpp b/src/util/thread_script_state.cpp index a937a599c..9e202e3a8 100644 --- a/src/util/thread_script_state.cpp +++ b/src/util/thread_script_state.cpp @@ -7,65 +7,78 @@ Author: Leonardo de Moura #include #include #include +#include #include "util/thread.h" #include "util/pair.h" #include "util/script_state.h" namespace lean { -static mutex g_code_mutex; -static std::vector> g_code; -static mutex g_state_mutex; -static std::vector g_states; -static std::vector g_available_states; +struct script_state_manager { + mutex g_code_mutex; + std::vector> g_code; + mutex g_state_mutex; + std::vector g_states; + std::vector g_available_states; +}; + +static script_state_manager & get_manager() { + static script_state_manager g_manager; + return g_manager; +} + +static script_state_manager & g_aux = get_manager(); // force manager to be initialized at startup /** \brief Execute \c code in all states in the pool */ void system_dostring(char const * code) { + script_state_manager & m = get_manager(); { // Execute code in all existing states - lock_guard lk(g_state_mutex); - for (auto & s : g_states) { + lock_guard lk(m.g_state_mutex); + for (auto & s : m.g_states) { s.dostring(code); } } { // Save code for future states - lock_guard lk(g_code_mutex); - g_code.push_back(mk_pair(true, code)); + lock_guard lk(m.g_code_mutex); + m.g_code.push_back(mk_pair(true, code)); } } /** \brief Import \c fname in all states in the pool */ void system_import(char const * fname) { + script_state_manager & m = get_manager(); { // Import file in all existing states - lock_guard lk(g_state_mutex); - for (auto & s : g_states) { + lock_guard lk(m.g_state_mutex); + for (auto & s : m.g_states) { s.import_explicit(fname); } } { // Save module for future states - lock_guard lk(g_code_mutex); - g_code.push_back(mk_pair(false, fname)); + lock_guard lk(m.g_code_mutex); + m.g_code.push_back(mk_pair(false, fname)); } } static script_state get_state() { + script_state_manager & m = get_manager(); { // Try to reuse existing state - lock_guard lk(g_state_mutex); - if (!g_available_states.empty()) { - script_state r(g_available_states.back()); - g_available_states.pop_back(); + lock_guard lk(m.g_state_mutex); + if (!m.g_available_states.empty()) { + script_state r(m.g_available_states.back()); + m.g_available_states.pop_back(); return r; } } { // Execute existing code in new state - lock_guard lk(g_code_mutex); + lock_guard lk(m.g_code_mutex); script_state r; - for (auto const & p : g_code) { + for (auto const & p : m.g_code) { if (p.first) r.dostring(p.second.c_str()); else @@ -73,16 +86,17 @@ static script_state get_state() { } { // save new state in vector of all states - lock_guard lk(g_state_mutex); - g_states.push_back(r); + lock_guard lk(m.g_state_mutex); + m.g_states.push_back(r); } return r; } } static void recycle_state(script_state s) { - lock_guard lk(g_state_mutex); - g_available_states.push_back(s); + script_state_manager & m = get_manager(); + lock_guard lk(m.g_state_mutex); + m.g_available_states.push_back(s); } struct script_state_ref {