// This file is part of the SV-Benchmarks collection of verification tasks: // https://gitlab.com/sosy-lab/benchmarking/sv-benchmarks // // SPDX-FileCopyrightText: 2005-2021 University of Tartu & Technische Universität München // // SPDX-License-Identifier: MIT #include #include "racemacros.h" struct s { int datum; struct s *next; } *A, *B; void init (struct s *p) { p -> datum = 0; p -> next = NULL; } // Naive implementation of synchronized list library: pthread_mutex_t list_mutex = PTHREAD_MUTEX_INITIALIZER; struct s *take(struct s *list) { pthread_mutex_lock(&list_mutex); struct s *p = list; while (p -> next != NULL && __VERIFIER_nondet_int()) p = p -> next; pthread_mutex_unlock(&list_mutex); return p; } void insert(struct s *node, struct s *list) { pthread_mutex_lock(&list_mutex); struct s *t = list->next; list->next = node; node->next = t; pthread_mutex_unlock(&list_mutex); } pthread_mutex_t A_mutex = PTHREAD_MUTEX_INITIALIZER; pthread_mutex_t B_mutex = PTHREAD_MUTEX_INITIALIZER; void *t1_fun(void *arg) { struct s *p = malloc(sizeof(struct s)); init(p); struct s *list = __VERIFIER_nondet_int() ? A : B; insert(p, list); return NULL; } void *t2_fun(void *arg) { struct s *p = take(A); pthread_mutex_lock(&A_mutex); access_or_assert_racefree(p->datum); // TODO pthread_mutex_unlock(&A_mutex); return NULL; } void *t3_fun(void *arg) { struct s *p = take(B); pthread_mutex_lock(&B_mutex); access_or_assert_racefree(p->datum); // TODO pthread_mutex_unlock(&B_mutex); return NULL; } int main () { A = malloc(sizeof(struct s)); init(A); B = malloc(sizeof(struct s)); init(B); create_threads(t1); create_threads(t2); create_threads(t3); join_threads(t1); join_threads(t2); join_threads(t3); return 0; }