| 1 | extern void __VERIFIER_error() __attribute__ ((__noreturn__));
|
|---|
| 2 |
|
|---|
| 3 | /* Testcase from Threader's distribution. For details see:
|
|---|
| 4 | http://www.model.in.tum.de/~popeea/research/threader
|
|---|
| 5 |
|
|---|
| 6 | This file is adapted from the example introduced in the paper:
|
|---|
| 7 | Thread-Modular Verification for Shared-Memory Programs
|
|---|
| 8 | by Cormac Flanagan, Stephen Freund, Shaz Qadeer.
|
|---|
| 9 | */
|
|---|
| 10 |
|
|---|
| 11 | #include <pthread.h>
|
|---|
| 12 | #define assert(e) if (!(e)) ERROR: __VERIFIER_error();
|
|---|
| 13 |
|
|---|
| 14 | int w=0, r=0, x, y;
|
|---|
| 15 |
|
|---|
| 16 | void __VERIFIER_atomic_take_write_lock() {
|
|---|
| 17 | __VERIFIER_assume(w==0 && r==0);
|
|---|
| 18 | w = 1;
|
|---|
| 19 | }
|
|---|
| 20 |
|
|---|
| 21 | void __VERIFIER_atomic_take_read_lock() {
|
|---|
| 22 | __VERIFIER_assume(w==0);
|
|---|
| 23 | r = r+1;
|
|---|
| 24 | }
|
|---|
| 25 |
|
|---|
| 26 | void *writer() { //writer
|
|---|
| 27 | __VERIFIER_atomic_take_write_lock();
|
|---|
| 28 | x = 3;
|
|---|
| 29 | w = 0;
|
|---|
| 30 | }
|
|---|
| 31 |
|
|---|
| 32 | void *reader() { //reader
|
|---|
| 33 | int l;
|
|---|
| 34 | __VERIFIER_atomic_take_read_lock();
|
|---|
| 35 | l = x;
|
|---|
| 36 | y = l;
|
|---|
| 37 | assert(y == x);
|
|---|
| 38 | l = r-1;
|
|---|
| 39 | r = l;
|
|---|
| 40 | }
|
|---|
| 41 |
|
|---|
| 42 | int main() {
|
|---|
| 43 | pthread_t t1, t2, t3, t4;
|
|---|
| 44 | pthread_create(&t1, 0, writer, 0);
|
|---|
| 45 | pthread_create(&t2, 0, reader, 0);
|
|---|
| 46 | pthread_create(&t3, 0, writer, 0);
|
|---|
| 47 | pthread_create(&t4, 0, reader, 0);
|
|---|
| 48 | pthread_join(t1, 0);
|
|---|
| 49 | pthread_join(t2, 0);
|
|---|
| 50 | pthread_join(t3, 0);
|
|---|
| 51 | pthread_join(t4, 0);
|
|---|
| 52 | return 0;
|
|---|
| 53 | }
|
|---|