-
Notifications
You must be signed in to change notification settings - Fork 795
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
[dd, aon_timer cov_closure] CCOV exclusion file for aon_timer #25705
base: master
Are you sure you want to change the base?
[dd, aon_timer cov_closure] CCOV exclusion file for aon_timer #25705
Conversation
Block 2 "2643129081" "assign data_intg = tl_i.d_user.data_intg;" | ||
CHECKSUM: "2815520955 1214062232" | ||
INSTANCE: tb.dut.u_reg.u_wkup_count_hi_cdc.u_arb | ||
ANNOTATION: "Default statement can't be hit unless some wrong value is forced in the `state_d` signal" |
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.
Hmm. It's a 2-bit signal whose higher bit is constant zero! The code comes from 23d3bb5 and I wonder whether Tim was thinking about safety against fault injection (in which case we probably would have wanted values 01 and 10).
@vogelpi: As a security expert who might have a bit more background on this, what do you think is supposed to happen here? Are we worried about fault injection? (If so, I claim we should change the enum encoding). If not, maybe this should be a single bit?
91fbc3f
to
268e0fd
Compare
Branch 4 "3547459906" "gen_wr_req.state_q" (6) "gen_wr_req.state_q default,-,-,-,-" | ||
CHECKSUM: "530940107 4102526809" | ||
INSTANCE: tb.dut | ||
ANNOTATION: "intr_test_flds_we[0/1] is asserted at the same time whenever the intr_test register is written, so the combination of only one Qe being set is not possible." |
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.
Hmm: I think this is an example where it would make more sense to make this explicit in the RTL. I'm imagining something like just:
assign intr_test_qe = reg2hw.intr_test.wkup_timer_expired.qe;
But with an assertion to make sure we don't accidentally change things in the future:
`ASSERT(IntrTestFieldsQeMatch_A,
reg2hw.intr_test.wkup_timer_expired.qe == reg2hw.intr_test.wdog_timer_bark.qe)
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.
Can you try to select the lines you're referring to? I have a hard time reading this exclusion file otherwise haha.
I think you mean lines 50-54?
Your change would definitely make closing coverage easier. However, are we confident the register generator (or spec?) will stay as is? I'm not familiar with the tool. But we if we grew the interrupt vector, would that mean that aon_timer.intr_test_q
would be ORed with all the separate intr_test
fiedls? Is there a chance that if the intr_test
was greater than 8-bit the RTL may generate a per-byte Qe?
I guess your assertion would catch the difference if that will ever happen.
Also, should we have a CCOV closure label for assertions that are just a logic check due to simplifying the logic to make closing coverage easier?
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 was talking about the lines that are referred to by this annotation. As you say, that's the CONDITIONs on lines 53 and 54.
I'm not confident! Which is why I suggested adding an assertion :-)
I'm not sure what you mean about closure labels? The easiest approach is probably
- Make the RTL change as a PR
- Run coverage collection locally with your change and keep working(!)
- Ignore this coverage hole: it will disappear once the RTL change lands.
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've pushed the changes in a new PR with your suggestion. PR is #25811
I'll push the changes with the amended exclusion file in a sec
Condition 4 "594611319" "(reg2hw.intr_test.wkup_timer_expired.qe | reg2hw.intr_test.wdog_timer_bark.qe) 1 -1" (2 "01") | ||
CHECKSUM: "1125143680 279071062" | ||
INSTANCE: tb.dut.u_reg.u_wkup_count_hi_cdc | ||
ANNOTATION: "src_ack is only 1 cycle pulse, and is an acknowledgment to a request from a read/write. |
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 another example where I think an RTL change is the right move. As your note says, there's even an assertion that checks the behaviour won't change :-)
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 RTL change is in PR #25812
I'll push the amended exclusion file in a sec
All the exclusions in this file are unreachable Signed-off-by: Antonio Martinez Zambrana <amz@lowrisc.org>
268e0fd
to
22abc8c
Compare
Condition 3 "86646938" "(src_busy_q && src_ack) 1 -1" (1 "01") | ||
CHECKSUM: "2815520955 4109606122" | ||
INSTANCE: tb.dut.u_reg.u_wkup_count_hi_cdc.u_arb | ||
ANNOTATION: "It can't be hit because dst_update_ack won't be set prior to dst_update_req" |
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'm not sure that's quite correct. These signals come through prim_sync_reqack
, where dst_update_req
is visible as src_req_i
and dst_update_ack
is visible as src_ack_o
.
I was slightly surprised that this module doesn't have an assumption that says src_ack_o -> src_req_i
. But I don't think it's true! If src_req_i
is asserted and acknowledged, what happens when src_req_i
drops again?
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 don't think dst_update_req
can't be unset until the dst_update_ack
is set just based on how this FSM is working:
opentitan/hw/ip/prim/rtl/prim_reg_cdc_arb.sv
Lines 186 to 228 in 7780c6b
always_comb begin | |
state_d = state_q; | |
dst_hold_req = '0; | |
// depending on when the request is received, we | |
// may latch d or q. | |
dst_lat_q = '0; | |
dst_lat_d = '0; | |
busy = 1'b1; | |
unique case (state_q) | |
StIdle: begin | |
busy = '0; | |
if (dst_req) begin | |
// there's a software issued request for change | |
state_d = StWait; | |
dst_lat_d = 1'b1; | |
end else if (dst_update) begin | |
state_d = StWait; | |
dst_lat_d = 1'b1; | |
end else if (dst_qs_o != dst_qs_i) begin | |
// there's a direct destination update | |
// that was blocked by an ongoing transaction | |
state_d = StWait; | |
dst_lat_q = 1'b1; | |
end | |
end | |
StWait: begin | |
dst_hold_req = 1'b1; | |
if (dst_update_ack) begin | |
state_d = StIdle; | |
end | |
end | |
default: begin | |
state_d = StIdle; | |
end | |
endcase // unique case (state_q) | |
end // always_comb | |
assign dst_update_req = dst_hold_req | dst_lat_d | dst_lat_q; |
Do you disagree?
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.
Looking now, I think I agree. But I guess the prim_sync_reqack
module still couldn't have that assertion because it depends on the code around where the module is instantiated.
ANNOTATION: "It would need dst_req_i=1, dst_req_q=1, busy=1. However, that's not possible. | ||
|
||
The FSM stays in IDLE until there's a request, and then stays in StWait 1-cycle. During StWait, dst_req_q is set to 0x1, but we are also transitioning back to StIdle where busy will be set to 0x0 making the missing condition coverage unhittable. |
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 think we can see this happen?
-
Suppose we are sitting in
StIdle
and nothing is going on. -
At cycle 0, suppose that
dst_req_i
is high. -
dst_req_q
will get a nonblocking assignment of1'b1
. -
Also (because
dst_req_i
is high)dst_req
will be high too. -
As a result,
state_d
will beStWait
. -
On cycle 1,
state_q
will beStWait
-
As a result,
busy
will be true. -
Also,
dst_req_q
will be 1.
So we'll see a hit unless dst_req_i
has dropped.
Is that correct? (And is it possible for this to happen?)
Condition 2 "1682497780" "(dst_req_i && ((!gen_wr_req.dst_req_q)) && gen_wr_req.busy) 1 -1" (2 "101") | ||
CHECKSUM: "2815520955 4109606122" | ||
INSTANCE: tb.dut.u_reg.u_wkup_count_lo_cdc.u_arb | ||
ANNOTATION: "It can't be hit because dst_update_ack won't be set prior to dst_update_req" |
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 think this is the same comment as the one for line 117?
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've created a PR for this: #25814
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 know that I haven't got to the bottom of the list, but I think there are a few items to work through so far :-)
22abc8c
to
71c55ec
Compare
Signed-off-by: Antonio Martinez Zambrana <amz@lowrisc.org>
71c55ec
to
d2a959e
Compare
All the exclusions in this file are unreachable