CBMC
shared_buffers.cpp
Go to the documentation of this file.
1 /*******************************************************************\
2 
3 Module:
4 
5 Author: Daniel Kroening, kroening@kroening.com
6 
7 \*******************************************************************/
8 
9 #include "shared_buffers.h"
10 
11 #include <util/c_types.h>
12 #include <util/fresh_symbol.h>
13 #include <util/message.h>
14 #include <util/pointer_expr.h>
15 #include <util/std_code.h>
16 
18 
19 #include <goto-instrument/rw_set.h>
20 
21 #include "fence.h"
22 
24 std::string shared_bufferst::unique(void)
25 {
27  return "$fresh#"+std::to_string(uniq++);
28 }
29 
32  const irep_idt &object)
33 {
34  var_mapt::const_iterator it=var_map.find(object);
35  if(it!=var_map.end())
36  return it->second;
37 
38  varst &vars=var_map[object];
39 
41  const symbolt &symbol=ns.lookup(object);
42 
43  vars.type=symbol.type;
44 
45  vars.w_buff0 = add(symbol, "$w_buff0", symbol.type);
46  vars.w_buff1 = add(symbol, "$w_buff1", symbol.type);
47 
48  vars.w_buff0_used = add(symbol, "$w_buff0_used", bool_typet());
49  vars.w_buff1_used = add(symbol, "$w_buff1_used", bool_typet());
50 
51  vars.mem_tmp = add(symbol, "$mem_tmp", symbol.type);
52  vars.flush_delayed = add(symbol, "$flush_delayed", bool_typet());
53 
54  vars.read_delayed = add(symbol, "$read_delayed", bool_typet());
55  vars.read_delayed_var =
56  add(symbol, "$read_delayed_var", pointer_type(symbol.type));
57 
58  for(unsigned cnt=0; cnt<nb_threads; cnt++)
59  {
60  vars.r_buff0_thds.push_back(shared_bufferst::add(
61  symbol, "$r_buff0_thd" + std::to_string(cnt), bool_typet()));
62  vars.r_buff1_thds.push_back(shared_bufferst::add(
63  symbol, "$r_buff1_thd" + std::to_string(cnt), bool_typet()));
64  }
65 
66  return vars;
67 }
68 
73  const symbolt &object,
74  const std::string &suffix,
75  const typet &type,
76  bool instrument)
77 {
78  const namespacet ns(symbol_table);
79 
80  symbolt &new_symbol = get_fresh_aux_symbol(
81  type,
82  id2string(object.name) + suffix,
83  id2string(object.base_name) + suffix,
84  object.location,
85  object.mode,
86  symbol_table);
87  new_symbol.is_static_lifetime=true;
88  new_symbol.value.make_nil();
89 
90  if(instrument)
91  instrumentations.insert(new_symbol.name);
92 
93  return new_symbol.name;
94 }
95 
97 {
98  goto_programt::targett t=goto_program.instructions.begin();
99  const namespacet ns(symbol_table);
100 
101  for(const auto &vars : var_map)
102  {
103  source_locationt source_location;
104  source_location.make_nil();
105 
106  assignment(goto_program, t, source_location, vars.second.w_buff0_used,
107  false_exprt());
108  assignment(goto_program, t, source_location, vars.second.w_buff1_used,
109  false_exprt());
110  assignment(goto_program, t, source_location, vars.second.flush_delayed,
111  false_exprt());
112  assignment(goto_program, t, source_location, vars.second.read_delayed,
113  false_exprt());
114  assignment(goto_program, t, source_location, vars.second.read_delayed_var,
115  null_pointer_exprt(pointer_type(vars.second.type)));
116 
117  for(const auto &id : vars.second.r_buff0_thds)
118  assignment(goto_program, t, source_location, id, false_exprt());
119 
120  for(const auto &id : vars.second.r_buff1_thds)
121  assignment(goto_program, t, source_location, id, false_exprt());
122  }
123 }
124 
126  goto_functionst &goto_functions)
127 {
128  // get "main"
129  goto_functionst::function_mapt::iterator
130  m_it=goto_functions.function_map.find(goto_functions.entry_point());
131 
132  if(m_it==goto_functions.function_map.end())
133  throw "weak memory instrumentation needs an entry point";
134 
135  goto_programt &main=m_it->second.body;
137 }
138 
141  goto_programt &goto_program,
143  const source_locationt &source_location,
144  const irep_idt &id_lhs,
145  const exprt &value)
146 {
147  const namespacet ns(symbol_table);
148  std::string identifier=id2string(id_lhs);
149 
150  const size_t pos=identifier.find("[]");
151 
152  if(pos!=std::string::npos)
153  {
154  /* we don't distinguish the members of an array for the moment */
155  identifier.erase(pos);
156  }
157 
158  try
159  {
160  const exprt symbol=ns.lookup(identifier).symbol_expr();
161 
162  t = goto_program.insert_before(
163  t, goto_programt::make_assignment(symbol, value, source_location));
164 
165  // instrumentations.insert((const irep_idt) (t->code.id()));
166 
167  t++;
168  }
169  catch(const std::string &s)
170  {
171  message.warning() << s << messaget::eom;
172  }
173 }
174 
177  goto_programt &goto_program,
178  goto_programt::targett &target,
179  const source_locationt &source_location,
180  const irep_idt &read_object,
181  const irep_idt &write_object)
182 {
183 /* option 1: */
184 /* trick using an additional variable whose value is to be defined later */
185 
186 #if 0
187  assignment(goto_program, target, source_location, vars.read_delayed,
188  true_exprt());
189  assignment(goto_program, target, source_location, vars.read_delayed_var,
190  read_object);
191 
192  const irep_idt &new_var=add_fresh_var(write_object, unique(), vars.type);
193 
194  assignment(goto_program, target, source_location, vars.read_new_var, new_var);
195 
196  // initial write, but from the new variable now
197  assignment(goto_program, target, source_location, write_object, new_var);
198 #endif
199 
200 /* option 2 */
201 /* pointer */
202 
203  const std::string identifier=id2string(write_object);
204 
205  message.debug()<<"delay_read: " << messaget::eom;
206  const varst &vars=(*this)(write_object);
207 
208  const symbol_exprt read_object_expr=symbol_exprt(read_object, vars.type);
209 
210  assignment(
211  goto_program,
212  target,
213  source_location,
214  vars.read_delayed,
215  true_exprt());
216  assignment(
217  goto_program,
218  target,
219  source_location,
220  vars.read_delayed_var,
221  address_of_exprt(read_object_expr));
222 }
223 
226  goto_programt &goto_program,
227  goto_programt::targett &target,
228  const source_locationt &source_location,
229  const irep_idt &write_object)
230 {
231 #if 0
232  // option 1
233  const varst &vars=(*this)(write_object);
234 
235  const symbol_exprt fresh_var_expr=symbol_exprt(vars.read_new_var, vars.type);
236  const symbol_exprt var_expr=symbol_exprt(vars.read_delayed_var, vars.type);
237  const exprt eq_expr=equal_exprt(var_expr, fresh_var_expr);
238 
239  const symbol_exprt cond_delayed_expr=symbol_exprt(vars.read_delayed,
240  bool_typet());
241  const exprt if_expr=if_exprt(cond_delayed_expr, eq_expr, true_exprt());
242 
243  target=goto_program.insert_before(target);
244  target->type=ASSUME;
245  target->guard=if_expr;
246  target->guard.source_location()=source_location;
247  target->source_location=source_location;
248 
249  target++;
250 
251  assignment(goto_program, target, source_location, vars.read_delayed,
252  false_exprt());
253 #else
254  // option 2: do nothing
255  // unused parameters
256  (void)goto_program;
257  (void)target;
258  (void)target;
259  (void)source_location;
260  (void)write_object;
261 #endif
262 }
263 
266  goto_programt &goto_program,
267  goto_programt::targett &target,
268  const source_locationt &source_location,
269  const irep_idt &object,
270  goto_programt::instructiont &original_instruction,
271  const unsigned current_thread)
272 {
273  const std::string identifier=id2string(object);
274 
275  message.debug() << "write: " << object << messaget::eom;
276  const varst &vars=(*this)(object);
277 
278  // We rotate the write buffers for anything that is written.
279  assignment(goto_program, target, source_location, vars.w_buff1, vars.w_buff0);
280  assignment(
281  goto_program,
282  target,
283  source_location,
284  vars.w_buff0,
285  original_instruction.code().op1());
286 
287  // We update the used flags
288  assignment(
289  goto_program,
290  target,
291  source_location,
292  vars.w_buff1_used,
293  vars.w_buff0_used);
294  assignment(
295  goto_program,
296  target,
297  source_location,
298  vars.w_buff0_used,
299  true_exprt());
300 
301  // We should not exceed the buffer size -- inserts assertion for dynamically
302  // checking this
303  const exprt buff0_used_expr=symbol_exprt(vars.w_buff0_used, bool_typet());
304  const exprt buff1_used_expr=symbol_exprt(vars.w_buff1_used, bool_typet());
305  const exprt cond_expr=
306  not_exprt(and_exprt(buff1_used_expr, buff0_used_expr));
307 
308  target = goto_program.insert_before(
309  target, goto_programt::make_assertion(cond_expr, source_location));
310  target++;
311 
312  // We update writers ownership of the values in the buffer
313  for(unsigned cnt=0; cnt<nb_threads; cnt++)
314  assignment(goto_program, target, source_location, vars.r_buff1_thds[cnt],
315  vars.r_buff0_thds[cnt]);
316 
317  // We update the lucky new author of this value in the buffer
318  assignment(
319  goto_program,
320  target,
321  source_location,
322  vars.r_buff0_thds[current_thread],
323  true_exprt());
324 }
325 
328  goto_programt &goto_program,
329  goto_programt::targett &target,
330  const source_locationt &source_location,
331  const irep_idt &object,
332  const unsigned current_thread)
333 {
334  const std::string identifier=id2string(object);
335 
336  // mostly for instrumenting the fences. A thread only flushes the values it
337  // wrote in the buffer.
338  message.debug() << "det flush: " << messaget::eom;
339  const varst &vars=(*this)(object);
340 
341  // current value in the memory
342  const exprt lhs=symbol_exprt(object, vars.type);
343 
344  // if buff0 from this thread, uses it to update the memory (the most recent
345  // value, or last write by -ws-> ); if not, if buff1 from this thread, uses
346  // it; if not, keeps the current memory value
347  const exprt buff0_expr=symbol_exprt(vars.w_buff0, vars.type);
348  const exprt buff1_expr=symbol_exprt(vars.w_buff1, vars.type);
349 
350  const exprt buff0_used_expr=symbol_exprt(vars.w_buff0_used, bool_typet());
351  const exprt buff1_used_expr=symbol_exprt(vars.w_buff1_used, bool_typet());
352 
353  const exprt buff0_mine_expr=symbol_exprt(vars.r_buff0_thds[current_thread],
354  bool_typet());
355  const exprt buff1_mine_expr=symbol_exprt(vars.r_buff1_thds[current_thread],
356  bool_typet());
357 
358  const exprt buff0_used_and_mine_expr=and_exprt(buff0_used_expr,
359  buff0_mine_expr);
360  const exprt buff1_used_and_mine_expr=and_exprt(buff1_used_expr,
361  buff1_mine_expr);
362 
363  const exprt new_value_expr=
364  if_exprt(
365  buff0_used_and_mine_expr,
366  buff0_expr,
367  if_exprt(
368  buff1_used_and_mine_expr,
369  buff1_expr,
370  lhs));
371 
372  // We update (or not) the value in the memory
373  assignment(goto_program, target, source_location, object, new_value_expr);
374 
375  // We update the flags of the buffer
376  // if buff0 used and mine, then it is no more used, as we flushed the last
377  // write and -ws-> imposes not to have other writes in the buffer
378  assignment(
379  goto_program,
380  target,
381  source_location,
382  vars.w_buff0_used,
383  if_exprt(
384  buff0_used_and_mine_expr,
385  false_exprt(),
386  buff0_used_expr));
387 
388  // buff1 used and mine & not buff0 used and mine, then it no more used
389  // if buff0 is used and mine, then, by ws, buff1 is no more used
390  // otherwise, remains as it is
391  const exprt buff0_or_buff1_used_and_mine_expr=
392  or_exprt(buff0_used_and_mine_expr, buff1_used_and_mine_expr);
393 
394  assignment(
395  goto_program,
396  target,
397  source_location,
398  vars.w_buff1_used,
399  if_exprt(
400  buff0_or_buff1_used_and_mine_expr,
401  false_exprt(),
402  buff1_used_expr));
403 
404  // We update the ownerships
405  // if buff0 mine and used, flushed, so belongs to nobody
406  const exprt buff0_thd_expr=
407  symbol_exprt(vars.r_buff0_thds[current_thread], bool_typet());
408 
409  assignment(
410  goto_program,
411  target,
412  source_location,
413  vars.r_buff0_thds[current_thread],
414  if_exprt(buff0_used_and_mine_expr, false_exprt(), buff0_thd_expr));
415 
416  // if buff1 used and mine, or if buff0 used and mine, then buff1 flushed and
417  // doesn't belong to anybody
418  const exprt buff1_thd_expr=
419  symbol_exprt(vars.r_buff1_thds[current_thread], bool_typet());
420 
421  assignment(
422  goto_program,
423  target,
424  source_location,
425  vars.r_buff1_thds[current_thread],
426  if_exprt(
427  buff0_or_buff1_used_and_mine_expr,
428  false_exprt(),
429  buff1_thd_expr));
430 }
431 
434  const irep_idt &function_id,
435  goto_programt &goto_program,
436  goto_programt::targett &target,
437  const source_locationt &source_location,
438  const irep_idt &object,
439  const unsigned current_thread,
440  const bool tso_pso_rmo) // true: tso/pso/rmo; false: power
441 {
442  const std::string identifier=id2string(object);
443 
444  message.debug() << "nondet flush: " << object << messaget::eom;
445 
446  try
447  {
448  const varst &vars=(*this)(object);
449 
450  // Non deterministic choice
451  irep_idt choice0 = choice(function_id, "0");
452  irep_idt choice2 = choice(function_id, "2"); // delays the write flush
453 
454  const symbol_exprt choice0_expr=symbol_exprt(choice0, bool_typet());
455  const symbol_exprt delay_expr=symbol_exprt(choice2, bool_typet());
456  const exprt nondet_bool_expr =
457  side_effect_expr_nondett(bool_typet(), source_location);
458 
459  // throw Boolean dice
460  assignment(goto_program, target, source_location, choice0, nondet_bool_expr);
461  assignment(goto_program, target, source_location, choice2, nondet_bool_expr);
462 
463  // Buffers and memory
464  const symbol_exprt buff0_expr=symbol_exprt(vars.w_buff0, vars.type);
465  const symbol_exprt buff1_expr=symbol_exprt(vars.w_buff1, vars.type);
466  const exprt lhs=symbol_exprt(object, vars.type);
467 
468  // Buffer uses
469  const symbol_exprt buff0_used_expr=symbol_exprt(vars.w_buff0_used,
470  bool_typet());
471  const symbol_exprt buff1_used_expr=symbol_exprt(vars.w_buff1_used,
472  bool_typet());
473 
474  // Buffer ownerships
475  const symbol_exprt buff0_thd_expr=
476  symbol_exprt(vars.r_buff0_thds[current_thread], bool_typet());
477  const symbol_exprt buff1_thd_expr=
478  symbol_exprt(vars.r_buff1_thds[current_thread], bool_typet());
479 
480 
481  // Will the write be directly flushed, or is it just a read?
482  assignment(
483  goto_program, target, source_location, vars.flush_delayed, delay_expr);
484  assignment(goto_program, target, source_location, vars.mem_tmp, lhs);
485 
486  // for POWER, only instrumented reads can read from the buffers of other
487  // threads
488  bool instrumented=false;
489 
490  if(!tso_pso_rmo)
491  {
492  if(cycles.find(object)!=cycles.end())
493  {
494  typedef std::multimap<irep_idt, source_locationt>::iterator m_itt;
495  std::pair<m_itt, m_itt> ran=cycles_loc.equal_range(object);
496  for(m_itt ran_it=ran.first; ran_it!=ran.second; ran_it++)
497  if(ran_it->second==source_location)
498  {
499  instrumented=true;
500  break;
501  }
502  }
503  }
504 
505  // TSO/PSO/RMO
506  if(tso_pso_rmo || !instrumented)
507  {
508  // 7 cases
509 
510  // (1) (3) (4)
511  // if buff0 unused
512  // or buff0 not mine and buff1 unused
513  // or buff0 not mine and buff1 not mine
514  // -> read from memory (and does not modify the buffer in any aspect)
515  const exprt cond_134_expr=
516  or_exprt(
517  not_exprt(buff0_used_expr),
518  or_exprt(
519  and_exprt(
520  not_exprt(buff0_thd_expr),
521  not_exprt(buff1_used_expr)),
522  and_exprt(
523  not_exprt(buff0_thd_expr),
524  not_exprt(buff1_thd_expr))));
525  const exprt val_134_expr=lhs;
526  const exprt buff0_used_134_expr=buff0_used_expr;
527  const exprt buff1_used_134_expr=buff1_used_expr;
528  const exprt buff0_134_expr=buff0_expr;
529  const exprt buff1_134_expr=buff1_expr;
530  const exprt buff0_thd_134_expr=buff0_thd_expr;
531  const exprt buff1_thd_134_expr=buff1_thd_expr;
532 
533  // (2) (6) (7)
534  // if buff0 used and mine
535  // -> read from buff0
536  const exprt cond_267_expr=and_exprt(buff0_used_expr, buff0_thd_expr);
537  const exprt val_267_expr=buff0_expr;
538  const exprt buff0_used_267_expr=false_exprt();
539  const exprt buff1_used_267_expr=false_exprt();
540  const exprt buff0_267_expr=buff0_expr;
541  const exprt buff1_267_expr=buff1_expr;
542  const exprt buff0_thd_267_expr=false_exprt();
543  const exprt buff1_thd_267_expr=false_exprt();
544 
545  // (5)
546  // buff0 and buff1 are used, buff0 is not mine, buff1 is mine
547  // -> read from buff1
548  const exprt cond_5_expr=
549  and_exprt(
550  buff0_used_expr,
551  and_exprt(
552  buff1_used_expr,
553  and_exprt(not_exprt(buff0_thd_expr), buff1_thd_expr)));
554  const exprt val_5_expr=buff1_expr;
555  const exprt buff0_used_5_expr=buff0_used_expr;
556  const exprt buff1_used_5_expr=false_exprt();
557  const exprt buff0_5_expr=buff0_expr;
558  const exprt buff1_5_expr=buff1_expr;
559  const exprt buff0_thd_5_expr=buff0_thd_expr;
560  const exprt buff1_thd_5_expr=false_exprt();
561 
562  // Updates
563  // memory
564  assignment(
565  goto_program,
566  target,
567  source_location,
568  object,
569  if_exprt(
570  cond_134_expr,
571  val_134_expr,
572  if_exprt(
573  cond_267_expr,
574  val_267_expr,
575  val_5_expr)));
576  // buff0
577  assignment(
578  goto_program,
579  target,
580  source_location,
581  vars.w_buff0,
582  if_exprt(
583  delay_expr,
584  buff0_expr,
585  if_exprt(
586  cond_134_expr,
587  buff0_134_expr,
588  if_exprt(
589  cond_267_expr,
590  buff0_267_expr,
591  buff0_5_expr))));
592  // buff1
593  assignment(
594  goto_program,
595  target,
596  source_location,
597  vars.w_buff1,
598  if_exprt(
599  delay_expr,
600  buff1_expr,
601  if_exprt(
602  cond_134_expr,
603  buff1_134_expr,
604  if_exprt(
605  cond_267_expr,
606  buff1_267_expr,
607  buff1_5_expr))));
608  // buff0_used
609  assignment(
610  goto_program,
611  target,
612  source_location,
613  vars.w_buff0_used,
614  if_exprt(
615  delay_expr,
616  buff0_used_expr,
617  if_exprt(
618  cond_134_expr,
619  buff0_used_134_expr,
620  if_exprt(
621  cond_267_expr,
622  buff0_used_267_expr,
623  buff0_used_5_expr))));
624  // buff1_used
625  assignment(
626  goto_program,
627  target,
628  source_location,
629  vars.w_buff1_used,
630  if_exprt(
631  delay_expr,
632  buff1_used_expr,
633  if_exprt(
634  cond_134_expr,
635  buff1_used_134_expr,
636  if_exprt(
637  cond_267_expr,
638  buff1_used_267_expr,
639  buff1_used_5_expr))));
640  // buff0_thd
641  assignment(
642  goto_program,
643  target,
644  source_location,
645  vars.r_buff0_thds[current_thread],
646  if_exprt(
647  delay_expr,
648  buff0_thd_expr,
649  if_exprt(
650  cond_134_expr,
651  buff0_thd_134_expr,
652  if_exprt(
653  cond_267_expr,
654  buff0_thd_267_expr,
655  buff0_thd_5_expr))));
656  // buff1_thd
657  assignment(
658  goto_program,
659  target,
660  source_location,
661  vars.r_buff1_thds[current_thread], if_exprt(
662  delay_expr,
663  buff1_thd_expr,
664  if_exprt(
665  cond_134_expr,
666  buff1_thd_134_expr,
667  if_exprt(
668  cond_267_expr,
669  buff1_thd_267_expr,
670  buff1_thd_5_expr))));
671  }
672  // POWER
673  else
674  {
675  // a thread can read the other threads' buffers
676 
677  // One extra non-deterministic choice needed
678  irep_idt choice1 = choice(function_id, "1");
679  const symbol_exprt choice1_expr=symbol_exprt(choice1, bool_typet());
680 
681  // throw Boolean dice
682  assignment(
683  goto_program, target, source_location, choice1, nondet_bool_expr);
684 
685  // 7 cases
686 
687  // (1)
688  // if buff0 unused
689  // -> read from memory (and does not modify the buffer in any aspect)
690  const exprt cond_1_expr=not_exprt(buff0_used_expr);
691  const exprt val_1_expr=lhs;
692  const exprt buff0_used_1_expr=buff0_used_expr;
693  const exprt buff1_used_1_expr=buff1_used_expr;
694  const exprt buff0_1_expr=buff0_expr;
695  const exprt buff1_1_expr=buff1_expr;
696  const exprt buff0_thd_1_expr=buff0_thd_expr;
697  const exprt buff1_thd_1_expr=buff1_thd_expr;
698 
699  // (2) (6) (7)
700  // if buff0 used and mine
701  // -> read from buff0
702  const exprt cond_267_expr=
703  and_exprt(buff0_used_expr, buff0_thd_expr);
704  const exprt val_267_expr=buff0_expr;
705  const exprt buff0_used_267_expr=false_exprt();
706  const exprt buff1_used_267_expr=false_exprt();
707  const exprt buff0_267_expr=buff0_expr;
708  const exprt buff1_267_expr=buff1_expr;
709  const exprt buff0_thd_267_expr=false_exprt();
710  const exprt buff1_thd_267_expr=false_exprt();
711 
712  // (3)
713  // if buff0 used and not mine, and buff1 not used
714  // -> read from buff0 or memory
715  const exprt cond_3_expr=
716  and_exprt(
717  buff0_used_expr,
718  and_exprt(
719  not_exprt(buff0_thd_expr),
720  not_exprt(buff1_used_expr)));
721  const exprt val_3_expr=if_exprt(choice0_expr, buff0_expr, lhs);
722  const exprt buff0_used_3_expr=choice0_expr;
723  const exprt buff1_used_3_expr=false_exprt();
724  const exprt buff0_3_expr=buff0_expr;
725  const exprt buff1_3_expr=buff1_expr;
726  const exprt buff0_thd_3_expr=false_exprt();
727  const exprt buff1_thd_3_expr=false_exprt();
728 
729  // (4)
730  // buff0 and buff1 are both used, and both not mine
731  // -> read from memory or buff0 or buff1
732  const exprt cond_4_expr=
733  and_exprt(
734  and_exprt(buff0_used_expr, not_exprt(buff1_thd_expr)),
735  and_exprt(buff1_used_expr, not_exprt(buff0_thd_expr)));
736  const exprt val_4_expr=
737  if_exprt(
738  choice0_expr,
739  lhs,
740  if_exprt(
741  choice1_expr,
742  buff0_expr,
743  buff1_expr));
744  const exprt buff0_used_4_expr=
745  or_exprt(choice0_expr, not_exprt(choice1_expr));
746  const exprt buff1_used_4_expr=choice0_expr;
747  const exprt buff0_4_expr=buff0_expr;
748  const exprt buff1_4_expr=buff1_expr;
749  const exprt buff0_thd_4_expr=buff0_thd_expr;
750  const exprt buff1_thd_4_expr=
751  if_exprt(choice0_expr, buff1_thd_expr, false_exprt());
752 
753  // (5)
754  // buff0 and buff1 are both used, and buff0 not mine, and buff1 mine
755  // -> read buff1 or buff0
756  const exprt cond_5_expr=
757  and_exprt(
758  and_exprt(buff0_used_expr, buff1_thd_expr),
759  and_exprt(buff1_used_expr, not_exprt(buff0_thd_expr)));
760  const exprt val_5_expr=
761  if_exprt(
762  choice0_expr,
763  buff1_expr,
764  buff0_expr);
765  const exprt buff0_used_5_expr=choice0_expr;
766  const exprt buff1_used_5_expr=false_exprt();
767  const exprt buff0_5_expr=buff0_expr;
768  const exprt buff1_5_expr=buff1_expr;
769  const exprt buff0_thd_5_expr=false_exprt();
770  const exprt buff1_thd_5_expr=false_exprt();
771 
772  // Updates
773  // memory
774  assignment(
775  goto_program,
776  target,
777  source_location,
778  object,
779  if_exprt(
780  cond_1_expr,
781  val_1_expr,
782  if_exprt(
783  cond_267_expr,
784  val_267_expr,
785  if_exprt(
786  cond_4_expr,
787  val_4_expr,
788  if_exprt(
789  cond_5_expr,
790  val_5_expr,
791  val_3_expr)))));
792  // buff0
793  assignment(
794  goto_program,
795  target,
796  source_location,
797  vars.w_buff0,
798  if_exprt(
799  delay_expr,
800  buff0_expr,
801  if_exprt(
802  cond_1_expr,
803  buff0_1_expr,
804  if_exprt(
805  cond_267_expr,
806  buff0_267_expr,
807  if_exprt(
808  cond_4_expr,
809  buff0_4_expr,
810  if_exprt(
811  cond_5_expr,
812  buff0_5_expr,
813  buff0_3_expr))))));
814  // buff1
815  assignment(
816  goto_program,
817  target,
818  source_location,
819  vars.w_buff1,
820  if_exprt(
821  delay_expr,
822  buff1_expr,
823  if_exprt(
824  cond_1_expr,
825  buff1_1_expr,
826  if_exprt(
827  cond_267_expr,
828  buff1_267_expr,
829  if_exprt(
830  cond_4_expr,
831  buff1_4_expr,
832  if_exprt(
833  cond_5_expr,
834  buff1_5_expr,
835  buff1_3_expr))))));
836  // buff0_used
837  assignment(
838  goto_program,
839  target,
840  source_location,
841  vars.w_buff0_used,
842  if_exprt(
843  delay_expr,
844  buff0_used_expr,
845  if_exprt(
846  cond_1_expr,
847  buff0_used_1_expr,
848  if_exprt(
849  cond_267_expr,
850  buff0_used_267_expr,
851  if_exprt(
852  cond_4_expr,
853  buff0_used_4_expr,
854  if_exprt(
855  cond_5_expr,
856  buff0_used_5_expr,
857  buff0_used_3_expr))))));
858  // buff1_used
859  assignment(
860  goto_program,
861  target,
862  source_location,
863  vars.w_buff1_used,
864  if_exprt(
865  delay_expr,
866  buff1_used_expr,
867  if_exprt(
868  cond_1_expr,
869  buff1_used_1_expr,
870  if_exprt(
871  cond_267_expr,
872  buff1_used_267_expr,
873  if_exprt(
874  cond_4_expr,
875  buff1_used_4_expr,
876  if_exprt(
877  cond_5_expr,
878  buff1_used_5_expr,
879  buff1_used_3_expr))))));
880  // buff0_thd
881  assignment(
882  goto_program,
883  target,
884  source_location,
885  vars.r_buff0_thds[current_thread],
886  if_exprt(
887  delay_expr,
888  buff0_thd_expr,
889  if_exprt(
890  cond_1_expr,
891  buff0_thd_1_expr,
892  if_exprt(
893  cond_267_expr,
894  buff0_thd_267_expr,
895  if_exprt(
896  cond_4_expr,
897  buff0_thd_4_expr,
898  if_exprt(
899  cond_5_expr,
900  buff0_thd_5_expr,
901  buff0_thd_3_expr))))));
902  // buff1_thd
903  assignment(
904  goto_program,
905  target,
906  source_location,
907  vars.r_buff1_thds[current_thread],
908  if_exprt(
909  delay_expr,
910  buff1_thd_expr,
911  if_exprt(
912  cond_1_expr,
913  buff1_thd_1_expr,
914  if_exprt(
915  cond_267_expr,
916  buff1_thd_267_expr,
917  if_exprt(
918  cond_4_expr,
919  buff1_thd_4_expr,
920  if_exprt(
921  cond_5_expr,
922  buff1_thd_5_expr,
923  buff1_thd_3_expr))))));
924  }
925  }
926  catch(const std::string &s)
927  {
928  message.warning() << s << messaget::eom;
929  }
930 }
931 
933  const namespacet &ns,
934  const symbol_exprt &symbol_expr,
935  bool is_write
936  // are we asking for the variable (false), or for the variable and
937  // the source_location in the code (true)
938 )
939 {
940  const irep_idt &identifier=symbol_expr.get_identifier();
941 
942  if(
943  identifier == CPROVER_PREFIX "alloc" ||
944  identifier == CPROVER_PREFIX "alloc_size" || identifier == "stdin" ||
945  identifier == "stdout" || identifier == "stderr" ||
946  identifier == "sys_nerr" || identifier.starts_with("__unbuffered_") ||
947  identifier.starts_with("__CPROVER"))
948  return false; // not buffered
949 
950  const symbolt &symbol=ns.lookup(identifier);
951 
952  if(!symbol.is_static_lifetime)
953  return false; // these are local
954 
955  if(symbol.is_thread_local)
956  return false; // these are local
957 
958  if(instrumentations.find(identifier)!=instrumentations.end())
959  return false; // these are instrumentations
960 
961  return is_buffered_in_general(symbol_expr, is_write);
962 }
963 
965  const symbol_exprt &symbol_expr,
966  bool is_write
967  // are we asking for the variable (false), or for the variable and the
968  // source_location in the code? (true)
969 )
970 {
971  if(cav11)
972  return true;
973 
974  const irep_idt &identifier=symbol_expr.get_identifier();
975  const source_locationt &source_location=symbol_expr.source_location();
976 
977  if(cycles.find(identifier)==cycles.end())
978  return false; // not to instrument
979 
980  if(!is_write)
981  {
982  // to be uncommented only when we are sure all the cycles
983  // are detected (before detection of the pairs -- no hack)
984  // WARNING: on the FULL cycle, not reduced by PO
985  /*typedef std::multimap<irep_idt,source_locationt>::iterator m_itt;
986  std::pair<m_itt,m_itt> ran=cycles_r_loc.equal_range(identifier);
987  for(m_itt ran_it=ran.first; ran_it!=ran.second; ran_it++)
988  if(ran_it->second==source_location)*/
989  return true;
990  }
991  else
992  {
993  typedef std::multimap<irep_idt, source_locationt>::iterator m_itt;
994  std::pair<m_itt, m_itt> ran=cycles_loc.equal_range(identifier);
995  for(m_itt ran_it=ran.first; ran_it!=ran.second; ran_it++)
996  if(ran_it->second==source_location)
997  return true; // not to instrument
998  }
999 
1000  return false;
1001 }
1002 
1007  value_setst &value_sets,
1008  goto_functionst &goto_functions)
1009 {
1011 
1012  for(auto &gf_entry : goto_functions.function_map)
1013  {
1014 #ifdef LOCAL_MAY
1015  local_may_aliast local_may(gf_entry.second);
1016 #endif
1017 
1018  Forall_goto_program_instructions(i_it, gf_entry.second.body)
1019  {
1020  rw_set_loct rw_set(
1021  ns,
1022  value_sets,
1023  gf_entry.first,
1024  i_it,
1025 #ifdef LOCAL_MAY
1026  local_may,
1027 #endif
1028  message.get_message_handler()); // NOLINT(whitespace/parens)
1029  for(const auto &w_entry : rw_set.w_entries)
1030  {
1031  for(const auto &r_entry : rw_set.r_entries)
1032  {
1033  message.debug() << "debug: " << id2string(w_entry.second.object)
1034  << " reads from " << id2string(r_entry.second.object)
1035  << messaget::eom;
1036  if(is_buffered_in_general(r_entry.second.symbol_expr, true))
1037  {
1038  // shouldn't it be true? false => overapprox
1039  affected_by_delay_set.insert(w_entry.second.object);
1040  }
1041  }
1042  }
1043  }
1044  }
1045 }
1046 
1049  value_setst &value_sets,
1050  const irep_idt &function_id,
1051  memory_modelt model)
1052 {
1054  << "visit function " << function_id << messaget::eom;
1055  if(function_id == INITIALIZE_FUNCTION)
1056  return;
1057 
1059  goto_programt &goto_program = goto_functions.function_map[function_id].body;
1060 
1061 #ifdef LOCAL_MAY
1062  local_may_aliast local_may(goto_functions.function_map[function_id]);
1063 #endif
1064 
1065  Forall_goto_program_instructions(i_it, goto_program)
1066  {
1067  goto_programt::instructiont &instruction=*i_it;
1068 
1070  << "instruction " << instruction.type() << messaget::eom;
1071 
1072  /* thread marking */
1073  if(instruction.is_start_thread())
1074  {
1078  }
1079  else if(instruction.is_end_thread())
1081 
1082  if(instruction.is_assign())
1083  {
1084  try
1085  {
1086  rw_set_loct rw_set(
1087  ns,
1088  value_sets,
1089  function_id,
1090  i_it,
1091 #ifdef LOCAL_MAY
1092  local_may,
1093 #endif
1095  .get_message_handler()); // NOLINT(whitespace/parens)
1096 
1097  if(rw_set.empty())
1098  continue;
1099 
1100  // add all the written values (which are not instrumentations)
1101  // in a set
1102  for(const auto &w_entry : rw_set.w_entries)
1103  {
1104  if(shared_buffers.is_buffered(ns, w_entry.second.symbol_expr, false))
1105  past_writes.insert(w_entry.second.object);
1106  }
1107 
1108  goto_programt::instructiont original_instruction;
1109  original_instruction.swap(instruction);
1110  const source_locationt &source_location =
1111  original_instruction.source_location();
1112 
1113  // ATOMIC_BEGIN: we make the whole thing atomic
1114  instruction = goto_programt::make_atomic_begin(source_location);
1115  i_it++;
1116 
1117  // we first perform (non-deterministically) up to 2 writes for
1118  // stuff that is potentially read
1119  for(const auto &r_entry : rw_set.r_entries)
1120  {
1121  // flush read -- do nothing in this implementation
1123  goto_program, i_it, source_location, r_entry.second.object);
1124 
1125  if(shared_buffers.is_buffered(ns, r_entry.second.symbol_expr, false))
1127  function_id,
1128  goto_program,
1129  i_it,
1130  source_location,
1131  r_entry.second.object,
1133  (model == TSO || model == PSO || model == RMO));
1134  }
1135 
1136  // Now perform the write(s).
1137  for(const auto &w_entry : rw_set.w_entries)
1138  {
1139  // if one of the previous read was to buffer, then delays the read
1140  if(model==RMO || model==Power)
1141  {
1142  for(const auto &r_entry : rw_set.r_entries)
1143  {
1145  ns, r_entry.second.symbol_expr, true))
1146  {
1148  goto_program,
1149  i_it,
1150  source_location,
1151  r_entry.second.object,
1152  w_entry.second.object);
1153  }
1154  }
1155  }
1156 
1157  if(shared_buffers.is_buffered(ns, w_entry.second.symbol_expr, true))
1158  {
1160  goto_program,
1161  i_it,
1162  source_location,
1163  w_entry.second.object,
1164  original_instruction,
1165  current_thread);
1166  }
1167  else
1168  {
1169  // unbuffered
1170  if(model==RMO || model==Power)
1171  {
1172  for(const auto &r_entry : rw_set.r_entries)
1173  {
1174  if(
1176  r_entry.second.object) !=
1178  {
1180  << "second: " << r_entry.second.object << messaget::eom;
1181  const varst &vars = (shared_buffers)(r_entry.second.object);
1182 
1184  << "writer " << w_entry.second.object << " reads "
1185  << r_entry.second.object << messaget::eom;
1186 
1187  // TO FIX: how to deal with rhs including calls?
1188  // if a read is delayed, use its alias instead of itself
1189  // -- or not
1190  symbol_exprt to_replace_expr =
1191  symbol_exprt(r_entry.second.object, vars.type);
1192  symbol_exprt new_read_expr=symbol_exprt(
1193  vars.read_delayed_var,
1194  pointer_type(vars.type));
1195  symbol_exprt read_delayed_expr=symbol_exprt(
1196  vars.read_delayed, bool_typet());
1197 
1198  // One extra non-deterministic choice needed
1199  irep_idt choice1 = shared_buffers.choice(function_id, "1");
1200  const symbol_exprt choice1_expr=symbol_exprt(choice1,
1201  bool_typet());
1202  const exprt nondet_bool_expr =
1203  side_effect_expr_nondett(bool_typet(), source_location);
1204 
1205  // throw Boolean dice
1207  goto_program,
1208  i_it,
1209  source_location,
1210  choice1,
1211  nondet_bool_expr);
1212 
1213  const if_exprt rhs(
1214  read_delayed_expr,
1215  if_exprt(
1216  choice1_expr,
1217  dereference_exprt{new_read_expr},
1218  to_replace_expr),
1219  to_replace_expr); // original_instruction.code().op1());
1220 
1222  goto_program,
1223  i_it,
1224  source_location,
1225  r_entry.second.object,
1226  rhs);
1227  }
1228  }
1229  }
1230 
1231  // normal assignment
1233  goto_program,
1234  i_it,
1235  source_location,
1236  w_entry.second.object,
1237  original_instruction.code().op1());
1238  }
1239  }
1240 
1241  // if last writes was flushed to make the lhs reads the buffer but
1242  // without affecting the memory, restore the previous memory value
1243  // (buffer flush delay)
1244  for(const auto &r_entry : rw_set.r_entries)
1245  {
1246  if(shared_buffers.is_buffered(ns, r_entry.second.symbol_expr, false))
1247  {
1249  << "flush restore: " << r_entry.second.object << messaget::eom;
1250  const varst vars = (shared_buffers)(r_entry.second.object);
1251  const exprt delayed_expr=symbol_exprt(vars.flush_delayed,
1252  bool_typet());
1253  const symbol_exprt mem_value_expr=symbol_exprt(vars.mem_tmp,
1254  vars.type);
1255  const exprt cond_expr = if_exprt(
1256  delayed_expr, mem_value_expr, r_entry.second.symbol_expr);
1257 
1259  goto_program,
1260  i_it,
1261  source_location,
1262  r_entry.second.object,
1263  cond_expr);
1265  goto_program, i_it, source_location,
1266  vars.flush_delayed, false_exprt());
1267  }
1268  }
1269 
1270  // ATOMIC_END
1271  i_it = goto_program.insert_before(
1272  i_it, goto_programt::make_atomic_end(source_location));
1273  i_it++;
1274 
1275  i_it--; // the for loop already counts us up
1276  }
1277  catch (...)
1278  {
1279  shared_buffers.message.warning() << "Identifier not found"
1280  << messaget::eom;
1281  }
1282  }
1283  else if(
1284  is_fence(instruction, ns) ||
1285  (instruction.is_other() &&
1286  instruction.code().get_statement() == ID_fence &&
1287  (instruction.code().get_bool("WRfence") ||
1288  instruction.code().get_bool("WWfence") ||
1289  instruction.code().get_bool("RWfence") ||
1290  instruction.code().get_bool("RRfence"))))
1291  {
1292  goto_programt::instructiont original_instruction;
1293  original_instruction.swap(instruction);
1294  const source_locationt &source_location =
1295  original_instruction.source_location();
1296 
1297  // ATOMIC_BEGIN
1298  instruction = goto_programt::make_atomic_begin(source_location);
1299  i_it++;
1300 
1301  // does it for all the previous statements
1302  for(std::set<irep_idt>::iterator s_it=past_writes.begin();
1303  s_it!=past_writes.end(); s_it++)
1304  {
1306  goto_program, i_it, source_location, *s_it,
1307  current_thread);
1308  }
1309 
1310  // ATOMIC_END
1311  i_it = goto_program.insert_before(
1312  i_it, goto_programt::make_atomic_end(source_location));
1313  i_it++;
1314 
1315  i_it--; // the for loop already counts us up
1316  }
1317  else if(is_lwfence(instruction, ns))
1318  {
1319  // po -- remove the lwfence
1320  *i_it = goto_programt::make_skip(i_it->source_location());
1321  }
1322  else if(instruction.is_function_call())
1323  {
1324  const exprt &fun = instruction.call_function();
1325  weak_memory(value_sets, to_symbol_expr(fun).get_identifier(), model);
1326  }
1327  }
1328 }
pointer_typet pointer_type(const typet &subtype)
Definition: c_types.cpp:235
Operator to return the address of an object.
Definition: pointer_expr.h:540
Boolean AND.
Definition: std_expr.h:2120
The Boolean type.
Definition: std_types.h:36
exprt & op1()
Definition: expr.h:136
const irep_idt & get_statement() const
Definition: std_code_base.h:65
Operator to dereference a pointer.
Definition: pointer_expr.h:834
dstringt has one field, an unsigned integer no which is an index into a static table of strings.
Definition: dstring.h:38
bool starts_with(const char *s) const
equivalent of as_string().starts_with(s)
Definition: dstring.h:95
Equality.
Definition: std_expr.h:1361
Base class for all expressions.
Definition: expr.h:56
const source_locationt & source_location() const
Definition: expr.h:231
The Boolean constant false.
Definition: std_expr.h:3072
A collection of goto functions.
function_mapt function_map
static irep_idt entry_point()
Get the identifier of the entry point to a goto model.
This class represents an instruction in the GOTO intermediate representation.
Definition: goto_program.h:181
const exprt & call_function() const
Get the function that is called for FUNCTION_CALL.
Definition: goto_program.h:272
const goto_instruction_codet & code() const
Get the code represented by this instruction.
Definition: goto_program.h:188
const source_locationt & source_location() const
Definition: goto_program.h:333
void swap(instructiont &instruction)
Swap two instructions.
Definition: goto_program.h:541
goto_program_instruction_typet type() const
What kind of instruction?
Definition: goto_program.h:344
A generic container class for the GOTO intermediate representation of one function.
Definition: goto_program.h:73
instructionst instructions
The list of instructions in the goto program.
Definition: goto_program.h:622
instructionst::iterator targett
Definition: goto_program.h:614
static instructiont make_atomic_end(const source_locationt &l=source_locationt::nil())
Definition: goto_program.h:990
static instructiont make_assignment(const code_assignt &_code, const source_locationt &l=source_locationt::nil())
Create an assignment instruction.
static instructiont make_skip(const source_locationt &l=source_locationt::nil())
Definition: goto_program.h:891
static instructiont make_atomic_begin(const source_locationt &l=source_locationt::nil())
Definition: goto_program.h:979
static instructiont make_assertion(const exprt &g, const source_locationt &l=source_locationt::nil())
Definition: goto_program.h:933
targett insert_before(const_targett target)
Insertion before the instruction pointed-to by the given instruction iterator target.
Definition: goto_program.h:692
The trinary if-then-else operator.
Definition: std_expr.h:2370
bool get_bool(const irep_idt &name) const
Definition: irep.cpp:57
void make_nil()
Definition: irep.h:446
mstreamt & warning() const
Definition: message.h:396
mstreamt & debug() const
Definition: message.h:421
message_handlert & get_message_handler()
Definition: message.h:183
static eomt eom
Definition: message.h:289
A namespacet is essentially one or two symbol tables bound together, to allow for symbol lookups in t...
Definition: namespace.h:94
bool lookup(const irep_idt &name, const symbolt *&symbol) const override
See documentation for namespace_baset::lookup().
Definition: namespace.cpp:148
Boolean negation.
Definition: std_expr.h:2327
The null pointer constant.
Definition: pointer_expr.h:909
Boolean OR.
Definition: std_expr.h:2228
entriest r_entries
Definition: rw_set.h:60
bool empty() const
Definition: rw_set.h:75
entriest w_entries
Definition: rw_set.h:60
shared_bufferst & shared_buffers
symbol_table_baset & symbol_table
void weak_memory(value_setst &value_sets, const irep_idt &function_id, memory_modelt model)
instruments the program for the pairs detected through the CFG
std::set< irep_idt > past_writes
goto_functionst & goto_functions
std::vector< irep_idt > r_buff0_thds
std::vector< irep_idt > r_buff1_thds
std::multimap< irep_idt, source_locationt > cycles_loc
const varst & operator()(const irep_idt &object)
instruments the variable
void nondet_flush(const irep_idt &function_id, goto_programt &goto_program, goto_programt::targett &t, const source_locationt &source_location, const irep_idt &object, const unsigned current_thread, const bool tso_pso_rmo)
instruments read
std::string unique()
returns a unique id (for fresh variables)
void write(goto_programt &goto_program, goto_programt::targett &t, const source_locationt &source_location, const irep_idt &object, goto_programt::instructiont &original_instruction, const unsigned current_thread)
instruments write
void affected_by_delay(value_setst &value_sets, goto_functionst &goto_functions)
analysis over the goto-program which computes in affected_by_delay_set the variables (non necessarily...
void det_flush(goto_programt &goto_program, goto_programt::targett &t, const source_locationt &source_location, const irep_idt &object, const unsigned current_thread)
flush buffers (instruments fence)
irep_idt add_fresh_var(const symbolt &object, const std::string &suffix, const typet &type)
void add_initialization(goto_programt &goto_program)
bool is_buffered_in_general(const symbol_exprt &, bool is_write)
void assignment(goto_programt &goto_program, goto_programt::targett &t, const source_locationt &source_location, const irep_idt &id_lhs, const exprt &rhs)
add an assignment in the goto-program
irep_idt add(const symbolt &object, const std::string &suffix, const typet &type, bool added_as_instrumentation)
add a new var for instrumenting the input var
std::set< irep_idt > affected_by_delay_set
class symbol_table_baset & symbol_table
std::set< irep_idt > cycles
void delay_read(goto_programt &goto_program, goto_programt::targett &t, const source_locationt &source_location, const irep_idt &read_object, const irep_idt &write_object)
delays a read (POWER)
void flush_read(goto_programt &goto_program, goto_programt::targett &t, const source_locationt &source_location, const irep_idt &write_object)
flushes read (POWER)
messaget & message
irep_idt choice(const irep_idt &function_id, const std::string &suffix)
void add_initialization_code(goto_functionst &goto_functions)
bool is_buffered(const namespacet &, const symbol_exprt &, bool is_write)
std::set< irep_idt > instrumentations
A side_effect_exprt that returns a non-deterministically chosen value.
Definition: std_code.h:1520
Expression to hold a symbol (variable)
Definition: std_expr.h:131
const irep_idt & get_identifier() const
Definition: std_expr.h:160
Symbol table entry.
Definition: symbol.h:28
bool is_static_lifetime
Definition: symbol.h:70
bool is_thread_local
Definition: symbol.h:71
typet type
Type of symbol.
Definition: symbol.h:31
irep_idt name
The unique identifier.
Definition: symbol.h:40
exprt value
Initial value of symbol.
Definition: symbol.h:34
The Boolean constant true.
Definition: std_expr.h:3063
The type of an expression, extends irept.
Definition: type.h:29
#define CPROVER_PREFIX
bool is_fence(const goto_programt::instructiont &instruction, const namespacet &ns)
Definition: fence.cpp:19
bool is_lwfence(const goto_programt::instructiont &instruction, const namespacet &ns)
Definition: fence.cpp:35
Fences for instrumentation.
int main(int argc, char *argv[])
symbolt & get_fresh_aux_symbol(const typet &type, const std::string &name_prefix, const std::string &basename_prefix, const source_locationt &source_location, const irep_idt &symbol_mode, const namespacet &ns, symbol_table_baset &symbol_table)
Installs a fresh-named symbol with respect to the given namespace ns with the requested name pattern ...
Fresh auxiliary symbol creation.
#define Forall_goto_program_instructions(it, program)
const std::string & id2string(const irep_idt &d)
Definition: irep.h:44
literalt pos(literalt a)
Definition: literal.h:194
API to expression classes for Pointers.
Race Detection for Threaded Goto Programs.
#define INITIALIZE_FUNCTION
const symbol_exprt & to_symbol_expr(const exprt &expr)
Cast an exprt to a symbol_exprt.
Definition: std_expr.h:272
std::string to_string(const string_not_contains_constraintt &expr)
Used for debug printing.
memory_modelt
Definition: wmm.h:18
@ TSO
Definition: wmm.h:20
@ RMO
Definition: wmm.h:22
@ PSO
Definition: wmm.h:21
@ Power
Definition: wmm.h:23