From 24dbae977ee0610d314b52a42a77f1a6f39752b3 Mon Sep 17 00:00:00 2001 From: Brian Demsky Date: Tue, 22 Jan 2013 16:31:16 -0800 Subject: [PATCH] fix bug in promise handling code... if any promise for a thread resolved, we assumed that the corresponding write happened after all promises for that threads...clearly doesn't need to be the case... [Brian Norris:] This fixes some bugs where we don't see all the expected behaviors: e.g., with ./run.sh test/rmwprog.o -- 3 Now we see all 20 behaviors. --- model.cc | 40 +++++++++++++++++++++++----------------- model.h | 2 +- 2 files changed, 24 insertions(+), 18 deletions(-) diff --git a/model.cc b/model.cc index e6df37f8..94f68e20 100644 --- a/model.cc +++ b/model.cc @@ -730,7 +730,7 @@ bool ModelChecker::process_read(ModelAction *curr, bool second_part_of_rmw) read_from(curr, reads_from); mo_graph->commitChanges(); - mo_check_promises(curr->get_tid(), reads_from); + mo_check_promises(curr->get_tid(), reads_from, NULL); updated |= r_status; } else if (!second_part_of_rmw) { @@ -881,7 +881,7 @@ bool ModelChecker::process_write(ModelAction *curr) } mo_graph->commitChanges(); - mo_check_promises(curr->get_tid(), curr); + mo_check_promises(curr->get_tid(), curr, NULL); get_thread(curr)->set_return_value(VALUE_NONE); return updated_mod_order || updated_promises; @@ -2310,7 +2310,7 @@ ClockVector * ModelChecker::get_cv(thread_id_t tid) const bool ModelChecker::resolve_promises(ModelAction *write) { bool resolved = false; - std::vector< thread_id_t, ModelAlloc > threads_to_check; + std::vector< ModelAction *, ModelAlloc > actions_to_check; for (unsigned int i = 0, promise_index = 0; promise_index < promises->size(); i++) { Promise *promise = (*promises)[promise_index]; @@ -2331,7 +2331,7 @@ bool ModelChecker::resolve_promises(ModelAction *write) delete(promise); promises->erase(promises->begin() + promise_index); - threads_to_check.push_back(read->get_tid()); + actions_to_check.push_back(read); resolved = true; } else @@ -2341,8 +2341,10 @@ bool ModelChecker::resolve_promises(ModelAction *write) //Check whether reading these writes has made threads unable to //resolve promises - for (unsigned int i = 0; i < threads_to_check.size(); i++) - mo_check_promises(threads_to_check[i], write); + for (unsigned int i = 0; i < actions_to_check.size(); i++) { + ModelAction *read=actions_to_check[i]; + mo_check_promises(read->get_tid(), write, read); + } return resolved; } @@ -2421,8 +2423,9 @@ void ModelChecker::check_promises_thread_disabled() { * write, or actually did the model action write. * * @param write The ModelAction representing the relevant write. + * @param read The ModelAction that reads a promised write, or NULL otherwise. */ -void ModelChecker::mo_check_promises(thread_id_t tid, const ModelAction *write) +void ModelChecker::mo_check_promises(thread_id_t tid, const ModelAction *write, const ModelAction *read) { void *location = write->get_location(); for (unsigned int i = 0; i < promises->size(); i++) { @@ -2435,20 +2438,23 @@ void ModelChecker::mo_check_promises(thread_id_t tid, const ModelAction *write) //same thread as the promise if (act->get_tid() == tid) { - - //do we have a pwrite for the promise, if not, set it - if (promise->get_write() == NULL) { - promise->set_write(write); - //The pwrite cannot happen before the promise - if (write->happens_before(act) && (write != act)) { + //make sure that the reader of this write happens after the promise + if (( read == NULL ) || ( promise->get_action() -> happens_before(read))) { + //do we have a pwrite for the promise, if not, set it + if (promise->get_write() == NULL) { + promise->set_write(write); + //The pwrite cannot happen before the promise + if (write->happens_before(act) && (write != act)) { + priv->failed_promise = true; + return; + } + } + + if (mo_graph->checkPromise(write, promise)) { priv->failed_promise = true; return; } } - if (mo_graph->checkPromise(write, promise)) { - priv->failed_promise = true; - return; - } } //Don't do any lookups twice for the same thread diff --git a/model.h b/model.h index 97a89892..5a127b1d 100644 --- a/model.h +++ b/model.h @@ -121,7 +121,7 @@ public: ClockVector * get_cv(thread_id_t tid) const; ModelAction * get_parent_action(thread_id_t tid) const; void check_promises_thread_disabled(); - void mo_check_promises(thread_id_t tid, const ModelAction *write); + void mo_check_promises(thread_id_t tid, const ModelAction *write, const ModelAction * read); void check_promises(thread_id_t tid, ClockVector *old_cv, ClockVector *merge_cv); bool isfeasibleprefix() const; -- 2.34.1