xref: /xnu-11215.81.4/tools/entropy_health_test_bounds.py (revision d4514f0bc1d3f944c22d92e68b646ac3fb40d452)
1*d4514f0bSApple OSS Distributions#!/usr/bin/env python3
2*d4514f0bSApple OSS Distributions
3*d4514f0bSApple OSS Distributionsfrom fractions import Fraction
4*d4514f0bSApple OSS Distributionsfrom math import ceil
5*d4514f0bSApple OSS Distributionsfrom math import comb
6*d4514f0bSApple OSS Distributions
7*d4514f0bSApple OSS Distributions
8*d4514f0bSApple OSS Distributions# The inverse of 2, i.e. 2^-1. To be used as a base in exponentiations
9*d4514f0bSApple OSS Distributions# representing probabilities.
10*d4514f0bSApple OSS DistributionsINV2 = Fraction(1, 2)
11*d4514f0bSApple OSS Distributions
12*d4514f0bSApple OSS Distributions
13*d4514f0bSApple OSS Distributions# The probability of a false positive health test failure expressed as
14*d4514f0bSApple OSS Distributions# the negative logarithm of the *actual* probability. In simpler
15*d4514f0bSApple OSS Distributions# terms, the actual probability is:
16*d4514f0bSApple OSS Distributions#
17*d4514f0bSApple OSS Distributions# INV2 ** A
18*d4514f0bSApple OSS Distributions#
19*d4514f0bSApple OSS Distributions# It is simpler to keep this representation when computing the bound
20*d4514f0bSApple OSS Distributions# of the Repetition Count Test (below).
21*d4514f0bSApple OSS DistributionsA = 40
22*d4514f0bSApple OSS Distributions
23*d4514f0bSApple OSS Distributions
24*d4514f0bSApple OSS Distributions# The estimated min-entropy per sample in bits. Min-entropy is the
25*d4514f0bSApple OSS Distributions# negative logarithm of the probability of the *most likely* outcome.
26*d4514f0bSApple OSS Distributions#
27*d4514f0bSApple OSS Distributions# We consider this estimate to be conservative.
28*d4514f0bSApple OSS DistributionsH = 1
29*d4514f0bSApple OSS Distributions
30*d4514f0bSApple OSS Distributions
31*d4514f0bSApple OSS Distributions# The probability of the most likely outcome occurring in a given
32*d4514f0bSApple OSS Distributions# sample. This derives from the definition of min-entropy (see above).
33*d4514f0bSApple OSS DistributionsP = INV2 ** H
34*d4514f0bSApple OSS Distributions
35*d4514f0bSApple OSS Distributions
36*d4514f0bSApple OSS Distributions# 4.4.1 Repetition Count Test
37*d4514f0bSApple OSS Distributions#
38*d4514f0bSApple OSS Distributions# The Repetition Count Test (RCT) detects catastrophic failures in the
39*d4514f0bSApple OSS Distributions# noise source when it becomes "stuck" generating a single value over
40*d4514f0bSApple OSS Distributions# many consecutive samples.
41*d4514f0bSApple OSS Distributions#
42*d4514f0bSApple OSS Distributions# The probability of generating C consecutive identical samples is:
43*d4514f0bSApple OSS Distributions#
44*d4514f0bSApple OSS Distributions# P^(C-1)
45*d4514f0bSApple OSS Distributions#
46*d4514f0bSApple OSS Distributions# Or equivalently:
47*d4514f0bSApple OSS Distributions#
48*d4514f0bSApple OSS Distributions# 2^(-H * (C-1))
49*d4514f0bSApple OSS Distributions#
50*d4514f0bSApple OSS Distributions# To keep this under our rate of acceptable false positives, we need
51*d4514f0bSApple OSS Distributions# to satisfy this inequality:
52*d4514f0bSApple OSS Distributions#
53*d4514f0bSApple OSS Distributions# 2^-A >= 2^(-H * (C-1))
54*d4514f0bSApple OSS Distributions#
55*d4514f0bSApple OSS Distributions# Taking the logarithm of both sides, we have:
56*d4514f0bSApple OSS Distributions#
57*d4514f0bSApple OSS Distributions# -A >= -H * (C-1)
58*d4514f0bSApple OSS Distributions#
59*d4514f0bSApple OSS Distributions# Solving for C, we have:
60*d4514f0bSApple OSS Distributions#
61*d4514f0bSApple OSS Distributions# (A / H) + 1 >= C
62*d4514f0bSApple OSS Distributionsdef repetition_count_bound():
63*d4514f0bSApple OSS Distributions    return 1 + ceil(Fraction(A, H))
64*d4514f0bSApple OSS Distributions
65*d4514f0bSApple OSS Distributions
66*d4514f0bSApple OSS Distributions# 4.4.2 Adaptive Proportion Test
67*d4514f0bSApple OSS Distributions#
68*d4514f0bSApple OSS Distributions# The Adaptive Proportion Test (APT) tries to detect more subtle noise
69*d4514f0bSApple OSS Distributions# source failures causing certain values to occur with unexpected
70*d4514f0bSApple OSS Distributions# frequency. It does this by taking a sample from the noise source and
71*d4514f0bSApple OSS Distributions# counting how many times the same sample occurs within a fixed-size
72*d4514f0bSApple OSS Distributions# window.
73*d4514f0bSApple OSS Distributions
74*d4514f0bSApple OSS Distributions
75*d4514f0bSApple OSS Distributions# The size of the window for non-binary alphabets for the APT.
76*d4514f0bSApple OSS DistributionsW = 512
77*d4514f0bSApple OSS Distributions
78*d4514f0bSApple OSS Distributions
79*d4514f0bSApple OSS Distributions# The probability mass function measuring the probability of exactly k
80*d4514f0bSApple OSS Distributions# occurrences of a given value within the observation window of size
81*d4514f0bSApple OSS Distributions# W. We use the probability of the most likely event (as above).
82*d4514f0bSApple OSS Distributions#
83*d4514f0bSApple OSS Distributions# There are three terms:
84*d4514f0bSApple OSS Distributions#
85*d4514f0bSApple OSS Distributions# 1. The binomial coefficient of k, i.e. W-choose-k. Simply, how many
86*d4514f0bSApple OSS Distributions# ways are there to get exactly k outcomes given W chances.
87*d4514f0bSApple OSS Distributions#
88*d4514f0bSApple OSS Distributions# 2. The probability of each of those k events occurring.
89*d4514f0bSApple OSS Distributions#
90*d4514f0bSApple OSS Distributions# 3. The probability that the other W-k events have some other
91*d4514f0bSApple OSS Distributions# outcome.
92*d4514f0bSApple OSS Distributionsdef pmf(k):
93*d4514f0bSApple OSS Distributions    return comb(W, k) * P**k * (1 - P)**(W-k)
94*d4514f0bSApple OSS Distributions
95*d4514f0bSApple OSS Distributions
96*d4514f0bSApple OSS Distributions# The sum of probabilties of all possible counts of occurrences is 1.
97*d4514f0bSApple OSS Distributionsassert sum(map(pmf, range(W+1))) == 1
98*d4514f0bSApple OSS Distributions
99*d4514f0bSApple OSS Distributions
100*d4514f0bSApple OSS Distributions# We want to find the minimal count of occurrences such that the
101*d4514f0bSApple OSS Distributions# cumulative probability of seeing *at least* that count of
102*d4514f0bSApple OSS Distributions# occurrences (but possibly more) is no more than our false
103*d4514f0bSApple OSS Distributions# positive threshold.
104*d4514f0bSApple OSS Distributionsdef adaptive_proportion_bound():
105*d4514f0bSApple OSS Distributions    # The list of probabilities for each of the possible counts of
106*d4514f0bSApple OSS Distributions    # occurrences.
107*d4514f0bSApple OSS Distributions    probs = [pmf(x) for x in range(W+1)]
108*d4514f0bSApple OSS Distributions
109*d4514f0bSApple OSS Distributions    # The list of cumulative distributions for each of the possible
110*d4514f0bSApple OSS Distributions    # counts of occurrences.
111*d4514f0bSApple OSS Distributions    #
112*d4514f0bSApple OSS Distributions    # Whereas probs is a list of probabilities of *exactly* k
113*d4514f0bSApple OSS Distributions    # occurrences, this is a list of probabilities of *k or more*
114*d4514f0bSApple OSS Distributions    # occurrences.
115*d4514f0bSApple OSS Distributions    #
116*d4514f0bSApple OSS Distributions    # These are just sums of probabilities across a range of counts.
117*d4514f0bSApple OSS Distributions    dists = [sum(probs[x:]) for x in range(W+1)]
118*d4514f0bSApple OSS Distributions
119*d4514f0bSApple OSS Distributions    # Because we have constructed dists as an ordered list of
120*d4514f0bSApple OSS Distributions    # cumulative probabilities, we can simply return the index of the
121*d4514f0bSApple OSS Distributions    # first value that is below our threshold.
122*d4514f0bSApple OSS Distributions    for i, d in enumerate(dists):
123*d4514f0bSApple OSS Distributions        if d <= INV2**A:
124*d4514f0bSApple OSS Distributions            return i
125*d4514f0bSApple OSS Distributions
126*d4514f0bSApple OSS Distributions
127*d4514f0bSApple OSS Distributionsdef main():
128*d4514f0bSApple OSS Distributions    print('Estimated min-entropy:', H)
129*d4514f0bSApple OSS Distributions    print('False positive rate: 2^-{}'.format(A))
130*d4514f0bSApple OSS Distributions    print('Repetition Count Test bound:', repetition_count_bound())
131*d4514f0bSApple OSS Distributions    print('Adaptive Proportion Test bound:', adaptive_proportion_bound())
132*d4514f0bSApple OSS Distributions
133*d4514f0bSApple OSS Distributions
134*d4514f0bSApple OSS Distributionsif __name__ == '__main__':
135*d4514f0bSApple OSS Distributions    main()
136