From: Brian Norris Date: Sat, 25 Aug 2012 01:37:15 +0000 (-0700) Subject: model: add resolve_release_sequences() function X-Git-Tag: pldi2013~245 X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=b8084847aa9aeec879968c6879d7e6a20c08ea1f;p=model-checker.git model: add resolve_release_sequences() function This function can check for release sequence resolutions then propagate synchronization and remove from the "lazy release" list. It does not add any new mo_graph edges yet. --- diff --git a/model.cc b/model.cc index d2e4d9b..69cffcd 100644 --- a/model.cc +++ b/model.cc @@ -647,6 +647,58 @@ void ModelChecker::get_release_seq_heads(ModelAction *act, } } +/** + * Attempt to resolve all stashed operations that might synchronize with a + * release sequence for a given location. This implements the "lazy" portion of + * determining whether or not a release sequence was contiguous, since not all + * modification order information is present at the time an action occurs. + * + * @param location The location/object that should be checked for release + * sequence resolutions + * @return True if any updates occurred (new synchronization, new mo_graph edges) + */ +bool ModelChecker::resolve_release_sequences(void *location) +{ + std::list *list; + list = lazy_sync_with_release->getptr(location); + if (!list) + return false; + + bool updated = false; + std::list::iterator it = list->begin(); + while (it != list->end()) { + ModelAction *act = *it; + const ModelAction *rf = act->get_reads_from(); + std::vector release_heads; + bool complete; + complete = release_seq_head(rf, &release_heads); + for (unsigned int i = 0; i < release_heads.size(); i++) { + if (!act->has_synchronized_with(release_heads[i])) { + updated = true; + act->synchronize_with(release_heads[i]); + } + } + + if (updated) { + /* propagate synchronization to later actions */ + action_list_t::reverse_iterator it = action_trace->rbegin(); + while ((*it) != act) { + ModelAction *propagate = *it; + if (act->happens_before(propagate)) + /** @todo new mo_graph edges along with + * this synchronization? */ + propagate->synchronize_with(act); + } + } + if (complete) + it = list->erase(it); + else + it++; + } + + return updated; +} + /** * Performs various bookkeeping operations for the current ModelAction. For * instance, adds action to the per-object, per-thread action vector and to the diff --git a/model.h b/model.h index 7d5489e..970881c 100644 --- a/model.h +++ b/model.h @@ -107,6 +107,7 @@ private: bool w_modification_order(ModelAction *curr); bool release_seq_head(const ModelAction *rf, std::vector *release_heads) const; + bool resolve_release_sequences(void *location); ModelAction *current_action; ModelAction *diverge;