5-6. DFA Foundations I & II
View the Iterative Algorithm in Another Way¶
Given a CFG (program) with k nodes, the iterative algorithm updates OUT[n]
for every node n in each iteration.
Assume the domain of the values in data flow analysis is \(V\),then we can define a \(k\)-tuple
as an element of set \((V_1, V_2, \dots, V_k)\) denoted as \(V^k\), to hold the values of the analysis after each iteration.
因此,每一次 iteration 可以抽象为函数 \(F: V^k \to V^k\)
Some Questions¶
本质上,我们就是找 \(F\) 的不动点。
- Is the algorithm guaranteed to terminate or reach the fixed-point, or does it always have a solution?
- Guaranteed to reach?
- If so, is there only one solution or only one fixed point? If more than one, is our solution the best one (most precise)?
- How many fixed-points?
- How "precise" is our solution?
- When will the algorithm reach the fixed point, or when can we get the solution?
- Time complexity?
Lattice¶
Background: Upper and Lower Bounds¶
Definition¶
Lattice: Given a poset \((P, \sqsubseteq)\), \(\forall a, b \in P\), if a \(a \sqcup b\) and \(a \sqcap b\) exist, then \((P, \sqsubseteq)\) is called a lattice.
Semilattice: Given a poset \((P, \sqsubseteq)\), \(\forall a, b \in P\),
- if only \(a \sqcup b\) exists, then \((P, \sqsubseteq)\) is called a join semilattice
- if only \(a \sqcap b\) exists, then \((P, \sqsubseteq)\) is called a meet semilattice
- 注:由于程序是有限的,因此,我们的 lattice 就是 finite lattice,也就是 complete lattice。
Dataflow Analysis Framework via Lattice¶
如上图(上图是一个 may analysis,采用 join operation):
- 首先,OUT[s1] 和 OUT[s3] 在 lattice 上对应 {a} 和 {b}。因此,join 就是 {a, b},我们把它赋值给
- 然后,IN[s1] 通过 transfer function 来得到 OUT[s1]
- And so on and so forth
Fix-Point Theorem¶
定义(单调性):A function \(F: L \to L\) is monotonic if
定理:
证明:
我们不妨只证明 \(\bot\) 的情况(\(\top\) 的情况同理)。
(存在性)构造升链 \(\newcommand{par}{\sqsubseteq}\bot \par f(\bot) \par f(f(\bot)) \par \dots\)。从而,通过 pigeon hole's theorem,必然存在 \(f^{n}(\bot) = f^{m}(\bot)\),从而又由偏序关系:\(f^{n}(\bot) = f^{n+1}(\bot) = f(f^{n}(\bot))\)。从而 \(f^n(\bot)\) 就是不动点。
(最小)对于任何不动点 \(x\),我们构造升链 \(x \par f(x) \par f(f(x)) \par \dots\)。又由于 \(\bot \par x\),因此 \(\forall n \in \mathbb N: f^n(\bot) \par f^n(x) = x\)。
由于我们最终求得的不动点,也是 \(f^n(x)\) 中的一员。因此,我们最终求得的不动点,一定是偏序小于等于其它所有不动点。换言之,就是 least fixed-point。
\(\blacksquare\)
\(F\) is monotonic¶
F 可以认为是
- 使用所有 OUT[i, n],来计算 IN[1, n+1]
- 使用 IN[1, n+1],计算 OUT[1, n+1]
- 使用 OUT[2~k, n] 和 OUT[1, n+1],计算 IN[2, n+1]
- 使用 IN[1, n+2],计算 OUT[1, n+2]
- ……
一共 2k 步。其中,
- 奇数步是 meet(must) 或者 join(may)
- 偶数步是 \((IN - kill) \cup gen\)
由于格上的偏序关系,就是集合的包含于关系,因此
奇数步(只是粗略证明):
偶数步(易证)。
Answer to the first two questions¶
从而,我们证明了前面三个问题中的两个:
- Is the algorithm guaranteed to terminate or reach the fixed-point, or does it always have a solution?
- Yes
- If so, is there only one solution or only one fixed point? If more than one, is our solution the best one (most precise)?
- For the first part, not necessarily!
- For the second part, our solution will be the least or greatest fixed point, depending on where we start.
Answer to the third question¶
其中,
- \(k\) 就是 \(F: V^k \to V^k\) 中的 \(k\),i.e. basic block 的数量。
- \(h\) 就是 set lattice 的高度,i.e. bit vector 的长度。
Fixed-Point And Algorithm¶
如上图所示,以 reaching definitions (may analysis) 为例
- 由于我们优化的是 unreachable definitions,因此 \(\bot\) 就是 unsafe,\(\top\) 就是 safe but useless
- 我们通过设计 data abstraction 以及 merge function,保证不动点都位于 safe 的区域
- 由于从 unsafe 到 safe,越来越不 precise,因此,我们为了找到最 precise 的不动点,就要找 least fixed point
- 由于 \(F\) 是单调函数,我们可以从 \(\bot\) 开始找,从而找到
Available expression analysis (must analysis) 同理。
总体而言,
- 我们从 unsafe result 开始找,不是因为它是 unsafe,而是因为它是 most precise。
- 由于我们已经保证了 fixed point 必须在 safe zone 里,我们不需要担心找到 unsafe fixed point
- 由于我们保证 \(F\) 是单调函数,我们一定可以找到不动点,而且是最精确不动点。
Another Point of View¶
对于 \(F\) 而言,transfer function 已经被写死了。因此,我们能够改变的,只有 merge function。
不论是 join 还是 meet,都是往上走/往下走的过程中,走最小的一步。
因此,通过“小步走”,我们可以避免走到过分 unprecise 的地方。
Precision¶
MOP: Meet Over All Paths¶
假如我们根本不知道静态分析的 iterative approach,那么,一个直观的想法是:
枚举所有可能的路径(可以有环),然后做一个 meet (must) / join (may)。
但是:
- 部分路径可能根本不会执行,从而这样的分析是 not fully precise 的(详见下面的代码)。
- 所有可能的路径可能指数爆炸,甚至有无穷多,因此这样的分析是 impractical 的。
- 也就是说:这样的分析方式只存在于概念上,而不能用于实际
li a5, 0
bne a5, zero, if # Assume no statement would jump to this
jal zero, else # Assume no statement would jump to this
if:
## ...
else:
## ...
Compare To Iterative Approach¶
对于这个简单的示例 (assuming it's may analysis) 而言:
- MOP:
IN[s4] = f_3(f_1(OUT[entry])) join f_3(f_2(OUT[entry]))
- Iterative:
IN[s4] = f_3(f_1(OUT[entry]) join f_2(OUT[entry]))
我们把 join 当作二元运算,那么这个 lattice 就是一个交换群。而 transfer function \(f\) 就是一个群同态。
证明:
从而:
上面讨论的是 join 的情况,对于 meet 同理。
Q.E.D.
A counterexample: Constant Propagation¶
Constant prop 是一个类 must analysis。但是,使用 MOP 和 iterative method 不是等价的。
Lattice and Meet Rules¶
Constant propagation 的 lattice 如上图所示。
其中 ,UNDEF 的意思就是:这个 variable 必然是常量(不论是什么路径),但是具体是什么,我们不知道,因为它没有被初始化。
meet 的运算是:
- UNDEF ⊓ u = u
- u ⊓ u = u
- u ⊓ v = NAC
- s.t. u ≠ v
- NAC ⊓ u = NAC
其中,UNDEF ⊓ u = u 可能会让人心生疑惑:假如一个变量在使用的时候确实没有被初始化,我们不就完了?
我们这里给出解答:constant propagation 的设计里面,已经假设了变量使用之前一定已经被初始化了。至于检测是否初始化,那就是 uninitialized variable detection 需要做的事了。
Transfer Function¶
一条语句就是一个 transfer function。如果语句不是赋值操作,那么 transfer function 就是 identity。
\(F\) is non-homomorphic¶
如上图,令 \(f\) 为语句 c = a + b
的转移函数。
则:
但是:
因此,\(F\) 并不是这个 lattice 上的同态。
- 不过,还是可以证明:\(f(X \sqcap Y) \sqsubseteq f(X) \sqcap f(Y)\),也就是说,使用 iterative method 的分析方式,不会优于 meet over all paths 的分析方式。也就是说:iterative method 肯定不会比 MOP 更准。
Optimization: Worklist Algorithm¶
如图,标黄的部分,就是 worklist algorithm 相对 naïve iterative algorithm 的区别。
优化的思想也很简单:如果上一轮和这一轮的值是不一样的,就把它的 successor 加入 worklist。
- 换言之,假如一个 basic block 的所有 predecessor 在上一轮和本轮(或者上一轮和上上一轮)中是一样的,那么这个 basic block 的 out 在上一轮和本轮肯定也不会变——本轮不需要再计算了。