[dv, otbn] Adust the `scramble_test` to tolerate the right amount of false positives
Signed-off-by: Douglas Reis <doreis@lowrisc.org>
diff --git a/sw/device/tests/otbn_mem_scramble_test.c b/sw/device/tests/otbn_mem_scramble_test.c
index 39532b8..31bc85e 100644
--- a/sw/device/tests/otbn_mem_scramble_test.c
+++ b/sw/device/tests/otbn_mem_scramble_test.c
@@ -22,19 +22,59 @@
uint32_t offset_bytes, const void *src,
size_t len_bytes);
-/**
- * Number of distinct addresses to check in each IMEM and DMEM.
- */
-static const int kNumAddrs = 50;
+enum {
+ /**
+ * Number of distinct addresses to check in each IMEM and DMEM.
+ */
+ kNumAddrs = 50,
-/**
- * Minimum number of expected integrity errors in IMEM and DMEM after
- * re-scrambling.
- *
- * Note that there is a non-zero chance for every word that the integrity bits
- * after re-scrambling are still valid.
- */
-static const int kNumIntgErrorsThreshold = kNumAddrs - 1;
+ /**
+ * Minimum number of expected integrity errors in IMEM and DMEM after
+ * re-scrambling.
+ * Note that there are `2^32` valid code words and that each non-valid code
+ * word triggers an error. Therefore, the probability that a random 39-bit
+ * word triggers an error is: `(2^39 - 2^32)/ 2^39 = 127/128`. Then the
+ * probability that all `kNumAddrs` triggers an errors is
+ * `(127/128)^kNumAddrs` after re-scrambling.
+ *
+ * The Generic formula:
+ * (w-i)
+ * 127
+ * Pr(i) = -------- x (w choose i)
+ * w
+ * 128
+ * Where:
+ * w = The number of words tested.
+ * i = The number of words that may not generate errors.
+ * Pr(i) = Probability that i words will not generate an ECC error.
+ *
+ * So for i in (0..4):
+ *
+ * ``` Python
+ * from math import comb
+ * w = 50
+ * t = 0
+ * for i in range(5):
+ * p = ((127**(w-i))/(128**w)) * comb(w,i)
+ * t += p
+ * print(f'Pr({i}): { round(p, 4)},\tsum{{Pr(0-{i})}}: {round(t, 6)}')
+ * ```
+ * ```
+ * Pr(0): 0.6756, sum{Pr(0-0)}: 0.675597
+ * Pr(1): 0.266, sum{Pr(0-1)}: 0.94158
+ * Pr(2): 0.0513, sum{Pr(0-2)}: 0.992891
+ * Pr(3): 0.0065, sum{Pr(0-3)}: 0.999356
+ * Pr(4): 0.0006, sum{Pr(0-4)}: 0.999954
+ * ```
+ * So by choosing `(kNumAddrs - 2) = 48` as the threshold we will have a
+ * probability of `1 - 0.992891 = 0.71%` that this test will fail randomly due
+ * to ECC errors not being generated. That seems a reasonable number.
+ */
+ kNumIntgErrorsThreshold = kNumAddrs - 2,
+};
+static_assert(kNumAddrs == 50,
+ "kNumAddrs changed, so kEccErrorProbability should be "
+ "computed again");
static volatile bool has_irq_fired;