/* Copyright 2016-2017 Tobias Grosser * * Use of this software is governed by the MIT license * * Written by Tobias Grosser, Weststrasse 47, CH-8003, Zurich */ #include #include #include #include #include #include #include namespace isl { using namespace checked; } static void assert_impl(bool condition, const char *file, int line, const char *message) { if (condition) return; fprintf(stderr, "Assertion failed in %s:%d %s\n", file, line, message); exit(EXIT_FAILURE); } static void assert_impl(isl::boolean condition, const char *file, int line, const char *message) { assert_impl(bool(condition), file, line, message); } /* Return the value encapsulated by "s". */ static int size_val(isl::size s) { return s.is_error() ? -1 : unsigned(s); } #undef assert #define assert(exp) assert_impl(exp, __FILE__, __LINE__, #exp) #define IS_TRUE(b) (b).is_true() #define SIZE_VAL(s) size_val(s) #include "isl_test_cpp-generic.cc" /* Test that isl_bool values are returned correctly. * * We check in detail the following parts of the isl::boolean class: * - The is_true, is_false, and is_error functions return true in case they * are called on a true, false, or error instance of isl::boolean, * respectively * - Explicit conversion to 'bool' * - Implicit conversion to 'bool' * - The complement operator * - Explicit construction from 'true' and 'false' * - Explicit construction form isl_bool */ void test_return_bool(isl::ctx ctx) { isl::set empty(ctx, "{ : false }"); isl::set univ(ctx, "{ : }"); isl::set null; isl::boolean b_true = empty.is_empty(); isl::boolean b_false = univ.is_empty(); isl::boolean b_error = null.is_empty(); assert(b_true.is_true()); assert(!b_true.is_false()); assert(!b_true.is_error()); assert(!b_false.is_true()); assert(b_false.is_false()); assert(!b_false.is_error()); assert(!b_error.is_true()); assert(!b_error.is_false()); assert(b_error.is_error()); assert(bool(b_true) == true); assert(bool(b_false) == false); assert(b_true); assert((!b_false).is_true()); assert((!b_true).is_false()); assert((!b_error).is_error()); assert(isl::boolean(true).is_true()); assert(!isl::boolean(true).is_false()); assert(!isl::boolean(true).is_error()); assert(isl::boolean(false).is_false()); assert(!isl::boolean(false).is_true()); assert(!isl::boolean(false).is_error()); assert(isl::manage(isl_bool_true).is_true()); assert(!isl::manage(isl_bool_true).is_false()); assert(!isl::manage(isl_bool_true).is_error()); assert(isl::manage(isl_bool_false).is_false()); assert(!isl::manage(isl_bool_false).is_true()); assert(!isl::manage(isl_bool_false).is_error()); assert(isl::manage(isl_bool_error).is_error()); assert(!isl::manage(isl_bool_error).is_true()); assert(!isl::manage(isl_bool_error).is_false()); } /* Test that return values are handled correctly. * * Test that isl C++ objects, integers, boolean values, and strings are * returned correctly. */ void test_return(isl::ctx ctx) { test_return_obj(ctx); test_return_int(ctx); test_return_bool(ctx); test_return_string(ctx); } /* Test that foreach functions are modeled correctly. * * Verify that lambdas are correctly called as callback of a 'foreach' * function and that variables captured by the lambda work correctly. Also * check that the foreach function takes account of the return value of the * lambda and aborts in case isl::stat::error is returned and then returns * isl::stat::error itself. */ void test_foreach(isl::ctx ctx) { isl::set s(ctx, "{ [0]; [1]; [2] }"); std::vector basic_sets; auto add_to_vector = [&] (isl::basic_set bs) { basic_sets.push_back(bs); return isl::stat::ok(); }; isl::stat ret1 = s.foreach_basic_set(add_to_vector); assert(ret1.is_ok()); assert(basic_sets.size() == 3); assert(isl::set(basic_sets[0]).is_subset(s).is_true()); assert(isl::set(basic_sets[1]).is_subset(s).is_true()); assert(isl::set(basic_sets[2]).is_subset(s).is_true()); assert(!basic_sets[0].is_equal(basic_sets[1]).is_true()); auto fail = [&] (isl::basic_set bs) { return isl::stat::error(); }; isl::stat ret2 = s.foreach_basic_set(fail); assert(ret2.is_error()); } /* Test the functionality of "every" functions. * * In particular, test the generic functionality and * test that error conditions are properly propagated. */ static void test_every(isl::ctx ctx) { isl::union_set us(ctx, "{ A[i]; B[j] }"); test_every_generic(ctx); auto fail = [] (isl::set s){ return isl::boolean::error(); }; assert(us.every_set(fail).is_error()); } /* Test basic schedule tree functionality. * * In particular, create a simple schedule tree and * - perform some generic tests * - test map_descendant_bottom_up in the failing case * - test foreach_descendant_top_down * - test every_descendant */ static void test_schedule_tree(isl::ctx ctx) { auto root = test_schedule_tree_generic(ctx); auto fail_map = [](isl::schedule_node node) { return isl::schedule_node(); }; assert(root.map_descendant_bottom_up(fail_map).is_null()); int count = 0; auto inc_count = [&count](isl::schedule_node node) { count++; return isl::boolean(true); }; assert(root.foreach_descendant_top_down(inc_count).is_ok()); assert(count == 8); count = 0; auto inc_count_once = [&count](isl::schedule_node node) { count++; return isl::boolean(false); }; assert(root.foreach_descendant_top_down(inc_count_once).is_ok()); assert(count == 1); auto is_not_domain = [](isl::schedule_node node) { return !node.isa(); }; assert(root.child(0).every_descendant(is_not_domain).is_true()); assert(root.every_descendant(is_not_domain).is_false()); auto fail = [](isl::schedule_node node) { return isl::boolean(); }; assert(root.every_descendant(fail).is_error()); auto domain = root.as().domain(); auto filters = isl::union_set(ctx, "{}"); auto collect_filters = [&filters](isl::schedule_node node) { if (node.isa().is_true()) { auto filter = node.as(); filters = filters.unite(filter.filter()); } return isl::boolean(true); }; assert(!root.every_descendant(collect_filters).is_error()); assert(domain.is_equal(filters).is_true()); } /* Test basic AST generation from a schedule tree. * * In particular, create a simple schedule tree and * - perform some generic tests * - test at_each_domain in the failing case */ static void test_ast_build(isl::ctx ctx) { auto schedule = test_ast_build_generic(ctx); bool do_fail = true; int count_ast_fail = 0; auto fail_inc_count_ast = [&count_ast_fail, &do_fail](isl::ast_node node, isl::ast_build build) { count_ast_fail++; return do_fail ? isl::ast_node() : node; }; auto build = isl::ast_build(ctx); build = build.set_at_each_domain(fail_inc_count_ast); auto ast = build.node_from(schedule); assert(ast.is_null()); assert(count_ast_fail > 0); auto build_copy = build; int count_ast = 0; auto inc_count_ast = [&count_ast](isl::ast_node node, isl::ast_build build) { count_ast++; return node; }; build_copy = build_copy.set_at_each_domain(inc_count_ast); ast = build_copy.node_from(schedule); assert(!ast.is_null()); assert(count_ast == 2); count_ast_fail = 0; do_fail = false; ast = build.node_from(schedule); assert(!ast.is_null()); assert(count_ast_fail == 2); } /* Test the isl checked C++ interface * * This includes: * - The isl C <-> C++ pointer interface * - Object construction * - Different parameter types * - Different return types * - Foreach functions * - Every functions * - Spaces * - Schedule trees * - AST generation * - AST expression generation */ int main() { isl_ctx *ctx = isl_ctx_alloc(); isl_options_set_on_error(ctx, ISL_ON_ERROR_ABORT); test_pointer(ctx); test_constructors(ctx); test_parameters(ctx); test_return(ctx); test_foreach(ctx); test_every(ctx); test_space(ctx); test_schedule_tree(ctx); test_ast_build(ctx); test_ast_build_expr(ctx); isl_ctx_free(ctx); return EXIT_SUCCESS; }