/* SPDX-License-Identifier: GPL-3.0-or-later */
/*
 * rbtree.c
 *
 * Copyright (C) 2020 David Oberhollenzer <goliath@infraroot.at>
 */
#include "config.h"

#include "rbtree.h"
#include "../test.h"

static int key_compare(const void *ctx, const void *a, const void *b)
{
	(void)ctx;
	return *((const sqfs_s32 *)a) - *((const sqfs_s32 *)b);
}

static size_t count_nodes_dfs(rbtree_node_t *n)
{
	return 1 + (n->left == NULL ? 0 : count_nodes_dfs(n->left))
		+ (n->right == NULL ? 0 : count_nodes_dfs(n->right));
}

static size_t min_depth(rbtree_node_t *n)
{
	size_t lhs, rhs;

	if (n == NULL)
		return 0;

	lhs = min_depth(n->left) + 1;
	rhs = min_depth(n->right) + 1;

	return lhs < rhs ? lhs : rhs;
}

static size_t max_depth(rbtree_node_t *n)
{
	size_t lhs, rhs;

	if (n == NULL)
		return 0;

	lhs = min_depth(n->left) + 1;
	rhs = min_depth(n->right) + 1;

	return lhs > rhs ? lhs : rhs;
}

static size_t get_ref_black_depth(rbtree_t *rb)
{
	rbtree_node_t *n;
	size_t count = 0;

	for (n = rb->root; n != NULL; n = n->left) {
		if (!n->is_red)
			count += 1;
	}

	return count;
}

static void check_binary_tree_dfs(rbtree_node_t *n)
{
	const void *key = rbtree_node_key(n);
	const void *cmp;

	if (n->left != NULL) {
		cmp = rbtree_node_key(n->left);
		TEST_ASSERT(key_compare(NULL, cmp, key) < 0);

		check_binary_tree_dfs(n->left);
	}

	if (n->right != NULL) {
		cmp = rbtree_node_key(n->right);
		TEST_ASSERT(key_compare(NULL, cmp, key) > 0);

		check_binary_tree_dfs(n->right);
	}
}

static void check_colors_dfs(rbtree_node_t *n)
{
	if (n->is_red) {
		TEST_ASSERT(n->left == NULL || !n->left->is_red);
		TEST_ASSERT(n->right == NULL || !n->right->is_red);
	}

	if (n->left != NULL)
		check_colors_dfs(n->left);

	if (n->right != NULL)
		check_colors_dfs(n->right);
}

static void check_black_depth_dfs(rbtree_node_t *n, size_t ref,
				  size_t counter)
{
	if (!n->is_red)
		counter += 1;

	if (n->left == NULL || n->right == NULL)
		TEST_EQUAL_UI(counter, ref);

	if (n->left != NULL)
		check_black_depth_dfs(n->left, ref, counter);

	if (n->right != NULL)
		check_black_depth_dfs(n->right, ref, counter);
}

static int check_subtrees_equal(const rbtree_node_t *lhs,
				const rbtree_node_t *rhs,
				size_t datasize)
{
	if (lhs == rhs)
		return -1;

	if (lhs->value_offset != rhs->value_offset)
		return -1;

	if ((lhs->is_red && !rhs->is_red) || (!lhs->is_red && rhs->is_red))
		return -1;

	if (memcmp(lhs->data, rhs->data, datasize) != 0)
		return -1;

	if (lhs->left == NULL) {
		if (rhs->left != NULL)
			return -1;
	} else {
		if (rhs->left == NULL)
			return -1;

		if (check_subtrees_equal(lhs->left, rhs->left, datasize))
			return -1;
	}

	if (lhs->right == NULL) {
		if (rhs->right != NULL)
			return -1;
	} else {
		if (rhs->right == NULL)
			return -1;

		if (check_subtrees_equal(lhs->right, rhs->right, datasize))
			return -1;
	}

	return 0;
}

int main(int argc, char **argv)
{
	size_t count, blkdepth, mind, maxd;
	sqfs_s32 key, key2;
	rbtree_t rb, copy;
	rbtree_node_t *n;
	sqfs_u64 value;
	int ret;
	(void)argc; (void)argv;

	TEST_ASSERT(rbtree_init(&rb, sizeof(sqfs_s32),
				sizeof(sqfs_u64), key_compare) == 0);

	count = 0;

	for (key = -1000; key < 1000; ++key) {
		/* lookup of current key must fail prior to insert */
		TEST_NULL(rbtree_lookup(&rb, &key));

		/* previous key/value pairs must still be there */
		for (key2 = -1000; key2 < key; ++key2) {
			n = rbtree_lookup(&rb, &key2);
			TEST_NOT_NULL(n);
			value = *((sqfs_u64 *)rbtree_node_value(n));
			TEST_EQUAL_UI((sqfs_u64)(key2 + 10000), value);
		}

		/* insert key value pair */
		value = key + 10000;
		TEST_ASSERT(rbtree_insert(&rb, &key, &value) == 0);
		count += 1;

		/* check if the tree has the right number of nodes */
		TEST_EQUAL_UI(count_nodes_dfs(rb.root), count);

		/* check if it is still a binary tree */
		check_binary_tree_dfs(rb.root);

		/* root node must be black. Every red node
		   must have black children. */
		TEST_ASSERT(!rb.root->is_red);
		check_colors_dfs(rb.root);

		/* every path from the root to a leave must have
		   the same number of black nodes. */
		blkdepth = get_ref_black_depth(&rb);
		check_black_depth_dfs(rb.root, blkdepth, 0);

		/* longest root to leaf path must be at most
		   twice as long as the shortest. */
		mind = min_depth(rb.root);
		maxd = max_depth(rb.root);
		TEST_ASSERT(maxd <= mind * 2);

		/* lookup of current key must work after insert */
		n = rbtree_lookup(&rb, &key);
		TEST_NOT_NULL(n);
		value = *((sqfs_u64 *)rbtree_node_value(n));
		TEST_EQUAL_UI((sqfs_u64)(key + 10000), value);
	}

	/* test if copy works */
	ret = rbtree_copy(&rb, &copy);
	TEST_EQUAL_I(ret, 0);

	TEST_EQUAL_UI(rb.key_size, copy.key_size);
	TEST_EQUAL_UI(rb.key_size_padded, copy.key_size_padded);
	TEST_EQUAL_UI(rb.value_size, copy.value_size);
	TEST_ASSERT(rb.key_compare == copy.key_compare);
	TEST_ASSERT(rb.root != copy.root);

	ret = check_subtrees_equal(rb.root, copy.root,
				   rb.key_size_padded + rb.value_size);
	TEST_EQUAL_I(ret, 0);

	/* cleanup */
	rbtree_cleanup(&rb);
	rbtree_cleanup(&copy);
	return EXIT_SUCCESS;
}