If we hit an invariant in the ScopedDebugInfo logging code while processing an invariant that happened during normal code flow, we hit some sort of deadlock state. The sequence of events is that the first invariant calls into invariantFailedWithMsg which calls callAbort which calls std::abort(). We have installed some custom signal handler functions (see here) which get invoked when we process the specified signals. This means that next we call abruptQuitAction which calls abruptQuit, which invokes the logging via printErrorBlock. While logging, we hit an invariant again, but this time the behavior is different when we reach callAbort. Notice that the call to std::abort() is within a function static (static variable declared in a function):
[[maybe_unused]] static auto initOnce = (std::abort(), 0);
This means that we initialize the value of initOnce upon the first entry of callAbort and then use that value for any subsequent calls. However, "initialization recursively enters the block in which the variable is being initialized", and according to the cpp docs this leads to undefined behavior. In our case, the thread handling the connection to the shell hangs. I tried attaching gdb to the process and it showed that the thread was waiting for a condition variable (I have a screenshot of this in the doc linked below, image 5). This also makes some sense because according to the same section of the cpp docs I linked earlier, "usual implementations of this feature [static block variables] use variants of the double-checked locking pattern."
- related to
- 
                    SERVER-95134 Add mechanism to avoid asserting while extracting debug info within tassert handler -         
- Closed
 
-