Skip to content

Commit 28a82db

Browse files
authoredFeb 14, 2025··
[Verif] Add simulation op (#8224)
Add the `verif.simulation` op as an entry point for simulations. The op works in a fashion similar to `verif.formal`, but encodes a simulation run instead of a formal check.
1 parent 349990b commit 28a82db

File tree

4 files changed

+201
-8
lines changed

4 files changed

+201
-8
lines changed
 

‎include/circt/Dialect/Verif/VerifOps.td

+108-7
Original file line numberDiff line numberDiff line change
@@ -9,17 +9,18 @@
99
#ifndef CIRCT_DIALECT_VERIF_VERIFOPS_TD
1010
#define CIRCT_DIALECT_VERIF_VERIFOPS_TD
1111

12-
include "circt/Dialect/Verif/VerifDialect.td"
13-
include "circt/Dialect/Verif/VerifOpInterfaces.td"
12+
include "circt/Dialect/HW/HWAttributes.td"
13+
include "circt/Dialect/HW/HWTypes.td"
1414
include "circt/Dialect/LTL/LTLDialect.td"
1515
include "circt/Dialect/LTL/LTLTypes.td"
16-
include "circt/Dialect/HW/HWTypes.td"
1716
include "circt/Dialect/Seq/SeqTypes.td"
18-
include "circt/Dialect/HW/HWAttributes.td"
17+
include "circt/Dialect/Verif/VerifDialect.td"
18+
include "circt/Dialect/Verif/VerifOpInterfaces.td"
1919
include "mlir/Interfaces/InferTypeOpInterface.td"
2020
include "mlir/Interfaces/SideEffectInterfaces.td"
2121
include "mlir/IR/AttrTypeBase.td"
2222
include "mlir/IR/EnumAttr.td"
23+
include "mlir/IR/OpAsmInterface.td"
2324
include "mlir/IR/PatternBase.td"
2425
include "mlir/IR/RegionKindInterface.td"
2526
include "mlir/IR/SymbolInterfaces.td"
@@ -255,8 +256,9 @@ def LogicEquivalenceCheckingOp : VerifOp<"lec", [
255256
def YieldOp : VerifOp<"yield", [
256257
Terminator,
257258
ParentOneOf<[
258-
"verif::LogicEquivalenceCheckingOp", "verif::BoundedModelCheckingOp",
259-
"verif::ContractOp"
259+
"verif::BoundedModelCheckingOp",
260+
"verif::LogicEquivalenceCheckingOp",
261+
"verif::SimulationOp",
260262
]>
261263
]> {
262264
let summary = "yields values from a region";
@@ -272,11 +274,12 @@ def YieldOp : VerifOp<"yield", [
272274
}
273275

274276
//===----------------------------------------------------------------------===//
275-
// Formal Test
277+
// Verification Entry Points
276278
//===----------------------------------------------------------------------===//
277279

278280
def FormalOp : VerifOp<"formal", [
279281
IsolatedFromAbove,
282+
NoRegionArguments,
280283
NoTerminator,
281284
RegionKindInterface,
282285
SingleBlock,
@@ -346,6 +349,104 @@ def FormalOp : VerifOp<"formal", [
346349
}];
347350
}
348351

352+
def SimulationOp : VerifOp<"simulation", [
353+
DeclareOpInterfaceMethods<OpAsmOpInterface, ["getAsmBlockArgumentNames"]>,
354+
IsolatedFromAbove,
355+
RegionKindInterface,
356+
SingleBlockImplicitTerminator<"verif::YieldOp">,
357+
Symbol,
358+
]> {
359+
let summary = "A simulation unit test";
360+
let description = [{
361+
This operation defines a simulation unit test that can be automatically run
362+
by various tools. To describe a test, the body of this op should contain the
363+
hardware to be tested, alongside any necessary forms of stimulus generation.
364+
365+
### Inputs
366+
367+
The body has two block arguments as input values: a "clock" signal of type
368+
`!seq.clock` and an "init" signal of type `i1`. The clock signal starts at 0
369+
and continuously toggles between 0 and 1 throughout the simulation. The init
370+
signal starts at 1, remains 1 during a single 0-to-1 transition of the
371+
clock, and then drops to 0 for the remainder of the simulation.
372+
373+
### Outputs
374+
375+
The body must have a `verif.yield` terminator op with exactly two operands:
376+
377+
The first operand is a "done" signal of type `i1` which indicates the end of
378+
the simulation. The simulation stops when the done signal is 1 during a
379+
0-to-1 transition of the clock. No additional clock toggles occur once done
380+
has been sampled as 1.
381+
382+
The second operand is an "exit code" signal which may be of any integer
383+
type. It indicates success of a test as 0, or failure as any non-zero
384+
integer. The exit code is sampled at the same time as the done signal. Only
385+
the zero versus non-zero value distinction is relevant to indicate
386+
simulation success and must be preserved. The exact non-zero value is for
387+
informational purposes only and may not be reproduced accurately by
388+
simulators. Simulators may in the worst case map all non-zero values to 1,
389+
reducing the exit code to a single bit indicating failure or success.
390+
391+
### Schedule
392+
393+
The clock and init values adhere to the following schedule during
394+
simulation:
395+
396+
| Time | Clock | Init | Comment
397+
|-----------|-------|-------| ----
398+
| t0 (>=0s) | undef | undef | Clock and init may initially be undefined.
399+
| t1 (>=t0) | 0 | 1 | Initialization code (e.g., `seq.initial`, Verilog `initial` procedures) may run before or after clock and init change to their initial value.
400+
| t2 (>t1) | 1 | 1 | Single rising clock edge occurs while init is high.
401+
| t3 (>t2) | 0 | 1 | Init may stay during the falling clock edge.
402+
| t4 (>=t3) | 0 | 0 | Init goes to 0 before second rising clock edge.
403+
| t5 (>t4) | 1 | 0 | Clock toggles continue indefinitely.
404+
| t6 (>t5) | 0 | 0 |
405+
406+
Simulation termination occurs when the done signal is 1 during a 0-to-1
407+
transition of the clock.
408+
409+
### Example
410+
```
411+
verif.simulation @AdderTest {myParam = 42, myTag = "hello"} {
412+
^bb0(%clock: !seq.clock, %init: i1):
413+
// Count the first 9001 simulation cycles.
414+
%c0_i19 = hw.constant 0 : i19
415+
%c1_i19 = hw.constant 1 : i19
416+
%c9001_i19 = hw.constant 9001 : i19
417+
%count = seq.compreg %0, %clock reset %init, %c0_i19 : i19
418+
%done = comb.icmp eq %count, %c9001_i19 : i19
419+
%0 = comb.add %count, %c1_i19 : i19
420+
421+
// Generate inputs to the adder.
422+
%1, %2 = func.call @generateAdderInputs(%count) : (i19) -> (i42, i42)
423+
%3 = hw.instance "dut" @Adder(a: %1: i42, b: %2: i42) -> (c: i42)
424+
425+
// Check results and track failures.
426+
%4 = comb.add %1, %2 : i42
427+
%5 = comb.icmp ne %3, %4 : i42
428+
%false = hw.constant false
429+
%failure = seq.compreg %6, %clock reset %init, %false : i1
430+
%6 = comb.or %failure, %5 : i1
431+
432+
verif.yield %done, %failure : i1, i1
433+
}
434+
```
435+
}];
436+
let arguments = (ins
437+
SymbolNameAttr:$sym_name,
438+
DictionaryAttr:$parameters
439+
);
440+
let regions = (region SizedRegion<1>:$region);
441+
let assemblyFormat = "$sym_name $parameters attr-dict-with-keyword $region";
442+
let extraClassDeclaration = [{
443+
static RegionKind getRegionKind(unsigned index) {
444+
return RegionKind::Graph;
445+
}
446+
}];
447+
let hasRegionVerifier = true;
448+
}
449+
349450
def SymbolicValueOp : VerifOp<"symbolic_value", [
350451
ParentOneOf<["verif::FormalOp", "hw::HWModuleOp"]>,
351452
]>{

‎lib/Dialect/Verif/VerifOps.cpp

+31
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,37 @@ LogicalResult BoundedModelCheckingOp::verifyRegions() {
159159
return success();
160160
}
161161

162+
//===----------------------------------------------------------------------===//
163+
// SimulationOp
164+
//===----------------------------------------------------------------------===//
165+
166+
LogicalResult SimulationOp::verifyRegions() {
167+
if (getBody()->getNumArguments() != 2)
168+
return emitOpError() << "must have two block arguments";
169+
if (!isa<seq::ClockType>(getBody()->getArgument(0).getType()))
170+
return emitOpError() << "block argument #0 must be a `!seq.clock`";
171+
if (!getBody()->getArgument(1).getType().isSignlessInteger(1))
172+
return emitOpError() << "block argument #1 must be a `i1`";
173+
174+
auto *yieldOp = getBody()->getTerminator();
175+
if (yieldOp->getNumOperands() != 2)
176+
return yieldOp->emitOpError() << "must have two operands";
177+
if (!yieldOp->getOperand(0).getType().isSignlessInteger(1))
178+
return yieldOp->emitOpError() << "operand #0 must be an `i1`";
179+
if (!isa<IntegerType>(yieldOp->getOperand(1).getType()))
180+
return yieldOp->emitOpError() << "operand #1 must be an integer";
181+
182+
return success();
183+
}
184+
185+
void SimulationOp::getAsmBlockArgumentNames(Region &region,
186+
OpAsmSetValueNameFn setNameFn) {
187+
if (region.empty() || region.getNumArguments() != 2)
188+
return;
189+
setNameFn(region.getArgument(0), "clock");
190+
setNameFn(region.getArgument(1), "init");
191+
}
192+
162193
//===----------------------------------------------------------------------===//
163194
// Generated code
164195
//===----------------------------------------------------------------------===//

‎test/Dialect/Verif/basic.mlir

+12-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// RUN: circt-opt %s | circt-opt | FileCheck %s
1+
// RUN: circt-opt --verify-roundtrip %s | FileCheck %s
22

33
%true = hw.constant true
44
%s = unrealized_conversion_cast to !ltl.sequence
@@ -64,6 +64,17 @@ verif.formal @FormalTestBody {} {
6464
%0 = verif.symbolic_value : i42
6565
}
6666

67+
//===----------------------------------------------------------------------===//
68+
// Simulation Test
69+
//===----------------------------------------------------------------------===//
70+
71+
verif.simulation @EmptySimulationTest {} {
72+
^bb0(%clock: !seq.clock, %init: i1):
73+
%0 = hw.constant true
74+
%1 = hw.constant 9001 : i42
75+
verif.yield %0, %1 : i1, i42
76+
}
77+
6778
//===----------------------------------------------------------------------===//
6879
// Contracts
6980
//===----------------------------------------------------------------------===//

‎test/Dialect/Verif/errors.mlir

+50
Original file line numberDiff line numberDiff line change
@@ -181,3 +181,53 @@ verif.bmc bound 10 num_regs 1 initial_values [unit] attributes {verif.some_attr}
181181
verif.assert %true : i1
182182
verif.yield %arg0 : i32
183183
}
184+
185+
// -----
186+
187+
// expected-error @below {{op must have two block arguments}}
188+
verif.simulation @foo {} {
189+
^bb0(%arg0: !seq.clock):
190+
%true = hw.constant true
191+
verif.yield %true, %true : i1, i1
192+
}
193+
194+
// -----
195+
196+
// expected-error @below {{op block argument #0 must be a `!seq.clock`}}
197+
verif.simulation @foo {} {
198+
^bb0(%arg0: i1, %arg1: i1):
199+
verif.yield %arg0, %arg1 : i1, i1
200+
}
201+
202+
// -----
203+
204+
// expected-error @below {{op block argument #1 must be a `i1`}}
205+
verif.simulation @foo {} {
206+
^bb0(%arg0: !seq.clock, %arg1: i42):
207+
%true = hw.constant true
208+
verif.yield %true, %true : i1, i1
209+
}
210+
211+
// -----
212+
213+
verif.simulation @foo {} {
214+
^bb0(%arg0: !seq.clock, %arg1: i1):
215+
// expected-error @below {{op must have two operands}}
216+
verif.yield
217+
}
218+
219+
// -----
220+
221+
verif.simulation @foo {} {
222+
^bb0(%arg0: !seq.clock, %arg1: i1):
223+
// expected-error @below {{op operand #0 must be an `i1`}}
224+
verif.yield %arg0, %arg1 : !seq.clock, i1
225+
}
226+
227+
// -----
228+
229+
verif.simulation @foo {} {
230+
^bb0(%arg0: !seq.clock, %arg1: i1):
231+
// expected-error @below {{op operand #1 must be an integer}}
232+
verif.yield %arg1, %arg0 : i1, !seq.clock
233+
}

0 commit comments

Comments
 (0)
Please sign in to comment.