Chromium Code Reviews

Unified Diff: src/compiler/graph-reducer.cc

Issue 727743002: Revert "[turbofan] Smartify the GraphReducer." (Closed) Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: Created 6 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments.
Jump to:
View side-by-side diff with in-line comments
« no previous file with comments | « src/compiler/graph-reducer.h ('k') | src/compiler/pipeline.cc » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: src/compiler/graph-reducer.cc
diff --git a/src/compiler/graph-reducer.cc b/src/compiler/graph-reducer.cc
index ff04f6ba5f01d643c473a2efb077ef94e9e510bf..f716b2a571b82816501a1d27da610a05934216ef 100644
--- a/src/compiler/graph-reducer.cc
+++ b/src/compiler/graph-reducer.cc
@@ -12,189 +12,79 @@ namespace v8 {
namespace internal {
namespace compiler {
-enum class GraphReducer::State : uint8_t {
- kUnvisited,
- kRevisit,
- kOnStack,
- kVisited
-};
-
-
-GraphReducer::GraphReducer(Graph* graph, Zone* zone)
- : graph_(graph),
- reducers_(zone),
- revisit_(zone),
- stack_(zone),
- state_(zone) {}
+GraphReducer::GraphReducer(Graph* graph)
+ : graph_(graph), reducers_(graph->zone()) {}
-void GraphReducer::AddReducer(Reducer* reducer) {
- reducers_.push_back(reducer);
+static bool NodeIdIsLessThan(const Node* node, NodeId id) {
+ return node->id() < id;
}
void GraphReducer::ReduceNode(Node* node) {
- DCHECK(stack_.empty());
- DCHECK(revisit_.empty());
- std::fill(state_.begin(), state_.end(), State::kUnvisited);
- Push(node);
- for (;;) {
- DCHECK(!stack_.empty() ||
- std::find(state_.begin(), state_.end(), State::kOnStack) ==
- state_.end());
- if (!stack_.empty()) {
- // Process the node on the top of the stack, potentially pushing more or
- // popping the node off the stack.
- ReduceTop();
- } else if (!revisit_.empty()) {
- // If the stack becomes empty, revisit any nodes in the revisit queue.
- Node* const node = revisit_.top();
- revisit_.pop();
- if (state_[node->id()] == State::kRevisit) {
- // state can change while in queue.
- Push(node);
- }
- } else {
- break;
- }
- }
- DCHECK(std::find(state_.begin(), state_.end(), State::kOnStack) ==
- state_.end());
- DCHECK(revisit_.empty());
- DCHECK(stack_.empty());
-}
-
-
-void GraphReducer::ReduceGraph() { ReduceNode(graph()->end()); }
-
-
-Reduction GraphReducer::Reduce(Node* const node) {
- auto skip = reducers_.end();
- for (auto i = reducers_.begin(); i != reducers_.end();) {
- if (i != skip) {
+ static const unsigned kMaxAttempts = 16;
+ bool reduce = true;
+ for (unsigned attempts = 0; attempts <= kMaxAttempts; ++attempts) {
+ if (!reduce) return;
+ reduce = false; // Assume we don't need to rerun any reducers.
+ int before = graph_->NodeCount();
+ for (ZoneVector<Reducer*>::iterator i = reducers_.begin();
+ i != reducers_.end(); ++i) {
Reduction reduction = (*i)->Reduce(node);
- if (!reduction.Changed()) {
+ Node* replacement = reduction.replacement();
+ if (replacement == NULL) {
// No change from this reducer.
- } else if (reduction.replacement() == node) {
- // {replacement} == {node} represents an in-place reduction. Rerun
- // all the other reducers for this node, as now there may be more
+ } else if (replacement == node) {
+ // {replacement == node} represents an in-place reduction.
+ // Rerun all the reducers for this node, as now there may be more
// opportunities for reduction.
- skip = i;
- i = reducers_.begin();
- continue;
+ reduce = true;
+ break;
} else {
- // {node} was replaced by another node.
- return reduction;
- }
- }
- ++i;
- }
- if (skip == reducers_.end()) {
- // No change from any reducer.
- return Reducer::NoChange();
- }
- // At least one reducer did some in-place reduction.
- return Reducer::Changed(node);
-}
-
-
-void GraphReducer::ReduceTop() {
- Node* const node = Top();
- if (node->IsDead()) return Pop(); // Node was killed while on stack.
-
- // Recurse on an input if necessary.
- for (auto const input : node->inputs()) {
- if (input != node && Recurse(input)) return;
- }
-
- // Remember the node count before reduction.
- const int node_count = graph()->NodeCount();
-
- // All inputs should be visited or on stack. Apply reductions to node.
- Reduction reduction = Reduce(node);
-
- // After reducing the node, pop it off the stack.
- Pop();
-
- // If there was a reduction, revisit the uses and reduce the replacement.
- if (reduction.Changed()) {
- for (Node* const use : node->uses()) {
- // Don't revisit this node if it refers to itself.
- if (use != node) Revisit(use);
- }
- Node* const replacement = reduction.replacement();
- if (replacement != node) {
- if (node == graph()->start()) graph()->SetStart(replacement);
- if (node == graph()->end()) graph()->SetEnd(replacement);
- // If {node} was replaced by an old node, unlink {node} and assume that
- // {replacement} was already reduced and finish.
- if (replacement->id() < node_count) {
- node->ReplaceUses(replacement);
- node->Kill();
- } else {
- // Otherwise {node} was replaced by a new node. Replace all old uses of
+ if (node == graph_->start()) graph_->SetStart(replacement);
+ if (node == graph_->end()) graph_->SetEnd(replacement);
+ // If {node} was replaced by an old node, unlink {node} and assume that
+ // {replacement} was already reduced and finish.
+ if (replacement->id() < before) {
+ node->ReplaceUses(replacement);
+ node->Kill();
+ return;
+ }
+ // Otherwise, {node} was replaced by a new node. Replace all old uses of
// {node} with {replacement}. New nodes created by this reduction can
// use {node}.
- node->ReplaceUsesIf([node_count](Node* const node) {
- return node->id() < node_count;
- },
- replacement);
+ node->ReplaceUsesIf(
+ std::bind2nd(std::ptr_fun(&NodeIdIsLessThan), before), replacement);
// Unlink {node} if it's no longer used.
if (node->uses().empty()) {
node->Kill();
}
-
- // If there was a replacement, reduce it after popping {node}.
- Recurse(replacement);
+ // Rerun all the reductions on the {replacement}.
+ node = replacement;
+ reduce = true;
+ break;
}
}
}
}
-void GraphReducer::Pop() {
- Node* const node = Top();
- state_[node->id()] = State::kVisited;
- stack_.pop();
-}
-
-
-void GraphReducer::Push(Node* const node) {
- size_t const id = static_cast<size_t>(node->id());
- if (id >= state_.size()) state_.resize(id + 1);
- DCHECK(id < state_.size());
- DCHECK(state_[id] != State::kOnStack);
- state_[id] = State::kOnStack;
- stack_.push(node);
-}
-
-
-Node* GraphReducer::Top() const {
- DCHECK(!stack_.empty());
- Node* const node = stack_.top();
- size_t const id = static_cast<size_t>(node->id());
- DCHECK(id < state_.size());
- DCHECK(state_[id] == State::kOnStack);
- USE(id);
- return node;
-}
+// A helper class to reuse the node traversal algorithm.
+struct GraphReducerVisitor FINAL : public NullNodeVisitor {
+ explicit GraphReducerVisitor(GraphReducer* reducer) : reducer_(reducer) {}
+ void Post(Node* node) { reducer_->ReduceNode(node); }
+ GraphReducer* reducer_;
+};
-bool GraphReducer::Recurse(Node* const node) {
- size_t const id = static_cast<size_t>(node->id());
- if (id < state_.size() && state_[id] > State::kRevisit) return false;
- Push(node);
- return true;
+void GraphReducer::ReduceGraph() {
+ GraphReducerVisitor visitor(this);
+ // Perform a post-order reduction of all nodes starting from the end.
+ graph()->VisitNodeInputsFromEnd(&visitor);
}
-void GraphReducer::Revisit(Node* const node) {
- size_t const id = static_cast<size_t>(node->id());
- if (id < state_.size() && state_[id] == State::kVisited) {
- state_[id] = State::kRevisit;
- revisit_.push(node);
- }
-}
+// TODO(titzer): partial graph reductions.
} // namespace compiler
} // namespace internal
« no previous file with comments | « src/compiler/graph-reducer.h ('k') | src/compiler/pipeline.cc » ('j') | no next file with comments »

Powered by Google App Engine