calixte created this revision.
calixte added a reviewer: marco-c.
Herald added subscribers: llvm-commits, Sanitizers, cfe-commits, jfb, hiraditya.
Herald added projects: clang, Sanitizers, LLVM.

When a program is forked with code coverage enabled, all the counters are 
flushed.
To avoid data race, a critical section has been added around `__gcov_flush` 
(https://reviews.llvm.org/D70910) but when forking the mutex is just duplicated 
and if it was in a locked state (in the parent process) then it's locked in the 
child process which leads to a deadlock when the child is trying to flush 
counters.
This patch aims to initialize the mutex in the child process when forking.
So a custom fork (`__gcov_fork`) is called from instrumented code and the mutex 
is then initialized in the child process.


Repository:
  rG LLVM Github Monorepo

https://reviews.llvm.org/D74071

Files:
  clang/lib/Driver/ToolChains/Darwin.cpp
  compiler-rt/lib/profile/GCDAProfiling.c
  llvm/lib/Transforms/Instrumentation/GCOVProfiling.cpp

Index: llvm/lib/Transforms/Instrumentation/GCOVProfiling.cpp
===================================================================
--- llvm/lib/Transforms/Instrumentation/GCOVProfiling.cpp
+++ llvm/lib/Transforms/Instrumentation/GCOVProfiling.cpp
@@ -631,20 +631,22 @@
 }
 
 void GCOVProfiler::AddFlushBeforeForkAndExec() {
-  SmallVector<Instruction *, 2> ForkAndExecs;
+  SmallVector<std::pair<bool, CallInst *>, 2> ForksAndExecs;
   for (auto &F : M->functions()) {
     auto *TLI = &GetTLI(F);
     for (auto &I : instructions(F)) {
       if (CallInst *CI = dyn_cast<CallInst>(&I)) {
         if (Function *Callee = CI->getCalledFunction()) {
           LibFunc LF;
-          if (TLI->getLibFunc(*Callee, LF) &&
-              (LF == LibFunc_fork || LF == LibFunc_execl ||
-               LF == LibFunc_execle || LF == LibFunc_execlp ||
-               LF == LibFunc_execv || LF == LibFunc_execvp ||
-               LF == LibFunc_execve || LF == LibFunc_execvpe ||
-               LF == LibFunc_execvP)) {
-            ForkAndExecs.push_back(&I);
+          if (TLI->getLibFunc(*Callee, LF)) {
+            if (LF == LibFunc_fork) {
+              ForksAndExecs.push_back({true, CI});
+            } else if (LF == LibFunc_execl || LF == LibFunc_execle ||
+                       LF == LibFunc_execlp || LF == LibFunc_execv ||
+                       LF == LibFunc_execvp || LF == LibFunc_execve ||
+                       LF == LibFunc_execvpe || LF == LibFunc_execvP) {
+              ForksAndExecs.push_back({false, CI});
+            }
           }
         }
       }
@@ -654,12 +656,21 @@
   // We need to split the block after the fork/exec call
   // because else the counters for the lines after will be
   // the same as before the call.
-  for (auto I : ForkAndExecs) {
-    IRBuilder<> Builder(I);
+  for (auto F : ForksAndExecs) {
+    IRBuilder<> Builder(F.second);
     FunctionType *FTy = FunctionType::get(Builder.getVoidTy(), {}, false);
     FunctionCallee GCOVFlush = M->getOrInsertFunction("__gcov_flush", FTy);
     Builder.CreateCall(GCOVFlush);
-    I->getParent()->splitBasicBlock(I);
+
+    if (F.first) {
+      // We've a fork, so we call a custom fork which will initialize the mutex
+      // used when flushing
+      FTy = FunctionType::get(Builder.getInt32Ty(), {}, false);
+      FunctionCallee GCOVFork = M->getOrInsertFunction("__gcov_fork", FTy);
+      F.second->setCalledFunction(GCOVFork);
+    }
+
+    F.second->getParent()->splitBasicBlock(F.second);
   }
 }
 
Index: compiler-rt/lib/profile/GCDAProfiling.c
===================================================================
--- compiler-rt/lib/profile/GCDAProfiling.c
+++ compiler-rt/lib/profile/GCDAProfiling.c
@@ -71,6 +71,9 @@
 static __inline void gcov_flush_unlock() {
   pthread_mutex_unlock(&gcov_flush_mutex);
 }
+static __inline void gcov_flush_init_mutex() {
+  pthread_mutex_init(&gcov_flush_mutex, NULL);
+}
 #else
 #include <windows.h>
 static SRWLOCK gcov_flush_mutex = SRWLOCK_INIT;
@@ -80,6 +83,9 @@
 static __inline void gcov_flush_unlock() {
   ReleaseSRWLockExclusive(&gcov_flush_mutex);
 }
+static __inline void gcov_flush_init_mutex() {
+  InitializeSRWLock(&gcov_flush_mutex);
+}
 #endif
 
 /* #define DEBUG_GCDAPROFILING */
@@ -638,6 +644,16 @@
   fn_list_insert(&flush_fn_list, fn);
 }
 
+pid_t __gcov_fork() {
+  pid_t pid = fork();
+
+  if (pid == 0) {
+    gcov_flush_init_mutex();
+  }
+
+  return pid;
+}
+
 void __gcov_flush() {
   gcov_flush_lock();
 
Index: clang/lib/Driver/ToolChains/Darwin.cpp
===================================================================
--- clang/lib/Driver/ToolChains/Darwin.cpp
+++ clang/lib/Driver/ToolChains/Darwin.cpp
@@ -1143,6 +1143,7 @@
   // runtime's functionality.
   if (hasExportSymbolDirective(Args)) {
     if (ForGCOV) {
+      addExportedSymbol(CmdArgs, "___gcov_fork");
       addExportedSymbol(CmdArgs, "___gcov_flush");
       addExportedSymbol(CmdArgs, "_flush_fn_list");
       addExportedSymbol(CmdArgs, "_writeout_fn_list");
_______________________________________________
cfe-commits mailing list
cfe-commits@lists.llvm.org
https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits

Reply via email to