io.questdb.griffin.PostOrderTreeTraversalAlgo Maven / Gradle / Ivy
/*******************************************************************************
* ___ _ ____ ____
* / _ \ _ _ ___ ___| |_| _ \| __ )
* | | | | | | |/ _ \/ __| __| | | | _ \
* | |_| | |_| | __/\__ \ |_| |_| | |_) |
* \__\_\\__,_|\___||___/\__|____/|____/
*
* Copyright (c) 2014-2019 Appsicle
* Copyright (c) 2019-2020 QuestDB
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
******************************************************************************/
package io.questdb.griffin;
import io.questdb.griffin.model.ExpressionNode;
import io.questdb.std.IntStack;
import java.util.ArrayDeque;
final public class PostOrderTreeTraversalAlgo {
private final ArrayDeque stackBackup = new ArrayDeque<>();
private final IntStack indexStackBackup = new IntStack();
private final IntStack backupDepth = new IntStack();
private final ArrayDeque stack = new ArrayDeque<>();
private final IntStack indexStack = new IntStack();
public void backup() {
int size = stack.size();
backupDepth.push(size);
for (int i = 0; i < size; i++) {
stackBackup.push(stack.poll());
indexStackBackup.push(indexStack.pop());
}
}
public void restore() {
if (backupDepth.size() > 0) {
int size = backupDepth.pop();
for (int i = 0; i < size; i++) {
stack.push(stackBackup.poll());
indexStack.push(indexStackBackup.pop());
}
}
}
public void traverse(ExpressionNode node, Visitor visitor) throws SqlException {
backup();
try {
// post-order iterative tree traversal
// see http://en.wikipedia.org/wiki/Tree_traversal
stack.clear();
indexStack.clear();
ExpressionNode lastVisited = null;
while (!stack.isEmpty() || node != null) {
if (node != null) {
stack.push(node);
indexStack.push(0);
node = node.rhs;
} else {
ExpressionNode peek = stack.peek();
assert peek != null;
if (peek.paramCount < 3) {
if (peek.lhs != null && lastVisited != peek.lhs) {
node = peek.lhs;
} else {
visitor.visit(peek);
lastVisited = stack.poll();
indexStack.pop();
}
} else {
int index = indexStack.peek();
if (index < peek.paramCount) {
node = peek.args.getQuick(index);
indexStack.update(index + 1);
} else {
visitor.visit(peek);
lastVisited = stack.poll();
indexStack.pop();
}
}
}
}
} finally {
restore();
}
}
public interface Visitor {
void visit(ExpressionNode node) throws SqlException;
}
}