#include #include #include #include #include #include #include #include using namespace willow; TEST_CASE("valid modules", "[verifier]") { WillowContext ctx; std::vector diags; DiagnosticEngine eng([&](Diagnostic d) { diags.push_back(std::move(d)); }); auto &m = *ctx.addModule("test"); SECTION("empty module") { REQUIRE(succeeded(verifyModule(ctx, m, eng))); REQUIRE(diags.empty()); } } TEST_CASE("valid function", "[verifier]") { WillowContext ctx; std::vector diags; DiagnosticEngine eng([&](Diagnostic d) { diags.push_back(std::move(d)); }); auto &m = *ctx.addModule("test"); Type fty = ctx.types().FunctionType(ctx.types().VoidType(), {}); auto &fn = *m.emplaceFunction("fn", &m, fty); REQUIRE(succeeded(verifyFunction(ctx, fn, eng))); REQUIRE(diags.empty()); } TEST_CASE("invalid basic block", "[verifier]") { WillowContext ctx; DiagnosticEngine eng; Type i64Ty = ctx.types().IntType(64); Type voidTy = ctx.types().VoidType(); auto *one = ctx.constants().getInt(i64Ty, 1); auto &m = *ctx.addModule("test"); Type fty = ctx.types().FunctionType(ctx.types().VoidType(), {}); Type ifty = ctx.types().FunctionType(i64Ty, {}); auto &fn = *m.emplaceFunction("fn", &m, fty); auto &fn2 = *m.emplaceFunction("fn2", &m, ifty); auto *bb = fn.addBlock( std::make_unique(&fn, ctx.types().BasicBlockType())); IRBuilder builder{ctx, *bb, bb->end()}; SECTION("Empty basic block") { REQUIRE(bb->empty()); REQUIRE(failed(verifyBasicBlock(ctx, *bb, eng))); } SECTION("Basic block with no terminator") { builder.buildAdd(i64Ty, one, one); REQUIRE(failed(verifyBasicBlock(ctx, *bb, eng))); } SECTION("Teminator must be the last instruction in a basic block") { builder.buildCall(i64Ty, &fn2); builder.buildAdd(i64Ty, one, one); REQUIRE(failed(verifyBasicBlock(ctx, *bb, eng))); } SECTION("Basic block with invalid instruction") { auto *bb2 = fn.addBlock( std::make_unique(&fn, ctx.types().BasicBlockType())); builder.setInsertPoint(bb2->end()); builder.buildAdd(voidTy, one, one); builder.buildRet(); REQUIRE(failed(verifyBasicBlock(ctx, *bb, eng))); } }