国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院

首頁 > 編程 > ASM > 正文

匯編語言學習筆記(十二)-浮點指令

2019-11-11 01:53:20
字體:
來源:轉載
供稿:網友
浮點數如何存儲浮點寄存器浮點數指令浮點計算例子浮點高級運算CMOV移動指令

浮點數如何存儲

浮點數的運算完全不同于整數,從寄存器到指令,都有一套獨特的處理流程,浮點單元也稱作x87 FPU。

現在看浮點數的表示方式,我們所知道的,計算機使用二進制存儲數據,所表示的數字都具有確定性,那是如何表示浮點這種具有近似效果的數據呢,答案是通過科學計數,科學計數由符號,尾數和指數表示,這三部分都是一個整數值,具體來看一下IEEE二進制浮點標準:

格式說明
單精度32位:符號占1位,指數占8位,尾數中的小數部分占23位
雙精度64位:符號占1位,指數占11位,尾數中的小數部分占52位
擴展精度80位:符號占1位,指數占16位,尾數中的小數部分占63位

以單精度為例,在內存中的儲存格式如下(左邊為高位):

	| 1位符號 | 8位指數 | 23位尾數 |

其中符號位1表示負數,0表示正數,這與整數形式的符號位意義相同; 科學計數法表示形式如 m * (b ^ e),m為尾數,b為基數,e是指數,再二進制中,基數毫無疑問是2,對單精度,指數為中間8位二進制表示的數字,其中的尾數是形如1.1101 小數點后面的整數值。

關于指數,由于需要表示正負兩種數據,IEEE標準規定單精度指數以127為分割線,實際存儲的數據是指數加127所得結果,127為高位為零,后7位為1所得,其他雙精度也以此方式計算。

為了解釋內存中浮點數的存儲方式,舉一個浮點數的例子說明:

float test = 123.456;int main(){    return 0;}

例子再簡單不過了,僅僅定義了一個全局的float類型,我們通過gcc -S test.c來生成匯編,看看123.456是如何存儲的,打開反匯編后的文件,看到符號_test后定義的數字是 1123477881(這里gcc定義成了long類型,不過沒有關系,因為都是四字節數字,具體的類型還得看如何使用)。可以使用計算器把十進制數字轉化為二進制:0 10000101 11101101110100101111001,這里根據單精度的劃分方式把32位劃分成三部分,符號位為0,為正數,指數為 133,減去127得6,尾數加上1.,形式為1.11101101110100101111001,擴大2 ^ 23次方為111101101110100101111001,十進制16181625,后除以2 ^ (23 – 6) = 131072,結果為123.45600128173828125,與我們所定義的浮點數正好相符。

浮點寄存器

這里介紹了浮點數的二進制表示,前面說過浮點單元計算使用獨立的寄存器,在寄存器那篇也稍有提及,這里詳細說明一下浮點單元的寄存器設施。

FPU有 8 個獨立尋址的80位寄存器,名稱分別為r0, r1, …, r7,他們以堆棧形式組織在一起,統稱為寄存器棧,編寫浮點指令時棧頂也寫為st(0),最后一個寄存器寫作st(7)。

FPU另有3個16位的寄存器,分別為控制寄存器、狀態寄存器、標記寄存器,現一一詳細說明此三個寄存器的作用:

狀態寄存器,為用戶記錄浮點計算過程中的狀態,其中各位的含義如下:

0 —— 非法操作異常1 —— 非規格化操作數異常2 —— 除數為0異常3 —— 溢出標志異常4 —— 下溢標志異常5 —— 精度異常標志6 —— 堆棧錯誤7 —— 錯誤匯總狀態8 —— 條件代碼位0(c0)9 —— 條件代碼位1(c1)10 —— 條件代碼位2 (c2)11-13 —— 堆棧頂指針14 —— 條件代碼位3(c3)15 —— 繁忙標志

其中讀取狀態寄存器內容可使用 fstsw %ax

控制寄存器的位含義如下:

0 —— 非法操作異常掩碼 1 —— 非法格式化異常掩碼 2 —— 除數為0異常掩碼 3 —— 溢出異常掩碼 4 —— 下溢異常掩碼 5 —— 精度異常亞曼 6-7 —— 保留 8-9 —— 精度控制(00單精度,01未使用,10雙精度,11擴展精度) 10-11 —— 舍入控制(00舍入到最近,01向下舍入,10向上舍入,11向0舍入) 12 —— 無窮大控制 13–15 —— 保留

其中讀取控制寄存器和設置控制寄存器的指令如下:

# 加載到內存fstcw control# 加載到控制器fldcw control

最后的標志寄存器最為簡單,分別0-15位分別標志r0-r7共8個寄存器,每個寄存器占2位,這兩位的含義如下:

11 —— 合法擴展精度 01 —— 零 10 —— 特殊浮點 11 —— 無內容

另外對浮點寄存器的一些控制指令如下:

# 初始化fpu,控制、狀態設為默認值,但不改變fpu的數據finit# 恢復保存環境fldenv bufferfstenv buffer#清空浮點異常fnclex#fpu狀態保存fssave

fstenv 保存控制寄存器、狀態寄存器、標記寄存器、FPU指令指針偏移量、FPU數據指針,FPU最后執行的操作碼到內存中。

浮點數指令

接下來將要詳細說明其計算過程,要計算數據首先得看如何從內存中加載數據到寄存器,同時把結果從寄存器取出到內存,除了加載內存中的浮點數據指令,另外還有一些常量的加載,現列舉如下:

指令說明
finit初始化控制和狀態寄存器,不改變fpu數據寄存器
fstcw control將控制寄存器內容放到內存control處
fstsw status將狀態寄存器內容放到內存status處
flds value加載內存中的單精浮點到fpu寄存器堆棧
fldl value加載內存中的雙精浮點到fpu寄存器堆棧
fldt value加載內存中的擴展精度點到fpu寄存器堆棧
fld %st(i)將%st(i)寄存器數據壓入fpu寄存器堆棧
fsts value單精度數據保存到value,不出棧
fstl value雙精度數據保存到value,不出棧
fstt value擴展精度數據保存到value,不出棧
fstps value單精度數據保存到value,出棧
fstpl value雙精度數據保存到value,出棧
fstpt value擴展精度數據保存到value,出棧
fxch %st(i)交換%st(0)和%st(i)
fld1把 +1.0 壓入 FPU 堆棧中
fldl2t把 10 的對數(底數2)壓入 FPU 堆棧中
fldl2e把 e 的對數(底數2)壓入 FPU 堆棧中
fldpi把 pi 的值壓入 FPU 堆棧中
fldlg2把 2 的對數(底數10)壓入 FPU 堆棧中
fldln2把 2 的對數(底數e) 壓入堆棧中
fldz把 +0.0 壓入壓入堆棧中

以上指令雖多,但是還是很有規律,前綴f表示fpu操作,ld加載,st保存設置,p后綴彈出堆棧,s、l、t后綴表示單精度,雙精度,擴展精度,c后綴表 示控制寄存器,s后綴表示狀態寄存器。當然這僅僅是對AT&T語法而言,對MASM語法沒有s,l,t之分,需要使用type ptr來指明精度,即內存大小。

學會靈活的加載彈出數據堆棧后,接下來就要看一些基本的計算:

fadd    浮點加法fdiv    浮點除法fdivr   反向浮點除法fmul    浮點乘法fsub    浮點減法fsubr   反向浮點減法

對于以上的每種指令,有幾種指令格式,以fadd為例,列舉如下:

# 內從中的32位或者64位值和%st(0)相加fadd source# 把%st(x)和%st(0)相加,結果存入%st(0)fadd %st(x), %st(0)# 把%st(0)和%st(x)相加,結果存入%st(x)fadd %st(0), %st(x)# 把%st(0)和%st(x)相加,結果存入%st(x),彈出%st(0)faddp %st(0), %st(x)# 把%st(0)和%st(1)相加,結果存入%st(1),彈出%st(0)faddp# 把16位或32位整數與%st(0)相加,結果存入%st(0)fiadd source

這僅僅是對AT&T語法而言,對MASM源操作數與目的操作數相反!另外,對AT&T,與內存相關指令可加s、l指定內存精度。其中反向加法和反向除法是計算過程中目的與源反向計算。

浮點計算例子

接下來舉一個AT&T語法的例子,來計算表達式的值 ( 12.34 * 13 ) + 334.75 ) / 17.8 :

# ( 12.34 * 13 ) + 334.75 ) / 17.8.section .data    values: .float 12.34, 13, 334.75, 17.8    result: .double 0.0    outstring: .asciz "result is %f/n".section .text.globl _main_main:    leal values, %ebx    flds 12(%ebx)    flds 8(%ebx)    flds 4(%ebx)    flds (%ebx)    fmulp    faddp    fdivp %st(0), %st(1)    fstl result    leal result, %ebx    pushl 4(%ebx)    pushl (%ebx)    pushl $outstring    call _PRintfend:    pushl $0    call _exit

前四個flds加載所有的數據到寄存器堆棧,可以單步運行并是用gdb的print $st0打印堆棧寄存器的值,可以看到為什么是堆棧寄存器。需要說明的是由于printf的%f是double類型的輸出,所以最后要把一個8字節浮點放 到棧中傳遞,最終結果為27.818541,可以看到與計算器計算的結果近似相等。

浮點高級運算

除了基本的浮點計算,x87還提供了一些諸如余弦運算等高級計算功能:

指令說明
f2xm1計算2的乘方(次數為st0中的值,減去1
fabs計算st0中的絕對值
fchs改變st0中的值的符號
fcos計算st0中的值的余弦
fpatan計算st0中的值的部分反正切
fprem計算st0中的值除以st1的值的部分余數
fprem1計算st0中的值除以st1的值的IEEE部分余弦
fptan計算st0中的值的部分正切
frndint把st0中的值舍入到最近的整數
fscale計算st0乘以2的st1次方
fsin計算st0中的值的正弦
fsincos計算st0中的值的正弦和余弦
fsqrt計算st0中的值的平方根
fyl2x計算st1*log st0 以2為底
fyl2xp1計算st1*log (st0 + 1) 以2為底

下面來看一下浮點條件分支,浮點數的比較不像整數,可以容易的使用cmp指令比較,判斷eflags的值,關于浮點數比較,fpu提供獨立的比較機制和指令,現對這組比較指令進行說明:

指令說明
fcom比較st0和st1寄存器的值
fcom %st(x)比較st0和stx寄存器的值
fcom source比較st0和32/64位內存值
fcomp比較st0和st1寄存器的值,并彈出堆棧
fcomp %st(x)比較st0和stx寄存器的值,并彈出堆棧
fcomp source比較st0和32/64位內存值,并彈出堆棧
fcompp比較st0和st1寄存器的值,并兩次彈出堆棧
ftst比較st0和0.0

浮點數比較的結果放入狀態寄存器的c0,c2,c3條件代碼位中,其值如下:

結果c3c2c0
st0 > source000
st0 < source001
st0 = source100

如此倘若直接判斷c0,c2,c3的值比較繁瑣,所以可以使用一些技巧,首先使用fstsw指令獲得fpu狀態寄存器的值并存入ax,再使用sahf指令把 ah寄存器中的值加載到eflags寄存器中,sahf指令把ah寄存器的第0、2、4、6、7分別傳送至進位、奇偶、對準、零、符號位,不影響其他標 志,ah寄存器中這些位剛好包含fpu狀態寄存器的條件代碼值,所以通過fstsw和sahf指令組合,可以傳送如下值:

把c0位傳送到eflags的進位標志 把c2位傳送到eflags的奇偶校驗標志 把c3位傳送到eflags的零標志

傳送完畢后,可以用條件跳轉使用不同的結果值,另外需要說明的是浮點數相等判斷,因為浮點數本身存儲結構決定了它僅僅是一個近似值,所以不能直接判斷是否相 等,這樣可能與自己預期的結果不同,應該判斷兩個浮點數之差是否在一個很小的誤差范圍內,來決定這兩個浮點數是否相等。

根據上面的技巧,使用fstsw和fpu指令組合,可以方便的使用浮點判斷結果,這對我們是一種便利,而intel的工程師又為我們設計了一個組合指令,fcomi指令執行浮點比較結果并把結果存放到eflags寄存器的進位,奇偶,和零標志。

指令說明
fcomi比較st0和stx寄存器的值
fcomip比較st0和stx寄存器,并彈出堆棧
fucomi比較之前檢查無序值
fucomip比較之前檢查無序值,之后彈出堆棧

判斷結束后eflags的標志設置如下:

結果ZFPFCF
st0 > st(x)000
st0 < st(x)001
st0 = st(x)100

CMOV移動指令

最后介紹的是類似cmov的指令,根據判斷結果決定是否需要移動數據,其AT&T格式為 fcmovxx source, destination,其中source是st(x)寄存器,destination是st(0)寄存器。

指令說明
fcmovb如果st(0)小于st(x),則進行傳送
fcmove如果st(0)等于st(x),則進行傳送
fcmovbe如果st(0)小于或等于st(x),則進行傳送
fcmovu如果st(0)無序,則進行傳送
fcmovnb如果st(0)不小于st(x),則進行傳送
fcmovne如果st(0)不等于st(x),則進行傳送
fcmovnbe如果st(0)不小于或等于st(x),則進行傳送
fcmovnu如果st(0)非無序,則進行傳送

以上可以看出,無論從寄存器的操作,還是計算過程,都比整數運算要繁瑣的多,而且看似很簡單的一個表達式,轉化成浮點匯編需要做很多工作,由于其復雜性,同 一個表達式可以有多種運算過程,當然其中的效率相差很大,這依賴于對浮點匯編的理解程度,好在有高級語言處理相關工作,編寫浮點指令的情況比較少見。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院
国产美女福利在线| 国产精品不卡一区二区三区在线观看| 麻豆av电影在线观看| 69精品视频| 亚洲videos| 久青青在线观看视频国产| 国产女人伦码一区二区三区不卡| 国产原创在线播放| 国产高清av在线| 四虎久久影院| 中文字幕不卡免费视频| 国产网站在线播放| www.操操| 国产黄网站在线观看| av麻豆国产| 在线观看电影av| 国产精品一区二区三区高清在线| 中文字幕亚洲精品视频| 亚洲尤物在线视频| 91九色在线看| 成年女人在线视频| 最新av中文字幕| 中文av字幕| 操操操综合网| 国产偷窥洗澡视频| 人人在线视频| 18av在线视频| 大香伊人中文字幕精品| av网站在线播放| 国产青草视频在线观看视频| 精品久久亚洲一级α| 国产永久免费高清在线观看视频| 国产精品扒开做爽爽爽的视频| a中文在线播放| 91美女主播在线视频| 二人午夜免费观看在线视频| 四虎a级欧美在线观看| 国产精品你懂的在线观看| 日本a级黄色| 在线观看视频污| 96久久久久久| 国产黄色免费网| 国产中文在线| 久久久久久久久久久久网站| 在线观看av的网站| 国产精品久久久久久福利| 日本中文字幕视频| 在线免费观看黄色av| av在线日韩国产精品| 国产午夜在线| 国产免费麻豆视频| av中文字幕在线看| wwww在线观看| 久久久久久国产视频| 免费男女羞羞的视频网站中文字幕| 精品视频二区三区| 99精品老司机免费视频| 国产精品日日爱| 中文字幕在线观看日本| www.久草.com| 91视频久色| 四虎成年永久免费网站| 日本不卡视频一区二区| 精品美女调教视频| 精品视频麻豆入口| 爱福利在线视频| 国产精品一区二三区| 国产视频你懂的| 国产91久久久久蜜臀青青天草二| 国产第一页在线视频| 国产黄大片在线观看画质优化| v天堂福利视频在线观看| 国产在线高清理伦片a| 成人日韩欧美| 91麻豆精品国产91久久| 在线看黄网址| 成年女人在线视频| 国产国产人免费人成免费视频| 在线视频中文字幕久| 久久精品国产亚洲a∨麻豆| 国产尤物一区二区三区| 国产va在线观看| 成年网在线观看免费观看网址| 国产黄色在线播放| www在线观看播放免费视频日本| 国产乱视频在线观看播放| 国产探花在线观看| 超碰在线国产| 亚洲欧美一区二区三区在线播放| 超碰免费在线| 国产污视频在线| 性欧美精品xxxx| 国产高清在线观看| 国产尤物一区二区三区| 日本亚洲欧美| 一区二区三区免费视频网站| 天天操人人爽| 久久五月精品中文字幕 | 中文字幕av高清在线观看| 国产女人在线视频| 牛牛精品视频在线| 日本在线视频www鲁啊鲁| 国产特级嫩嫩嫩bbb| 成人免费一区二区三区视频网站| 在线亚洲精品自拍| 久热国产在线| 精品美女在线观看视频在线观看 | 懂色av中文在线| 午夜在线观看91| 永久免费网站在线| 97操碰视频| 亚洲精品在线视频免费| 国产黄色一级电影| 在线天堂视频| 免费在线观看a| 国产麻豆一级片| 国产一卡2卡3卡免费网站| 国产色视频网站| 在线视频1区2区| eeuss影院www在线播放| 久草国产视频| 最新黄网在线观看| 懂色av一区| 午夜亚洲成人| 永久免费不卡在线观看黄网站| 亚洲图区欧美| www.五月色.com| 午夜在线不卡| 五月综合网站| 国产麻豆精品视频一区二区 | 精品福利视频导航大全| 99高清免费国产自产拍| 免费国产视频| 国产网红女主播精品视频| av中文在线资源| 国产在线精品一区二区不卡| 国产精品理人伦一区二区三区| 亚洲精品天堂在线观看| 最新亚洲精品国自产在线观看| 国产精品毛片一区二区三区四区| 在线三级av| 黄网站app在线观看下载视频大全官网| 国产在线精品一区二区不卡| 91涩漫在线观看c| 国产麻豆精品高清在线播放 | 国产黄色网页| 成年人在线观看| jizz国产| av免费在线免费| 国产色a在线观看| www狠狠操| 国产91足控脚交在线观看| 波多野结衣中文字幕久久| 国产亚洲精品自在线观看| 在线天堂中文| 国产黄大片在线观看画质优化| 人人在线视频| av在线中文| av一级在线| 激情亚洲综合网| 亚洲国产精华液| 性网站在线播放| 国产精品jvid在线观看| 国产a国产a国产a| 亚洲一本大道| av在线首页| 99在线视频影院| 91欧洲在线视精品在亚洲| 狠狠综合久久久综合| 伊人免费视频| 日本一本久久| 九九热视频免费在线观看| 欧美高清视频| 国产精品一区牛牛影视| 超碰免费在线播放| 午夜羞羞小视频在线观看| 91高清国产| 亚洲尤物在线视频| 亚洲欧美精品中文第三| 永久免费不卡在线观看黄网站| 精品乱码一区二区三四区视频| 国产精品久久麻豆| 91免费日韩| 最新亚洲精品国自产在线观看| 中文字幕在线视频不卡| 国产欧美日本亚洲精品一4区| 国产黄色av免费看| 亚洲欧美日韩综合精品网| gogo在线高清视频| 黄色免费av| 国产极品视频| 日本福利午夜视频在线| 久久久久久久久亚洲精品| 国产一级网站视频在线| 国产成人午夜| 日本不卡影院| 国产精品久久久精品a级小说| 亚洲国产日韩在线人成电影| 亚洲欧美综合乱码精品成人网| 高清色视频在线观看|