Skip to content

Commit 2bd44c8

Browse files
authored
[Verif] Add ignore attribute to formal (#7719)
By default, `circt-test` will not include "ignored" verif.formal ops in the output list but has an option `--list-ignored` to emit them.
1 parent 6405aa7 commit 2bd44c8

File tree

3 files changed

+93
-10
lines changed

3 files changed

+93
-10
lines changed

integration_test/circt-test/basic.mlir

+37-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
// RUN: circt-test %s -r circt-test-runner-sby.py | FileCheck %s
22
// REQUIRES: sby
33

4-
// CHECK: all 6 tests passed
4+
// CHECK: 1 tests FAILED, 6 passed, 1 ignored
55

66
hw.module @FullAdder(in %a: i1, in %b: i1, in %ci: i1, out s: i1, out co: i1) {
77
%0 = comb.xor %a, %b : i1
@@ -113,3 +113,39 @@ verif.formal @ALUWorks {
113113
%eq = comb.icmp eq %z0, %z1 : i4
114114
verif.assert %eq : i1
115115
}
116+
117+
verif.formal @ALUIgnoreFailure attributes {ignore = true} {
118+
%a = verif.symbolic_value : i4
119+
%b = verif.symbolic_value : i4
120+
%sub = verif.symbolic_value : i1
121+
122+
// Custom ALU implementation.
123+
%z0 = hw.instance "dut" @ALU(a: %a: i4, b: %b: i4, sub: %sub: i1) -> (z: i4)
124+
125+
// Reference add/sub function.
126+
%ref_add = comb.add %a, %b : i4
127+
%ref_sub = comb.sub %a, %b : i4
128+
%z1 = comb.mux %sub, %ref_sub, %ref_add : i4
129+
130+
// Check the two don't match (failure will be ignored).
131+
%ne = comb.icmp ne %z0, %z1 : i4
132+
verif.assert %ne : i1
133+
}
134+
135+
verif.formal @ALUFailure {
136+
%a = verif.symbolic_value : i4
137+
%b = verif.symbolic_value : i4
138+
%sub = verif.symbolic_value : i1
139+
140+
// Custom ALU implementation.
141+
%z0 = hw.instance "dut" @ALU(a: %a: i4, b: %b: i4, sub: %sub: i1) -> (z: i4)
142+
143+
// Reference add/sub function.
144+
%ref_add = comb.add %a, %b : i4
145+
%ref_sub = comb.sub %a, %b : i4
146+
%z1 = comb.mux %sub, %ref_sub, %ref_add : i4
147+
148+
// Check the two don't match (should fail).
149+
%ne = comb.icmp ne %z0, %z1 : i4
150+
verif.assert %ne : i1
151+
}

test/circt-test/basic.mlir

+13-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
// RUN: circt-test -l %s | FileCheck %s
22
// RUN: circt-test -l --json %s | FileCheck --check-prefix=JSON %s
33
// RUN: circt-as %s -o - | circt-test -l | FileCheck %s
4+
// RUN: circt-test -l %s --list-ignored | FileCheck --check-prefix=CHECK-WITH-IGNORED %s
45

56
// JSON: [
67

@@ -23,6 +24,7 @@ verif.formal @Some.TestB {}
2324
// JSON-NEXT: "attrs": {
2425
// JSON-NEXT: "awesome": true
2526
// JSON-NEXT: "engine": "bmc"
27+
// JSON-NEXT: "ignore": false
2628
// JSON-NEXT: "offset": 42
2729
// JSON-NEXT: "tags": [
2830
// JSON-NEXT: "sby"
@@ -31,13 +33,22 @@ verif.formal @Some.TestB {}
3133
// JSON-NEXT: "wow": false
3234
// JSON-NEXT: }
3335
// JSON-NEXT: }
34-
// CHECK: Attrs formal {awesome = true, engine = "bmc", offset = 42 : i64, tags = ["sby", "induction"], wow = false}
36+
// CHECK: Attrs formal {awesome = true, engine = "bmc", ignore = false, offset = 42 : i64, tags = ["sby", "induction"], wow = false}
3537
verif.formal @Attrs attributes {
3638
awesome = true,
3739
engine = "bmc",
3840
offset = 42 : i64,
3941
tags = ["sby", "induction"],
40-
wow = false
42+
wow = false,
43+
ignore = false
44+
} {}
45+
46+
// CHECK-NOT: "name": "Ignore"
47+
// JSON-NOT: "name": "Ignore"
48+
// CHECK-WITH-IGNORED: Ignore formal {another = "attr", ignore = true}
49+
verif.formal @Ignore attributes {
50+
ignore = true,
51+
another = "attr"
4152
} {}
4253

4354
// JSON: ]

tools/circt-test/circt-test.cpp

+43-7
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,9 @@ struct Options {
7171
cl::opt<bool> json{"json", cl::desc("Emit test list as JSON array"),
7272
cl::init(false), cl::cat(cat)};
7373

74+
cl::opt<bool> listIgnored{"list-ignored", cl::desc("List ignored tests"),
75+
cl::init(false), cl::cat(cat)};
76+
7477
cl::opt<std::string> resultDir{
7578
"d", cl::desc("Result directory (default `.circt-test`)"),
7679
cl::value_desc("dir"), cl::init(".circt-test"), cl::cat(cat)};
@@ -107,6 +110,8 @@ class Test {
107110
/// An optional location indicating where this test was discovered. This can
108111
/// be the location of an MLIR op, or a line in some other source file.
109112
LocationAttr loc;
113+
/// Whether or not the test should be ignored
114+
bool ignore;
110115
/// The user-defined attributes of this test.
111116
DictionaryAttr attrs;
112117
};
@@ -120,7 +125,10 @@ class TestSuite {
120125
/// The tests discovered in the input.
121126
std::vector<Test> tests;
122127

123-
TestSuite(MLIRContext *context) : context(context) {}
128+
bool listIgnored;
129+
130+
TestSuite(MLIRContext *context, bool listIgnored)
131+
: context(context), listIgnored(listIgnored) {}
124132
void discoverInModule(ModuleOp module);
125133
};
126134
} // namespace
@@ -141,6 +149,10 @@ void TestSuite::discoverInModule(ModuleOp module) {
141149
test.name = op.getSymNameAttr();
142150
test.kind = TestKind::Formal;
143151
test.loc = op.getLoc();
152+
if (auto boolAttr = op->getAttrOfType<BoolAttr>("ignore"))
153+
test.ignore = boolAttr.getValue();
154+
else
155+
test.ignore = false;
144156
test.attrs = op->getDiscardableAttrDictionary();
145157
tests.push_back(std::move(test));
146158
});
@@ -150,6 +162,11 @@ void TestSuite::discoverInModule(ModuleOp module) {
150162
// Tool Implementation
151163
//===----------------------------------------------------------------------===//
152164

165+
// Check if test should be included in output listing
166+
bool ignoreTestListing(Test &test, TestSuite &suite) {
167+
return !suite.listIgnored && test.ignore;
168+
}
169+
153170
/// List all the tests in a given module.
154171
static LogicalResult listTests(TestSuite &suite) {
155172
// Open the output file for writing.
@@ -164,6 +181,8 @@ static LogicalResult listTests(TestSuite &suite) {
164181
json.arrayBegin();
165182
auto guard = make_scope_exit([&] { json.arrayEnd(); });
166183
for (auto &test : suite.tests) {
184+
if (ignoreTestListing(test, suite))
185+
continue;
167186
json.objectBegin();
168187
auto guard = make_scope_exit([&] { json.objectEnd(); });
169188
json.attribute("name", test.name.getValue());
@@ -182,13 +201,22 @@ static LogicalResult listTests(TestSuite &suite) {
182201
}
183202

184203
// Handle regular text output.
185-
for (auto &test : suite.tests)
204+
for (auto &test : suite.tests) {
205+
if (ignoreTestListing(test, suite))
206+
continue;
186207
output->os() << test.name.getValue() << " " << toString(test.kind) << " "
187208
<< test.attrs << "\n";
209+
}
188210
output->keep();
189211
return success();
190212
}
191213

214+
void reportIgnored(unsigned numIgnored) {
215+
if (numIgnored > 0)
216+
WithColor(llvm::errs(), raw_ostream::SAVEDCOLOR, true).get()
217+
<< ", " << numIgnored << " ignored";
218+
}
219+
192220
/// Entry point for the circt-test tool. At this point an MLIRContext is
193221
/// available, all dialects have been registered, and all command line options
194222
/// have been parsed.
@@ -202,7 +230,7 @@ static LogicalResult execute(MLIRContext *context) {
202230
return failure();
203231

204232
// Discover all tests in the input.
205-
TestSuite suite(context);
233+
TestSuite suite(context, opts.listIgnored);
206234
suite.discoverInModule(*module);
207235
if (suite.tests.empty()) {
208236
llvm::errs() << "no tests discovered\n";
@@ -254,7 +282,12 @@ static LogicalResult execute(MLIRContext *context) {
254282

255283
// Run the tests.
256284
std::atomic<unsigned> numPassed(0);
285+
std::atomic<unsigned> numIgnored(0);
257286
mlir::parallelForEach(context, suite.tests, [&](auto &test) {
287+
if (test.ignore) {
288+
++numIgnored;
289+
return;
290+
}
258291
// Create the directory in which we are going to run the test.
259292
SmallString<128> testDir(opts.resultDir);
260293
llvm::sys::path::append(testDir, test.name.getValue());
@@ -302,18 +335,21 @@ static LogicalResult execute(MLIRContext *context) {
302335
});
303336

304337
// Print statistics about how many tests passed and failed.
305-
assert(numPassed <= suite.tests.size());
306-
unsigned numFailed = suite.tests.size() - numPassed;
338+
assert((numPassed + numIgnored) <= suite.tests.size());
339+
unsigned numFailed = suite.tests.size() - numPassed - numIgnored;
307340
if (numFailed > 0) {
308341
WithColor(llvm::errs(), raw_ostream::SAVEDCOLOR, true).get()
309342
<< numFailed << " tests ";
310343
WithColor(llvm::errs(), raw_ostream::RED, true).get() << "FAILED";
311-
llvm::errs() << ", " << numPassed << " passed\n";
344+
llvm::errs() << ", " << numPassed << " passed";
345+
reportIgnored(numIgnored);
346+
llvm::errs() << "\n";
312347
return failure();
313348
}
314349
WithColor(llvm::errs(), raw_ostream::SAVEDCOLOR, true).get()
315-
<< "all " << numPassed << " tests ";
350+
<< numPassed << " tests ";
316351
WithColor(llvm::errs(), raw_ostream::GREEN, true).get() << "passed";
352+
reportIgnored(numIgnored);
317353
llvm::errs() << "\n";
318354
return success();
319355
}

0 commit comments

Comments
 (0)