-
Notifications
You must be signed in to change notification settings - Fork 309
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[ImportVerilog] Add case inside #7928
base: main
Are you sure you want to change the base?
Conversation
always_comb begin | ||
case (caseExpr) | ||
32'h1, 32'h2, 32'h3: result = 1; | ||
default: result = 0; | ||
endcase | ||
end |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Your example is not a case inside statement. Maybe the example in the SystemVerilog IEEE Std § 12.5.4 is better. What do @ankolesn think 😃?
logic [2:0] status;
always @(posedge clock)
priority case (status) inside
1, 3 : task1; // matches 'b001 and 'b011
3'b0?0, [4:7]: task2; // matches 'b000 'b010 'b0x0 'b0z0 'b100 'b101 'b110 'b111
endcase // priority case fails all other values including 'b00x 'b01x 'bxxx
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for your comments! I tried to fix it, you can check it 😃
9c679a1
to
83e9720
Compare
module priority_case_test; | ||
// CHECK: logic clock | ||
logic clock; | ||
// CHECK: logic [2:0] status | ||
logic [2:0] status; | ||
|
||
task task1; | ||
// CHECK: task1 executed | ||
$display("task1 executed at time %0t", $time); | ||
endtask | ||
|
||
task task2; | ||
// CHECK: task2 executed | ||
$display("task2 executed at time %0t", $time); | ||
endtask | ||
|
||
initial begin | ||
// CHECK: clock initialized to 0 | ||
clock = 0; | ||
// CHECK: clock toggles every 5 ns | ||
forever #5 clock = ~clock; | ||
end | ||
|
||
initial begin | ||
// CHECK: Test Case 1: status = 3'b001 | ||
status = 3'b001; | ||
@(posedge clock); | ||
|
||
// CHECK: Test Case 2: status = 3'b011 | ||
status = 3'b011; | ||
@(posedge clock); | ||
|
||
// CHECK: Test Case 3: status = 3'b010 (matches task2) | ||
status = 3'b010; | ||
@(posedge clock); | ||
|
||
// CHECK: Test Case 4: status = 3'b100 (matches task2) | ||
status = 3'b100; | ||
@(posedge clock); | ||
|
||
// CHECK: Test Case 5: status = 3'b00x (no match) | ||
status = 3'b00x; | ||
@(posedge clock); | ||
|
||
// CHECK: Test Case 6: status = 3'b111 (matches task2) | ||
status = 3'b111; | ||
@(posedge clock); | ||
|
||
// CHECK: Test Case 7: status = 3'bxxx (no match) | ||
status = 3'bxxx; | ||
@(posedge clock); | ||
|
||
// CHECK: End simulation | ||
$finish; | ||
end | ||
|
||
always @(posedge clock) begin | ||
// CHECK: Entering priority case | ||
priority case (status) | ||
// CHECK: Matches 3'b001 or 3'b011, executes task1 | ||
3'b001, 3'b011: task1; | ||
// CHECK: Matches 3'b0?0 or [3'b100:3'b111], executes task2 | ||
3'b0?0, [3'b100:3'b111]: task2; | ||
// CHECK: No match, default case | ||
default: $display("No match for status = %b at time %0t", status, $time); | ||
endcase | ||
end |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hey @ankolesn, thanks a lot for working on this! One comment regarding your unit test: we don't simulate the design as part of the unit test, so things like $display(...)
will not show up as prints in the output. Instead, the unit tests check that the expected IR operations get generated. For your PR, I'd add different examples that exercise interesting corner cases of your implementation. If you look through basic.sv
, you should find existing tests for case
statements that can give you some inspiration on how to do that. Then you can check if your code generates the correct IR ops for inside
case statements.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is the first time I've encountered this type of test, so I'm sorry that I have to upload new versions many times.
I tried to rewrite it. 😄
case CaseStatementCondition::Inside: | ||
mlir::emitError(loc, "unsupported set membership case statement"); | ||
return failure(); | ||
std::vector<Value> values; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
SmallVector per MLIR coding guidelines.
|
||
if (values.size() == 1) { | ||
cond = builder.create<moore::WildcardEqOp>(loc, caseExpr, | ||
values.front()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: why use front here and [0] two lines later?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I wanted to show that values.front() - means exactly the last element 😃
I fixed it.
return failure(); | ||
} | ||
|
||
if (values.size() == 1) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This conditional could be completely eliminated.
649b52e
to
dcf122c
Compare
dcf122c
to
bc56154
Compare
Implementation of functionality for checking whether a value belongs to a set of expressions (like inside in SystemVerilog)