Make Graph more thread save by removing unneeded sharing of data between function calls:
1. Q_
When Q_ is used then it always has been cleared before usage in bfs_init 
(if bfs_init returns false Q_ is not used).
Therefore we could make Q_ local to the functions.

2. vertices_[].path
.path is only used in getPath, and the pathes are always cleared before usage,
so the pathes could also be local.

The locks are not needed here any more. When there will be errors due to 
race conditions, we could not solve them here. We have to lock at higher
places. 



git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@37257 a592a061-630c-0410-9148-cb99ea01b6c8
This commit is contained in:
Peter Kümmel 2011-01-19 09:03:41 +00:00
parent 7e7f98532d
commit 2cd88f07b3
2 changed files with 36 additions and 62 deletions

View File

@ -24,12 +24,12 @@ using namespace std;
namespace lyx { namespace lyx {
bool Graph::bfs_init(int s, bool clear_visited) bool Graph::bfs_init(int s, bool clear_visited, queue<int>* Q)
{ {
if (s < 0) if (s < 0 || !Q)
return false; return false;
Q_ = queue<int>(); *Q = queue<int>();
if (clear_visited) { if (clear_visited) {
vector<Vertex>::iterator it = vertices_.begin(); vector<Vertex>::iterator it = vertices_.begin();
@ -38,40 +38,30 @@ bool Graph::bfs_init(int s, bool clear_visited)
it->visited = false; it->visited = false;
} }
if (!vertices_[s].visited) { if (!vertices_[s].visited) {
Q_.push(s); Q->push(s);
vertices_[s].visited = true; vertices_[s].visited = true;
} }
return true; return true;
} }
void Graph::clearPaths()
{
vector<Vertex>::iterator it = vertices_.begin();
vector<Vertex>::iterator en = vertices_.end();
for (; it != en; ++it)
it->path.clear();
}
vector<int> const vector<int> const
Graph::getReachableTo(int target, bool clear_visited) Graph::getReachableTo(int target, bool clear_visited)
{ {
Mutex::Locker lock(&mutex_);
vector<int> result; vector<int> result;
if (!bfs_init(target, clear_visited)) queue<int> Q;
if (!bfs_init(target, clear_visited, &Q))
return result; return result;
// Here's the logic, which is shared by the other routines. // Here's the logic, which is shared by the other routines.
// Q_ holds a list of nodes we have been able to reach (in this // Q holds a list of nodes we have been able to reach (in this
// case, reach backwards). It is initialized to the current node // case, reach backwards). It is initialized to the current node
// by bfs_init, and then we recurse, adding the nodes we can reach // by bfs_init, and then we recurse, adding the nodes we can reach
// from the current node as we go. That makes it a breadth-first // from the current node as we go. That makes it a breadth-first
// search. // search.
while (!Q_.empty()) { while (!Q.empty()) {
int const current = Q_.front(); int const current = Q.front();
Q_.pop(); Q.pop();
if (current != target || formats.get(target).name() != "lyx") if (current != target || formats.get(target).name() != "lyx")
result.push_back(current); result.push_back(current);
@ -81,7 +71,7 @@ vector<int> const
const int cv = (*it)->from; const int cv = (*it)->from;
if (!vertices_[cv].visited) { if (!vertices_[cv].visited) {
vertices_[cv].visited = true; vertices_[cv].visited = true;
Q_.push(cv); Q.push(cv);
} }
} }
} }
@ -94,15 +84,14 @@ vector<int> const
Graph::getReachable(int from, bool only_viewable, Graph::getReachable(int from, bool only_viewable,
bool clear_visited) bool clear_visited)
{ {
Mutex::Locker lock(&mutex_);
vector<int> result; vector<int> result;
if (!bfs_init(from, clear_visited)) queue<int> Q;
if (!bfs_init(from, clear_visited, &Q))
return result; return result;
while (!Q_.empty()) { while (!Q.empty()) {
int const current = Q_.front(); int const current = Q.front();
Q_.pop(); Q.pop();
Format const & format = formats.get(current); Format const & format = formats.get(current);
if (!only_viewable || !format.viewer().empty()) if (!only_viewable || !format.viewer().empty())
result.push_back(current); result.push_back(current);
@ -121,7 +110,7 @@ vector<int> const
int const cv = (*cit)->to; int const cv = (*cit)->to;
if (!vertices_[cv].visited) { if (!vertices_[cv].visited) {
vertices_[cv].visited = true; vertices_[cv].visited = true;
Q_.push(cv); Q.push(cv);
} }
} }
} }
@ -132,17 +121,16 @@ vector<int> const
bool Graph::isReachable(int from, int to) bool Graph::isReachable(int from, int to)
{ {
Mutex::Locker lock(&mutex_);
if (from == to) if (from == to)
return true; return true;
if (to < 0 || !bfs_init(from)) queue<int> Q;
if (to < 0 || !bfs_init(from, true, &Q))
return false; return false;
while (!Q_.empty()) { while (!Q.empty()) {
int const current = Q_.front(); int const current = Q.front();
Q_.pop(); Q.pop();
if (current == to) if (current == to)
return true; return true;
@ -154,7 +142,7 @@ bool Graph::isReachable(int from, int to)
int const cv = (*cit)->to; int const cv = (*cit)->to;
if (!vertices_[cv].visited) { if (!vertices_[cv].visited) {
vertices_[cv].visited = true; vertices_[cv].visited = true;
Q_.push(cv); Q.push(cv);
} }
} }
} }
@ -165,18 +153,18 @@ bool Graph::isReachable(int from, int to)
Graph::EdgePath const Graph::getPath(int from, int to) Graph::EdgePath const Graph::getPath(int from, int to)
{ {
Mutex::Locker lock(&mutex_);
if (from == to) if (from == to)
return EdgePath(); return EdgePath();
if (to < 0 || !bfs_init(from)) queue<int> Q;
if (to < 0 || !bfs_init(from, true, &Q))
return EdgePath(); return EdgePath();
clearPaths(); vector<EdgePath> pathes;
while (!Q_.empty()) { pathes.resize(vertices_.size());
int const current = Q_.front(); while (!Q.empty()) {
Q_.pop(); int const current = Q.front();
Q.pop();
vector<Arrow *>::const_iterator cit = vector<Arrow *>::const_iterator cit =
vertices_[current].out_arrows.begin(); vertices_[current].out_arrows.begin();
@ -186,16 +174,16 @@ Graph::EdgePath const Graph::getPath(int from, int to)
int const cv = (*cit)->to; int const cv = (*cit)->to;
if (!vertices_[cv].visited) { if (!vertices_[cv].visited) {
vertices_[cv].visited = true; vertices_[cv].visited = true;
Q_.push(cv); Q.push(cv);
// NOTE If we wanted to collect all the paths, then // NOTE If we wanted to collect all the paths, then
// we just need to collect them here and not worry // we just need to collect them here and not worry
// about "visited". // about "visited".
EdgePath lastpath = vertices_[(*cit)->from].path; EdgePath lastpath = pathes[(*cit)->from];
lastpath.push_back((*cit)->id); lastpath.push_back((*cit)->id);
vertices_[cv].path = lastpath; pathes[cv] = lastpath;
} }
if (cv == to) { if (cv == to) {
return vertices_[cv].path; return pathes[cv];
} }
} }
} }
@ -206,8 +194,6 @@ Graph::EdgePath const Graph::getPath(int from, int to)
void Graph::init(int size) void Graph::init(int size)
{ {
Mutex::Locker lock(&mutex_);
vertices_ = vector<Vertex>(size); vertices_ = vector<Vertex>(size);
arrows_.clear(); arrows_.clear();
numedges_ = 0; numedges_ = 0;
@ -216,8 +202,6 @@ void Graph::init(int size)
void Graph::addEdge(int from, int to) void Graph::addEdge(int from, int to)
{ {
Mutex::Locker lock(&mutex_);
arrows_.push_back(Arrow(from, to, numedges_)); arrows_.push_back(Arrow(from, to, numedges_));
numedges_++; numedges_++;
Arrow * ar = &(arrows_.back()); Arrow * ar = &(arrows_.back());

View File

@ -13,7 +13,6 @@
#ifndef GRAPH_H #ifndef GRAPH_H
#define GRAPH_H #define GRAPH_H
#include "support/mutex.h"
#include <list> #include <list>
#include <queue> #include <queue>
@ -47,10 +46,7 @@ public:
private: private:
/// ///
bool bfs_init(int, bool clear_visited = true); bool bfs_init(int, bool clear_visited, std::queue<int>* Q);
/// clears the paths from a previous search. should be
/// called before each new one.
void clearPaths();
/// used to recover a marked path /// used to recover a marked path
void getMarkedPath(int from, int to, EdgePath & path); void getMarkedPath(int from, int to, EdgePath & path);
/// these represent the arrows connecting the nodes of the graph. /// these represent the arrows connecting the nodes of the graph.
@ -83,8 +79,6 @@ private:
std::vector<Arrow *> out_arrows; std::vector<Arrow *> out_arrows;
/// used in the search routines /// used in the search routines
bool visited; bool visited;
///
EdgePath path;
}; };
/// a container for the vertices /// a container for the vertices
/// the index into the vector functions as the identifier by which /// the index into the vector functions as the identifier by which
@ -94,17 +88,13 @@ private:
/// of Format, this is easy, since the Format objects already have ints /// of Format, this is easy, since the Format objects already have ints
/// as identifiers.) /// as identifiers.)
std::vector<Vertex> vertices_; std::vector<Vertex> vertices_;
///
std::queue<int> Q_;
/// a counter that we use to assign id's to the arrows /// a counter that we use to assign id's to the arrows
/// FIXME This technique assumes a correspondence between the /// FIXME This technique assumes a correspondence between the
/// ids of the arrows and ids associated with Converters that /// ids of the arrows and ids associated with Converters that
/// seems kind of fragile. Perhaps a better solution would be /// seems kind of fragile. Perhaps a better solution would be
/// to pass the ids as we create the arrows. /// to pass the ids as we create the arrows.
int numedges_; int numedges_;
/// make public functions thread save
Mutex mutex_;
}; };