/*@

inductive tree = empty(Tree node) | nonempty(Tree node, tree left, tree right);

fixpoint int node_count(tree tree) {
    switch (tree) {
        case empty(node): return 1;
        case nonempty(node, left, right): return 1 + node_count(left) + node_count(right);
    }
}

lemma_auto void node_count_positive(tree tree)
    requires true;
    ensures node_count(tree) >= 1;
{
    switch (tree) {
        case empty(node):
        case nonempty(node, left, right):
            node_count_positive(left);
            node_count_positive(right);
    }
}

predicate tree(Tree node, boolean marked; Tree parent, tree shape) =
    node.left |-> ?left &*&
    node.right |-> ?right &*&
    node.mark |-> ?mark &*& (marked ? mark == true : true) &*&
    node.parent |-> parent &*&
    left == null ?
        right == null &*&
        shape == empty(node)
    :
        right != null &*&
        tree(left, marked, node, ?leftShape) &*&
        tree(right, marked, node, ?rightShape) &*&
        shape == nonempty(node, leftShape, rightShape);

predicate stack(Tree parent, Tree current, tree currentNodeShape, Tree root, tree rootShape, int stepsLeft) =
    current != null &*&
    parent == null ?
        root == current &*& rootShape == currentNodeShape &*& stepsLeft == 0
    :
        parent.left |-> ?left &*&
        parent.right |-> ?right &*&
        parent.mark |-> true &*&
        parent.parent |-> current &*&
        exists<boolean>(?currentIsLeftChild) &*&
        currentIsLeftChild ?
            tree(left, false, parent, ?rightShape) &*& left != null &*&
            stack(right, parent, nonempty(parent, currentNodeShape, rightShape), root, rootShape, ?stepsLeft1) &*& stepsLeft1 >= 0 &*&
            stepsLeft == node_count(rightShape) * 2 + 1 + stepsLeft1
        :
            tree(right, true, parent, ?leftShape) &*& right != null &*&
            stack(left, parent, nonempty(parent, leftShape, currentNodeShape), root, rootShape, ?stepsLeft1) &*& stepsLeft1 >= 0 &*&
            stepsLeft == 1 + stepsLeft1;

lemma void tree_nonnull(Tree t)
    requires tree(t, ?marked, ?parent, ?shape);
    ensures tree(t, marked, parent, shape) &*& t != null;
{
    open tree(t, marked, parent, shape);
    close tree(t, marked, parent, shape);
}

predicate inv(boolean xIsNew, Tree x, Tree root, tree rootShape, int stepsLeft) =
        xIsNew ?
            tree(x, false, ?parent, ?xShape) &*& stack(parent, x, xShape, root, rootShape, ?stepsLeft1) &*&
            stepsLeft1 >= 0 &*& stepsLeft == node_count(xShape) * 2 - 1 + stepsLeft1
        :
            stack(x, ?child, ?childShape, root, rootShape, stepsLeft) &*& stepsLeft >= 0 &*&
            tree(child, true, x, childShape);
    

@*/

class Tree {
	Tree left, right, parent;
	boolean mark;
	
	static void markTree(Tree root)
	    //@ requires tree(root, false, null, ?rootShape);
	    //@ ensures tree(root, true, null, rootShape);
	{
		TODO
	}
}

