// 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

extern int __VERIFIER_nondet_int();
extern void abort(void);
void assume_abort_if_not(int cond) {
  if(!cond) {abort();}
}

#include<pthread.h>
#include<stdlib.h>
#include<stdio.h>

struct s {
  int datum;
  struct s *next;
};

struct cache {
  struct s       *slots[10];
  pthread_mutex_t mutex[10];
} c;

struct s *new(int x) {
  struct s *p = malloc(sizeof(struct s));
  p->datum = x;
  p->next = NULL;
  return p;
}

void list_add(struct s *node, struct s *list) {
  struct s *temp = list->next;
  list->next = node;
  node->next = temp;
}

void *t_fun(void *arg) {
  int i = __VERIFIER_nondet_int();
  assume_abort_if_not(0 <= i && i < 10);
  pthread_mutex_lock(&c.mutex[i]);
  list_add(new(3), c.slots[i]);
  pthread_mutex_unlock(&c.mutex[i]);
  return NULL;
}

int main () {
  for (int i = 0; i < 10; i++)
    pthread_mutex_init(&c.mutex[i], NULL);

  int j = __VERIFIER_nondet_int();
  assume_abort_if_not(0 <= j && j < 10);
  struct s *p;
  pthread_t t1;

  c.slots[j] = new(1);
  list_add(new(2), c.slots[j]);

  pthread_create(&t1, NULL, t_fun, NULL);

  pthread_mutex_lock(&c.mutex[j]);
  p = c.slots[j]->next; // NORACE
  printf("%d\n", p->datum);
  pthread_mutex_unlock(&c.mutex[j]);
  return 0;
}
