catch urcu-paulmck.spin to my local version
authorPaul E. McKenney <paulmck@linux.vnet.ibm.com>
Tue, 26 May 2009 17:35:53 +0000 (13:35 -0400)
committerMathieu Desnoyers <mathieu.desnoyers@polymtl.ca>
Tue, 26 May 2009 17:35:53 +0000 (13:35 -0400)
The attached patch catches the repository up to my local version.
This adds the following:

o       Checking for no-progress cycles.

o       Allow enabling and disabling of update-side signal-mediated
        broadcast memory barriers via #define statements.

o       Enabling only those update-side signal-mediated broadcast
        memory barriers that are required for correctness.

This version assumes that synchronize_rcu() is locally ordered, for
example, with smp_mb() calls between each pair of major statements.
Please note that smp_mb() is -not- assumed within the loops nor
immediately before the second counter flip.

Signed-off-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
Signed-off-by: Mathieu Desnoyers <mathieu.desnoyers@polymtl.ca>
formal-model/urcu-paulmck/urcu-paulmck.spin
formal-model/urcu-paulmck/urcu.sh

index e20a6e555ad1e51d2d01fefc8f034fc90d3f5db5..15338de4a10f7a2b1d684fdbbb29d948eb8915dc 100644 (file)
@@ -4,6 +4,15 @@
  *      git archive at git://lttng.org/userspace-rcu.git, but with
  *     memory barriers removed.
  *
+ * This validates a single reader against a single updater.  The
+ * updater is assumed to have smp_mb() barriers between each pair
+ * of operations, and this model validates that a signal-mediated
+ * broadcast barrier is required only at the beginning and end of
+ * the synchronize_rcu().
+ *
+ * Note that the second half of a prior synchronize_rcu() is modelled
+ * as well as the current synchronize_rcu().
+ *
  * This program is free software; you can redistribute it and/or modify
  * it under the terms of the GNU General Public License as published by
  * the Free Software Foundation; either version 2 of the License, or
 
 /* Promela validation variables. */
 
-bit removed = 0;  /* Has RCU removal happened, e.g., list_del_rcu()? */
-bit free = 0;     /* Has RCU reclamation happened, e.g., kfree()? */
-bit need_mb = 0;  /* =1 says need reader mb, =0 for reader response. */
-byte reader_progress[4];
-                 /* Count of read-side statement executions. */
+bit removed = 0;       /* Has RCU removal happened, e.g., list_del_rcu()? */
+bit free = 0;          /* Has RCU reclamation happened, e.g., kfree()? */
+bit need_mb = 0;       /* =1 says need reader mb, =0 for reader response. */
+byte reader_progress[4]; /* Count of read-side statement executions. */
+bit reader_done = 0;   /* =0 says reader still running, =1 says done. */
+bit updater_done = 0;  /* =0 says updater still running, =1 says done. */
+
+/* Broadcast memory barriers to enable, prior synchronize_rcu() instance. */
+
+/* #define MB_A */     /* Not required for correctness. */
+/* #define MB_B */     /* Not required for correctness. */
+/* #define MB_C */     /* Not required for correctness. */
+
+/* Broadcast memory barriers to enable, current synchronize_rcu() instance. */
+
+#define MB_D           /* First broadcast barrier in synchronize_rcu(). */
+/* #define MB_E */     /* Not required for correctness. */
+/* #define MB_F */     /* Not required for correctness. */
+#define MB_G           /* Last broadcast barrier in synchronize_rcu(). */
 
 /* urcu definitions and variables, taken straight from the algorithm. */
 
@@ -51,7 +74,6 @@ proctype urcu_reader()
        do
        :: need_mb == 1 ->
                need_mb = 0;
-       :: 1 -> skip;
        :: 1 -> break;
        od;
 
@@ -93,7 +115,7 @@ proctype urcu_reader()
                                    reader_progress[2] +
                                    reader_progress[3] == 0) && need_mb == 1 ->
                                        need_mb = 0;
-                               :: 1 -> skip;
+                               :: need_mb == 0 && !updater_done -> skip;
                                :: 1 -> break;
                                od;
                                urcu_active_readers = tmp;
@@ -151,7 +173,6 @@ proctype urcu_reader()
                        do
                        :: need_mb == 1 ->
                                need_mb = 0;
-                       :: 1 -> skip;
                        :: 1 -> break;
                        od;
                :: else -> skip;
@@ -168,11 +189,21 @@ proctype urcu_reader()
        od;
        assert((tmp_free == 0) || (tmp_removed == 1));
 
-       /* Process any late-arriving memory-barrier requests. */
+       /* Reader has completed. */
+       reader_done = 1;
+
+       /*
+        * Process any late-arriving memory-barrier requests, and
+        * in addition create a progress cycle.
+        */
+       reader_done = 1;
+
        do
        :: need_mb == 1 ->
                need_mb = 0;
-       :: 1 -> skip;
+       :: 1 ->
+progress_reader:
+               skip;
        :: 1 -> break;
        od;
 }
@@ -183,18 +214,27 @@ proctype urcu_updater()
 {
        byte tmp;
 
+       /*
+        * ----------------------------------------------------------------
+        * prior synchronize_rcu().
+        */
+
        /* prior synchronize_rcu(), second counter flip. */
-       need_mb = 1; /* mb() A */
+#ifdef MB_A
+       need_mb = 1; /* mb() A (analogous to omitted barrier between E and F) */
        do
-       :: need_mb == 1 -> skip;
-       :: need_mb == 0 -> break;
+       :: need_mb == 1 && !reader_done -> skip;
+       :: need_mb == 0 || reader_done -> break;
        od;
+#endif /* #ifdef MB_A */
        urcu_gp_ctr = urcu_gp_ctr + RCU_GP_CTR_BIT;
-       need_mb = 1; /* mb() B */
+#ifdef MB_B
+       need_mb = 1; /* mb() B (analogous to F) */
        do
-       :: need_mb == 1 -> skip;
-       :: need_mb == 0 -> break;
+       :: need_mb == 1 && !reader_done -> skip;
+       :: need_mb == 0 || reader_done -> break;
        od;
+#endif /* #ifdef MB_B */
        do
        :: 1 ->
                if
@@ -205,27 +245,43 @@ proctype urcu_updater()
                :: else -> break;
                fi
        od;
+#ifdef MB_C
        need_mb = 1; /* mb() C absolutely required by analogy with G */
        do
-       :: need_mb == 1 -> skip;
-       :: need_mb == 0 -> break;
+       :: need_mb == 1 && !reader_done -> skip;
+       :: need_mb == 0 || reader_done -> break;
        od;
+#endif /* #ifdef MB_C */
 
        /* Removal statement, e.g., list_del_rcu(). */
        removed = 1;
 
+       /*
+        * prior synchronize_rcu().
+        * ----------------------------------------------------------------
+        */
+
+       /*
+        * ----------------------------------------------------------------
+        * current synchronize_rcu().
+        */
+
        /* current synchronize_rcu(), first counter flip. */
+#ifdef MB_D
        need_mb = 1; /* mb() D suggested */
        do
-       :: need_mb == 1 -> skip;
-       :: need_mb == 0 -> break;
+       :: need_mb == 1 && !reader_done -> skip;
+       :: need_mb == 0 || reader_done -> break;
        od;
+#endif /* #ifdef MB_D */
        urcu_gp_ctr = urcu_gp_ctr + RCU_GP_CTR_BIT;
+#ifdef MB_E
        need_mb = 1;  /* mb() E required if D not present */
        do
-       :: need_mb == 1 -> skip;
-       :: need_mb == 0 -> break;
+       :: need_mb == 1 && !reader_done -> skip;
+       :: need_mb == 0 || reader_done -> break;
        od;
+#endif /* #ifdef MB_E */
 
        /* current synchronize_rcu(), first-flip check plus second flip. */
        if
@@ -257,11 +313,13 @@ proctype urcu_updater()
        fi;
 
        /* current synchronize_rcu(), second counter flip check. */
+#ifdef MB_F
        need_mb = 1; /* mb() F not required */
        do
-       :: need_mb == 1 -> skip;
-       :: need_mb == 0 -> break;
+       :: need_mb == 1 && !reader_done -> skip;
+       :: need_mb == 0 || reader_done -> break;
        od;
+#endif /* #ifdef MB_F */
        do
        :: 1 ->
                if
@@ -272,14 +330,32 @@ proctype urcu_updater()
                :: else -> break;
                fi;
        od;
+#ifdef MB_G
        need_mb = 1; /* mb() G absolutely required */
        do
-       :: need_mb == 1 -> skip;
-       :: need_mb == 0 -> break;
+       :: need_mb == 1 && !reader_done -> skip;
+       :: need_mb == 0 || reader_done -> break;
        od;
+#endif /* #ifdef MB_G */
+
+       /*
+        * current synchronize_rcu().
+        * ----------------------------------------------------------------
+        */
 
        /* free-up step, e.g., kfree(). */
        free = 1;
+
+       /* We are done, so kill all the infinite loops in the reader. */
+       updater_done = 1;
+
+       /* Create a progress cycle.  Correctness requires we get here. */
+       do
+       :: 1 ->
+progress_updater:
+               skip;
+       :: 1 -> break;
+       od;
 }
 
 /*
index 8ef0e871f2cd8ef4cc3b9de59bfc334a1870ea02..0c5e492ad9da008e812beeae5d915a9cdb5f65d6 100644 (file)
@@ -1,3 +1,3 @@
 spin -a urcu-paulmck.spin 
-cc -DSAFETY -o pan pan.c
-./pan
+cc -DNP -o pan pan.c
+./pan -f -l
This page took 0.028229 seconds and 4 git commands to generate.