網(wǎng)站首頁 編程語言 正文
1. 實驗說明
問題要求:針對靜態(tài)單賦值(SSA)形式的函數(shù)中間代碼輸入,輸出函數(shù)返回值的范圍
實現(xiàn)思路: 基本根據(jù) 2013年在CGO會議上提出的“三步法”范圍分析法加以實現(xiàn)[3],求得各個變量的范圍
算法優(yōu)勢:空間復(fù)雜度和時間復(fù)雜度都是 O(n),效率高
算法瓶頸: “三步法”的功能存在較大局限,它只能分析各個變量的最大范圍,對活躍變量只做了最簡單的考慮,因此最終得到的范圍比較不準(zhǔn)確,往往只能得到范圍的一個界
2. 項目使用
python main.py
(ssa文件路徑在main.py中設(shè)置)
不需要安裝任何庫。
3. 算法原理
簡單概括:采用三步法(2013年在CGO會議上提出)
3.1 構(gòu)建CFG
代碼:\src\eSSAConstraintGraph.py; \src\structure.py
功能:解析SSA,構(gòu)建CFG。
由于函數(shù)之間存在調(diào)用關(guān)系,因此首先把SSA劃分成不同的函數(shù)的SSA,再分別構(gòu)建CFG。CFG中保留了每一個函數(shù)的語句、Block之間的關(guān)系,為下一步構(gòu)建Constraint Graph
打基礎(chǔ)。
CFG的結(jié)構(gòu)如下:
# CFG類 ? ? ? class CFG: ? ? def __init__(self): ? ? ? ? self.name = '' ? ? ? ? self.Blocks = [] ? ? ? ? self.Edges = [] ? ? ? ? self.Arguments = []
3.2 構(gòu)建Constraint Graph
代碼:\src\eSSAConstraintGraph.py
三步法的前提是構(gòu)建Constraint Graph
。數(shù)據(jù)結(jié)構(gòu)如下。在這一步中,我用自己定義的數(shù)據(jù)類型MyNode來表示一條Constraint
。
# Constraint Graph類 ? ? ? class ConstraintGraph: ? ? def __init__(self, cfg): ? ? ? ? self.MyNodes = []?? ??? ??? ?#基本節(jié)點,每一個節(jié)點是一個Constraint ? ? ? ? self.MyConditions = []?? ??? ?#用于后面E-SSA Constraint Graph補(bǔ)充條件 ? ? ? ? self.cfg = cfg?? ??? ??? ?? ? ? ? ? self.Arguments = []?? ??? ??? ?#輸入?yún)?shù) ? ? ? ? self.returnName = ''?? ??? ?#輸出參數(shù) # MyNode : Constraint Graph的節(jié)點,也就是保存變量范圍的地方 class MyNode: ? ? def __init__(self, t= "", name = "", ?args = [], result = [], fromBlock = 0, Statement = ''): ? ? ? ? self.type = t ?? ??? ??? ?#節(jié)點類型:leave 葉節(jié)點存放范圍和值 #op運算符 #var變量名 ? ? ? ? self.name = name.strip() ?#節(jié)點名稱:運算名稱,或變量名稱 ? ? ? ? self.args = args?? ?#參數(shù),一個節(jié)點是另一個節(jié)點的argument,意味著二者之間有邊相連 ? ? ? ? self.result = result ? ? ? ?#被用到哪,一個節(jié)點是另一個節(jié)點的result,意味著二者之間有邊相連 ? ? ? ? self.Conditions = [] ? ? ? ?#約束條件, 在后面E-SSA Constraint Graph中補(bǔ)充條件 ? ? ? ? self.fromBlock = fromBlock ?#在CFG的哪個Block中定義的 ? ? ? ? self.Statement = Statement ?#在SSA中的哪條Statement中 ? ? ? ? self.Range = Range()?? ??? ?#節(jié)點范圍 ? ? ? ? self.size = '' ? ? ? ? self.input = False # Range由兩個Bound組成? class Range: ? ? def __init__(self ): ? ? ? ? self.lowBound = Bound() ? ? ? ? self.highBound = Bound() # Bound由值和類型組成 class Bound: ? ? def __init__(self): ? ? ? ? self.value = 'None' ? ? ?# inf 最大值 ; -inf 最小值; None 未設(shè)置; Not Exists 不存在 ? ? ? ? self.size = 'None' ? ? ? #邊界是 int or float
需要注意的是,在解決兩個函數(shù)之間的調(diào)用關(guān)系時,將被調(diào)用的函數(shù)**內(nèi)聯(lián)進(jìn)原函數(shù)**。我將被調(diào)用的函數(shù)的所有變量名都加入相應(yīng)的后綴,比如`foo`調(diào)用`bar`函數(shù),那么`bar`中的變量`i_1`將被更名保存為`i_1#bar$1`,其中#是變量原名和后綴分割符,$是函數(shù)名和一個隨機(jī)數(shù)的分割符,\$的作用是為了區(qū)分多次調(diào)用同一個函數(shù)的情況。
3.3 構(gòu)建E-SSA Constraint Graph
代碼:`\src\eSSAConstraintGraph.py`
這一步用于解決條件的添加。諸如`if (i_2 < j_3)`
這樣的條件。在MyNode節(jié)點類型中,我設(shè)置了Conditions結(jié)構(gòu)用于保存條件。Condition
的數(shù)據(jù)結(jié)構(gòu)如下:
?Class Description : Constraint Graph
中的條件,附加在MyNode中
class MyCondition: ? ? def __init__(self, condition, index): ? ? ? ? self.condition = condition ? ? ? ? self.arg1 = re.sub("\(.*\)", "",condition.split()[0].strip()) ? ? ? ? self.arg2 = re.sub("\(.*\)", "",condition.split()[2].strip()) ? ? ? ? self.op = condition.split()[1].strip() ? ? ? ? self.index = index
其中,arg1和arg2分別表示條件的兩個參數(shù),op表示條件的比較運算符。在Future Resolution
這一步會進(jìn)行比較,進(jìn)行范圍的約束。
以t7.ssa為例,得到的E-SSA Constraint Graph如下:
call bar$1 ?in 2 : |Arguments: i_2,|Result: |Conditions:? var i_2 ?in 2 : |Arguments: |Result: bar$1,i#bar$1,i_2#bar$1,|Conditions:? var j_4 ?in 2 : |Arguments: _1#bar$1,|Result: bar$2,i#bar$2,i_2#bar$2,|Conditions:? ret bar$1 ?in 2 : |Arguments: |Result: j_4,|Conditions:? call bar$2 ?in 2 : |Arguments: j_4,|Result: |Conditions:? var k_6 ?in 2 : |Arguments: _1#bar$2,|Result: _7,|Conditions:? ret bar$2 ?in 2 : |Arguments: |Result: k_6,|Conditions:? var _7 ?in 2 : |Arguments: k_6,|Result: |Conditions:? var i_2#bar$1 ?in 3 : |Arguments: i_2,|Result: +,-,|Conditions: 0#bar$1 0| leaf 10 ?in 3 : |Arguments: |Result: +,|Conditions:? op + ?in 3 : |Arguments: i_2#bar$1,10,|Result: _3#bar$1,|Conditions: 0#bar$1 0| var _3#bar$1 ?in 3 : |Arguments: +,|Result: PHI,|Conditions: 0#bar$1 0| leaf 5 ?in 4 : |Arguments: |Result: -,|Conditions:? op - ?in 4 : |Arguments: 5,i_2#bar$1,|Result: _4#bar$1,|Conditions: 0#bar$1 1| var _4#bar$1 ?in 4 : |Arguments: -,|Result: PHI,|Conditions: 0#bar$1 1| op PHI ?in 4 : |Arguments: _3#bar$1,_4#bar$1,|Result: _1#bar$1,|Conditions: 0#bar$1 1| var _1#bar$1 ?in 4 : |Arguments: PHI,|Result: j_4,|Conditions: 0#bar$1 1| leaf i#bar$1 ?in ?: |Arguments: i_2,|Result: |Conditions:? var i_2#bar$2 ?in 3 : |Arguments: j_4,|Result: +,-,|Conditions: 0#bar$2 0| leaf 10 ?in 3 : |Arguments: |Result: +,|Conditions:? op + ?in 3 : |Arguments: i_2#bar$2,10,|Result: _3#bar$2,|Conditions: 0#bar$2 0| var _3#bar$2 ?in 3 : |Arguments: +,|Result: PHI,|Conditions: 0#bar$2 0| leaf 5 ?in 4 : |Arguments: |Result: -,|Conditions:? op - ?in 4 : |Arguments: 5,i_2#bar$2,|Result: _4#bar$2,|Conditions: 0#bar$2 1| var _4#bar$2 ?in 4 : |Arguments: -,|Result: PHI,|Conditions: 0#bar$2 1| op PHI ?in 4 : |Arguments: _3#bar$2,_4#bar$2,|Result: _1#bar$2,|Conditions: 0#bar$2 1| var _1#bar$2 ?in 4 : |Arguments: PHI,|Result: k_6,|Conditions: 0#bar$2 1| leaf i#bar$2 ?in ?: |Arguments: j_4,|Result: |Conditions:? Conditions: i_2(D) >= 0#bar$1 0#bar$1,i_2(D) >= 0#bar$2 0#bar$2, ```http://www.biyezuopin.vip
3.4 三步法
3.4.1 Widen
代碼:`\src\rangeAnalysis.py`
Widen
步驟用于將 變量范圍擴(kuò)大。此步驟可以在O(n)階段內(nèi)完成。基于原理如下:可以形象的理解為:在進(jìn)行Φ操作時,如果發(fā)現(xiàn)變量范圍向上增加,就直接擴(kuò)大到inf,如果發(fā)現(xiàn)變量范圍向下減小,就直接減小到-inf。
這樣下來后,每一個MyNode
的范圍都會擴(kuò)大到最大。
3.4.2 Future Resolution & ?Narrow
代碼:`\src\rangeAnalysis.py`
在Widen步驟中,只能解決每一個變量內(nèi)部之間的賦值行為,在Future Resolution
步驟,可以對變量之間的運算、以及條件進(jìn)行處理。
我用了復(fù)雜的`ConditionHandle()
`函數(shù)來解決條件變量的Constraint問題。我在每一個MyNode中添加了Conditions結(jié)構(gòu),用Condition約束來代替變量替換。這樣可以大大減少變量替換帶來的麻煩。
在`ConditionHandle()
`中,我將條件拆分成`arg1` `arg2`和`op`三部分,將他們組合成條件為真的范圍,和條件為假的范圍。并把相應(yīng)的范圍賦給相應(yīng)的變量,以及檢查此路徑是否可以相通。
以`t7.ssa`為例,三步法得到的所有變量的范圍如下:
Enter Range For i: -10 10 bar$1 None None | Range: ?Not Exists Not Exists i_2 int int | Range: ?-10 10 j_4 int int | Range: ?0 20 bar$1 None None | Range: ?Not Exists Not Exists bar$2 None None | Range: ?Not Exists Not Exists k_6 int int | Range: ?5 30 bar$2 None None | Range: ?Not Exists Not Exists _7 int int | Range: ?5 30 i_2#bar$1 int int | Range: ?-10 10 10 None None | Range: ?10 10 + int int | Range: ?0 20 _3#bar$1 int int | Range: ?0 20 5 None None | Range: ?5 5 - int int | Range: ?Not Exists Not Exists _4#bar$1 int int | Range: ?15 -5 PHI int int | Range: ?0 20 _1#bar$1 int int | Range: ?0 20 i#bar$1 None None | Range: ?Not Exists Not Exists i_2#bar$2 int int | Range: ?0 20 10 None None | Range: ?10 10 + int int | Range: ?10 30 _3#bar$2 int int | Range: ?10 30 5 None None | Range: ?5 5 - int int | Range: ?Not Exists Not Exists _4#bar$2 int int | Range: ?5 -15 PHI int int | Range: ?5 30 _1#bar$2 int int | Range: ?5 30 i#bar$2 None None | Range: ?Not Exists Not Exists
可以直接得到結(jié)果變量_7的范圍為:_7 int int | Range: 5 30
4. 實驗結(jié)果
# t1.SSA Reference Range:[100, 100] Output Range: [100, +inf] # t2.SSA Reference Range:[200, 300] Output Range: [200, +inf] # t3.SSA Reference Range:[20, 50] Output Range: [20, +inf] # t4.SSA Reference Range:[0, +inf] Output Range: [0, +inf] # t5.SSA Reference Range:[210, 210] Output Range: [0, +inf] # t6.SSA Reference Range:[-9, 10] Output Range: [-9, 10] # t7.SSA Reference Range:[16, 30] Output Range: [5, 30] # t8.SSA Reference Range:[-3.2192308, 5.94230769] Output Range: [-0.41923075526423315, 14.700000286102295] # t9.SSA Reference Range:[9791, 9791] Output Range: [-10, +inf] # t10.SSA Reference Range:[-10, 40] Output Range: [1, 1]
5. 總結(jié)
在本實驗中,我采用python語言對SSA形式的C程序進(jìn)行解析,并采用三步法針對特定輸入進(jìn)行了相應(yīng)的范圍分析。收貨了寫代碼的樂趣,也為最后的效果遺憾。
最后的效果中,10個benchmark
的結(jié)果中準(zhǔn)確結(jié)果寥寥無幾。尤其是上界,很多都直接到無窮了。這一方面是為了追求時間效率和空間效率,放棄了模擬執(zhí)行采用三步法的缺陷,另一方面也是因為我沒有想到合適的改進(jìn)方法。
原文鏈接:https://blog.csdn.net/newlw/article/details/122978669
相關(guān)推薦
- 2022-10-27 C++移動語義介紹與使用講解_C 語言
- 2022-04-22 Element UI 使用表單校驗,正確輸入后,仍然有提示信息
- 2022-07-19 sprintf和sscanf的用法及應(yīng)用
- 2022-07-11 Android?Studio實現(xiàn)注冊頁面跳轉(zhuǎn)登錄頁面的創(chuàng)建_Android
- 2023-11-16 Linux查看某目錄下的文件個數(shù)
- 2022-05-19 C++實現(xiàn)教師管理系統(tǒng)_C 語言
- 2022-08-12 Python3.8安裝tensorflow的簡單方法步驟_python
- 2022-04-18 Unity命令行打包WebGL的示例代碼_C#教程
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支