File: with_lock.c

package info (click to toggle)
cbmc 6.6.0-4
  • links: PTS
  • area: main
  • in suites: forky, sid, trixie
  • size: 153,852 kB
  • sloc: cpp: 386,459; ansic: 114,466; java: 28,405; python: 6,003; yacc: 4,552; makefile: 4,041; lex: 2,487; xml: 2,388; sh: 2,050; perl: 557; pascal: 184; javascript: 163; ada: 36
file content (120 lines) | stat: -rw-r--r-- 3,468 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
#include <assert.h>
#include <pthread.h>
#include <sched.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>

struct sharedt
{
  pthread_rwlock_t lock;
  int a;
  int b;
  int c;
};

void set_state1(struct sharedt *shared)
{
  shared->a = 1;
  shared->b = 2;
  // The following call to yield is here in order to increase the chance of
  // thread swaps during concrete execution in order to show unsoundness.
  sched_yield();
  shared->c = 3;
}

void set_state2(struct sharedt *shared)
{
  shared->a = 4;
  shared->b = 5;
  // The following call to yield is here in order to increase the chance of
  // thread swaps during concrete execution in order to show unsoundness.
  sched_yield();
  shared->c = 6;
}

void *writer1(void *arguments)
{
  struct sharedt *shared = (struct sharedt *)arguments;
  for(int i = 0; i < 1000; ++i)
  {
    const int lock_error = pthread_rwlock_wrlock(&(shared->lock));
    assert(!lock_error);
    set_state1(shared);
    const int unlock_error = pthread_rwlock_unlock(&(shared->lock));
    assert(!unlock_error);
  }
  pthread_exit(NULL);
}

void *writer2(void *arguments)
{
  struct sharedt *shared = (struct sharedt *)arguments;
  for(int i = 0; i < 1000; ++i)
  {
    const int lock_error = pthread_rwlock_wrlock(&(shared->lock));
    assert(!lock_error);
    set_state2(shared);
    const int unlock_error = pthread_rwlock_unlock(&(shared->lock));
    assert(!unlock_error);
  }
  pthread_exit(NULL);
}

void *checker(void *arguments)
{
  struct sharedt *shared = (struct sharedt *)arguments;
  for(int i = 0; i < 1000; ++i)
  {
    const int lock_error = pthread_rwlock_rdlock(&(shared->lock));
    assert(!lock_error);
    const bool is_state1 = shared->a == 1 && shared->b == 2 && shared->c == 3;
    // The following call to yield is here in order to increase the chance of
    // thread swaps during concrete execution in order to show unsoundness.
    sched_yield();
    const bool is_state2 = shared->a == 4 && shared->b == 5 && shared->c == 6;
    assert(is_state1 != is_state2);
    const int unlock_error = pthread_rwlock_unlock(&(shared->lock));
    assert(!unlock_error);
    printf(is_state1 ? "State1\n" : "State2\n");
  }
  pthread_exit(NULL);
}

int main(void)
{
  struct sharedt shared;
  set_state1(&shared);
  const pthread_rwlockattr_t *init_attributes = NULL;
  const int init_error = pthread_rwlock_init(&(shared.lock), init_attributes);
  assert(!init_error);

  const pthread_attr_t *const attributes = NULL;
  void *const thread_argument = &shared;
  pthread_t thread_writer1;
  assert(
    !pthread_create(&thread_writer1, attributes, &writer1, thread_argument));
  pthread_t thread_writer2;
  assert(
    !pthread_create(&thread_writer2, attributes, &writer2, thread_argument));

  pthread_t thread_checker1;
  pthread_t thread_checker2;
  pthread_t thread_checker3;
  assert(
    !pthread_create(&thread_checker1, attributes, &checker, thread_argument));
  assert(
    !pthread_create(&thread_checker2, attributes, &checker, thread_argument));
  assert(
    !pthread_create(&thread_checker3, attributes, &checker, thread_argument));

  assert(!pthread_join(thread_writer1, NULL));
  assert(!pthread_join(thread_writer2, NULL));
  assert(!pthread_join(thread_checker1, NULL));
  assert(!pthread_join(thread_checker2, NULL));
  assert(!pthread_join(thread_checker3, NULL));

  const int destroy_error = pthread_rwlock_destroy(&(shared.lock));
  assert(!destroy_error);
  return EXIT_SUCCESS;
}