Add freeScheduler/freeTaskManager and call it later than exitScheduler
authorIan Lynagh <igloo@earth.li>
Mon, 11 Dec 2006 13:10:04 +0000 (13:10 +0000)
committerIan Lynagh <igloo@earth.li>
Mon, 11 Dec 2006 13:10:04 +0000 (13:10 +0000)
We were freeing the tasks in exitScheduler (stopTaskManager) before
exitStorage (stat_exit), but the latter needs to walk down the list
printing stats. Resulted in segfaults with commands like
    ghc -v0 -e main q.hs -H32m -H32m +RTS -Sstderr
(where q.hs is trivial), but very sensitive to exact commandline and
libc version or something.

rts/RtsStartup.c
rts/Schedule.c
rts/Schedule.h
rts/Task.c
rts/Task.h

index 5bb9806..937334b 100644 (file)
@@ -438,6 +438,9 @@ hs_exit(void)
     // also outputs the stats (+RTS -s) info.
     exitStorage();
     
+    /* free the tasks */
+    freeScheduler();
+
     /* free shared Typeable store */
     exitTypeableStore();
 
index cf62c5a..77dcf41 100644 (file)
@@ -2582,6 +2582,12 @@ exitScheduler( void )
 #endif
 }
 
+void
+freeScheduler( void )
+{
+    freeTaskManager();
+}
+
 /* ---------------------------------------------------------------------------
    Where are the roots that we know about?
 
index 2afedee..ba5efc2 100644 (file)
@@ -19,6 +19,7 @@
  */
 void initScheduler (void);
 void exitScheduler (void);
+void freeScheduler (void);
 
 // Place a new thread on the run queue of the current Capability
 void scheduleThread (Capability *cap, StgTSO *tso);
index 588d414..0e51dbf 100644 (file)
@@ -68,19 +68,33 @@ initTaskManager (void)
 void
 stopTaskManager (void)
 {
-    Task *task, *next;
+    Task *task;
 
     debugTrace(DEBUG_sched, 
               "stopping task manager, %d tasks still running",
               tasksRunning);
 
     ACQUIRE_LOCK(&sched_mutex);
-    for (task = task_free_list; task != NULL; task = next) {
-        next = task->next;
+    for (task = task_free_list; task != NULL; task = task->next) {
 #if defined(THREADED_RTS)
         closeCondition(&task->cond);
         closeMutex(&task->lock);
 #endif
+    }
+    RELEASE_LOCK(&sched_mutex);
+}
+
+
+void
+freeTaskManager (void)
+{
+    Task *task, *next;
+
+    debugTrace(DEBUG_sched, "freeing task manager");
+
+    ACQUIRE_LOCK(&sched_mutex);
+    for (task = task_free_list; task != NULL; task = next) {
+        next = task->next;
         stgFree(task);
     }
     task_free_list = NULL;
index cc450a7..3b7a08e 100644 (file)
@@ -170,6 +170,7 @@ extern Task *all_tasks;
 //
 void initTaskManager (void);
 void stopTaskManager (void);
+void freeTaskManager (void);
 
 // Create a new Task for a bound thread
 // Requires: sched_mutex.