




版權(quán)說(shuō)明:本文檔由用戶(hù)提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡(jiǎn)介
1、第。章寫(xiě)在前面我不想夸大或者貶低匯編語(yǔ)言。但我想說(shuō),匯編語(yǔ)言改變了20世紀(jì)的歷史。與前輩相比,我們這一代編程人員足夠的幸福,因?yàn)槲覀冇懈魇礁鳂拥木幊陶Z(yǔ)言,我們可以操作鍵盤(pán)、坐在顯示器面前,甚至使用鼠標(biāo)、語(yǔ)音識(shí)別。我們可以使用鍵盤(pán)、鼠標(biāo)來(lái)駕馭“個(gè)人計(jì)算機(jī)”,而不是和一群 人共享一臺(tái)使用笨重的繼電器、開(kāi)關(guān)去操作的巨型機(jī)。相比之下,我們的前輩不得不使用機(jī)器語(yǔ)言編寫(xiě)程序,他們甚至沒(méi)有最簡(jiǎn)單的匯編程序來(lái) 把助記符翻譯成機(jī)器語(yǔ)言,而我們可以從上千種計(jì)算機(jī)語(yǔ)言中選擇我們喜歡的一種,而匯編,雖然不是一種“常用”的具有“快速原型開(kāi)發(fā)”能 力的語(yǔ)言,卻也是我們可以選擇的語(yǔ)言中的一種。每種計(jì)算機(jī)都有自己的匯編語(yǔ)言一
2、一沒(méi)必要指望匯編語(yǔ)言的可移植性,選擇匯編,意味著選擇性能而不是可移植或便于調(diào)試。這份文檔中講述的ia32上的x86匯編語(yǔ)言。是x86匯編語(yǔ)言,此后的“匯編語(yǔ)言” 一詞,如果不明示則表示匯編語(yǔ)言是一種易學(xué),卻很難精通的語(yǔ)言?;叵氘?dāng)年,我從初學(xué)匯編到寫(xiě)出第一個(gè)可運(yùn)行的程序,只用了不到4個(gè)小時(shí);然而直到今天,我仍然不敢說(shuō)自己精通它。編寫(xiě)快速、高效、并且能夠讓處理器“很舒服地執(zhí)行”的程序是一件很困難的事情,如果利用業(yè)余時(shí)間學(xué)習(xí),通常需要2-3年的時(shí)間才能做到。這份教材并不期待能夠教給你大量的匯編語(yǔ)言技巧。對(duì)于讀者來(lái)說(shuō),x86匯編語(yǔ)言就在這里”。然而,不要僵化地局限于這份教材講述的內(nèi)容,因?yàn)樗荒芨嬖V你
3、匯編語(yǔ)言是“這樣一回事”。學(xué)好匯編語(yǔ)言,更多的要靠一個(gè)人的創(chuàng)造力于悟性,我可以告訴你我所知道的技巧,但肯定這是不夠的。一位對(duì)我的編程生涯產(chǎn)生過(guò)重要影響的人曾經(jīng)對(duì)我說(shuō)過(guò)這么一句話:寫(xiě)匯編語(yǔ)言程序 不是匯編語(yǔ)言最難的部分,創(chuàng)新才是。我想,愿意看這份文檔的人恐怕不會(huì)問(wèn)我“為什么要學(xué)習(xí)匯編語(yǔ)言”這樣的問(wèn)題;不過(guò),我還是想說(shuō)幾句:首先,匯編語(yǔ)言非常有用,我個(gè)人主張把它作為C語(yǔ)言的先修課程,因?yàn)橥ㄟ^(guò)學(xué)習(xí)匯編語(yǔ)言, 你可以了解到如何有效地設(shè)計(jì)數(shù)據(jù)結(jié)構(gòu),讓計(jì)算機(jī)處理得更快, 并使用更少的存儲(chǔ)空間;同時(shí),學(xué)習(xí)匯編語(yǔ)言可以讓你熟悉計(jì)算機(jī)內(nèi)部運(yùn)行機(jī)制,并且,有效地提高調(diào)試能力。就我個(gè)人的經(jīng)驗(yàn)而言,調(diào)試一個(gè)非結(jié)構(gòu)化的
4、程序的困難程 度,要比調(diào)試一個(gè)結(jié)構(gòu)化的程序的難度高很多,因?yàn)椤敖Y(jié)構(gòu)化”是以犧牲運(yùn)行效率來(lái)提高可讀性與可調(diào)試性,這對(duì)于完成一般軟件工程的編碼階 段是非常必要的。然而,在一些地方,比如,硬件驅(qū)動(dòng)程序、操作系統(tǒng)底層,或者程序中經(jīng)常需要執(zhí)行的代碼,結(jié)構(gòu)化程序設(shè)計(jì)的這些優(yōu)點(diǎn)有時(shí)就會(huì)被它的低效率所抹煞。另外,如果你想真正地控制自己的程序,只知道源代碼級(jí)的調(diào)試是遠(yuǎn)遠(yuǎn)不夠的。浮躁的人喜歡說(shuō),用 C+寫(xiě)程序足夠了,甚至說(shuō),他不僅僅掌握 C+,而且精通STL、MFC我不贊成這個(gè)觀點(diǎn),掌握上面的那些是每一個(gè)編程人 員都應(yīng)該做到的,然而 C+只是我們常用”的一種語(yǔ)言,它不是編程的全部。低層次的開(kāi)發(fā)者喜歡說(shuō),嘿,C+是
5、多么的強(qiáng)大,它可以做任何事情這不是事實(shí)。便于維護(hù)、調(diào)試,這些確實(shí)是我們的追求目標(biāo),但是,寫(xiě)程序不能僅僅追求這個(gè)目標(biāo),因?yàn)槲覀冏罱K的目的是滿(mǎn)足設(shè)計(jì)需求, 而不是個(gè)人非理性的理想。這份教材適合已經(jīng)學(xué)習(xí)過(guò)某種結(jié)構(gòu)化程序設(shè)計(jì)語(yǔ)言的讀者。其內(nèi)容基于我在1995年給別人講述匯編語(yǔ)言時(shí)所寫(xiě)的講義。當(dāng)然,如大家所希望的,它包含了最新的處理器所支持的特性,以及相應(yīng)的內(nèi)容。我假定讀者已經(jīng)知道了程序設(shè)計(jì)的一些基本概念,因?yàn)闆](méi)有這些是無(wú)法理解匯編語(yǔ)言程 序設(shè)計(jì)的;此外,我希望讀者已經(jīng)有了比較良好的程序設(shè)計(jì)基礎(chǔ),因?yàn)槿绻闳狈?duì)于結(jié)構(gòu)化程序設(shè)計(jì)的認(rèn)識(shí),編寫(xiě)匯編語(yǔ)言程序很可能很快就 破壞了你的結(jié)構(gòu)化編程習(xí)慣,大大降低程序
6、的可讀性、可維護(hù)性,最終讓你的程序陷于不得不廢棄的代碼堆之中?;旧?,這份文檔撰寫(xiě)的目標(biāo)是盡可能地便于自學(xué)。不過(guò),它對(duì)你也有一些要求,盡管不是很高,但我還是強(qiáng)調(diào)一下。學(xué)習(xí)匯編語(yǔ)言,你需要*膽量。不要害怕去接觸那些計(jì)算機(jī)的內(nèi)部工作機(jī)制。知識(shí)。了解計(jì)算機(jī)常用的數(shù)制,特別是二進(jìn)制、十六進(jìn)制、八進(jìn)制,以及計(jì)算機(jī)保存數(shù)據(jù)的方法。*開(kāi)放。接受匯編語(yǔ)言與高級(jí)語(yǔ)言的差異,而不是去指責(zé)它如何的不好讀。經(jīng)驗(yàn)。要求你擁有任意其他編程語(yǔ)言的一點(diǎn)點(diǎn)編程經(jīng)驗(yàn)。* 頭腦。祝您編程愉快!第一章匯編語(yǔ)言簡(jiǎn)介先說(shuō)一點(diǎn)和實(shí)際編程關(guān)系不太大的東西。當(dāng)然,如果你迫切的想看到更實(shí)質(zhì)的內(nèi)容,完全可以先跳過(guò)這一章。那么,我想可能有一個(gè)問(wèn)題對(duì)
7、于初學(xué)匯編的人來(lái)說(shuō)非常重要,那就是:匯編語(yǔ)言到底是什么?匯編語(yǔ)言是一種最接近計(jì)算機(jī)核心的編碼語(yǔ)言。不同于任何高級(jí)語(yǔ)言,匯編語(yǔ)言幾乎可以完全和機(jī)器語(yǔ)言一對(duì)應(yīng)。不錯(cuò),我們可以用機(jī)器語(yǔ)言寫(xiě)程序,但現(xiàn)在除了沒(méi)有匯編程序的那些電腦之外,直接用機(jī)器語(yǔ)言寫(xiě)超過(guò) 1000條以上指令的人大概只能算作那些被我們成為“圣人”的犧牲者一類(lèi)了。畢竟,記憶一些短小的助記符、由機(jī)器去考慮那些瑣碎的配位過(guò)程和檢查錯(cuò)誤,比記憶大量的隨計(jì)算機(jī)而改變的十六進(jìn)制代碼、可能弄 錯(cuò)而沒(méi)有任何提示要強(qiáng)的多。熟練的匯編語(yǔ)言編碼員甚至可以直接從十六進(jìn)制代碼中讀出匯編語(yǔ)言的大致意思。當(dāng)然,我們有更好的工具一一匯 編器和反匯編器。簡(jiǎn)單地說(shuō),匯編語(yǔ)
8、言就是機(jī)器語(yǔ)言的一種可以被人讀懂的形式,只不過(guò)它更容易記憶。至于宏匯編,則是包含了宏支持的匯編語(yǔ)言,這可以讓你 編程的時(shí)候更專(zhuān)注于程序本身,而不是忙于計(jì)算和重寫(xiě)代碼。匯編語(yǔ)言除了機(jī)器語(yǔ)言之外最接近計(jì)算機(jī)硬件的編程語(yǔ)言。由于它如此的接近計(jì)算機(jī)硬件,因此,它可以最大限度地發(fā)揮計(jì)算機(jī)硬件的性能。用匯編語(yǔ)言編寫(xiě)的程序的速度通常要比高級(jí)語(yǔ)言和C/C+快很多-幾倍,幾十倍,甚至成百上千倍。當(dāng)然,解釋語(yǔ)言,如解釋型LISP,沒(méi)有采用JIT技術(shù)的Java虛機(jī)中運(yùn)行的Java等等,其程序速度更無(wú)法與匯編語(yǔ)言程序同日而語(yǔ)。永遠(yuǎn)不要忽視匯編語(yǔ)言的高速。實(shí)際的應(yīng)用系統(tǒng)中,我們往往會(huì)用匯編徹底重寫(xiě)某些經(jīng)常調(diào)用的部分以期
9、獲得更高的性能。應(yīng)用匯編也許不能提 高你的程序的穩(wěn)定性,但至少,如果你非常小心的話,它也不會(huì)降低穩(wěn)定性;與此同時(shí),它可以大大地提高程序的運(yùn)行速度。我強(qiáng)烈建議所有的 軟件產(chǎn)品在最后 Release之前對(duì)整個(gè)代碼進(jìn)行Profile ,并適當(dāng)?shù)赜脜R編取代部分高級(jí)語(yǔ)言代碼。至少,匯編語(yǔ)言的知識(shí)可以告訴你一些有用的東西,比如,你有多少個(gè)寄存器可以用。有時(shí),手工的優(yōu)化比編譯器的優(yōu)化更為有效,而且,你可以完全控制程序的實(shí)際行為。我想我在羅嗦了??傊?,在我們結(jié)束這一章之前,我想說(shuō),不要在優(yōu)化的時(shí)候把希望完全寄托在編譯器上一一現(xiàn)實(shí)一些,再好的編譯器也不可能 總是產(chǎn)生最優(yōu)的代碼。簡(jiǎn)明X86匯編語(yǔ)言教程(2)-認(rèn)識(shí)
10、處理器作者:司徒彥南來(lái)源:互聯(lián)網(wǎng) 酷勤網(wǎng)收集 2008-04-11摘要酷勤網(wǎng)一般說(shuō)來(lái),處理器擁有對(duì)整個(gè)系統(tǒng)的所有總線的控制權(quán)。對(duì)于In tel平臺(tái)而言,處理器擁有對(duì)數(shù)據(jù)、內(nèi)存和控制總線的控制權(quán),根據(jù)指令控制整個(gè)計(jì)算機(jī)的運(yùn)行。在以后的章節(jié)中,我們還將討論系統(tǒng)中同時(shí)存在多個(gè)處理器的情況。處理器中有一些寄存器,可以保存特定長(zhǎng)度的數(shù)據(jù)。第二章認(rèn)識(shí)處理器中央處理器(CPU)在微機(jī)系統(tǒng)處于“領(lǐng)導(dǎo)核心”的地位。匯編語(yǔ)言被編譯成機(jī)器語(yǔ)言之后,將由處理器來(lái)執(zhí)行。那么,首先讓我們來(lái)了解一下處理 器的主要作用,這將幫助你更好地駕馭它。典型的處理器的主要任務(wù)包括從內(nèi)存中獲取機(jī)器語(yǔ)言指令,譯碼,執(zhí)行根據(jù)指令代碼管理它自
11、己的寄存器*根據(jù)指令或自己的的需要修改內(nèi)存的內(nèi)容 響應(yīng)其他硬件的中斷請(qǐng)求一般說(shuō)來(lái),處理器擁有對(duì)整個(gè)系統(tǒng)的所有總線的控制權(quán)。對(duì)于Intel平臺(tái)而言,處理器擁有對(duì)數(shù)據(jù)、內(nèi)存和控制總線的控制權(quán),根據(jù)指令控制整個(gè)計(jì)算機(jī)的運(yùn)行。在以后的章節(jié)中,我們還將討論系統(tǒng)中同時(shí)存在多個(gè)處理器的情況。處理器中有一些寄存器,這些寄存器可以保存特定長(zhǎng)度的數(shù)據(jù)。某些寄存器中保存的數(shù)據(jù)對(duì)于系統(tǒng)的運(yùn)行有特殊的意義。新的處理器往往擁有更多、具有更大字長(zhǎng)的寄存器,提供更靈活的取指、尋址方式。寄存器如前所述,處理器中有一些可以保存數(shù)據(jù)的地方被稱(chēng)作寄存器。寄存器可以被裝入數(shù)據(jù),你也可以在不同的寄存器之間移動(dòng)這些數(shù)據(jù),或者做類(lèi)似的事情。
12、基本上,像四則運(yùn)算、位運(yùn)算等這些計(jì)算操作,都主 要是針對(duì)寄存器進(jìn)行的。首先讓我來(lái)介紹一下80386上最常用的4個(gè)通用寄存器。先瞧瞧下面的圖形,試著理解一下:1EAK15e?AH : :AL蝕I150上圖中,數(shù)字表示的是位。我們可以看出,EAX是一個(gè)32-bit寄存器。同時(shí),它的低 16-bit又可以通過(guò)AX這個(gè)名字來(lái)訪問(wèn);AX又被分為高、低 8bit兩部分,分別由 AH和AL來(lái)表示。32-bit的寄存器EAX而它可以通過(guò)4種不同的對(duì)于EAX AX AH AL的改變同時(shí)也會(huì)影響與被修改的那些寄存器的值。從而事實(shí)上只存在一個(gè) 途徑訪問(wèn)。也許通過(guò)名字能夠更容易地理解這些寄存器之間的關(guān)系。EAX中的E
13、的意思是“擴(kuò)展的”,整個(gè)EAX的意思是擴(kuò)展的 AX X的意思Intel沒(méi)有明示,我個(gè)人認(rèn)為表示它是一個(gè)可變的量。而AH AL中的H和L分別代表高和低 。為什么要這么做呢?主要由于歷史原因。 早期的計(jì)算機(jī)是8位的,8086是第一個(gè)16位處理器,其通用寄存器的名字是 AX, BX等等;80386是Intel 推出的第一款I(lǐng)A-32系列處理器,所有的寄存器都被擴(kuò)充為 32位。為了能夠兼容以前的 16位應(yīng)用程序,80386不能將這些寄存器依舊命名為 AX BX,并且簡(jiǎn)單地將他們擴(kuò)充為 32位一一這將增加處理器在處理指令方面的成本。In tel微處理器的寄存器列表(在本章先只介紹80386的寄存器,MM
14、X寄存器以及其他新一代處理器的新寄存器將在以后的章節(jié)介紹)通用寄存器下面介紹通用寄存器及其習(xí)慣用法。顧名思義,通用寄存器是那些你可以根據(jù)自己的意愿使用的寄存器,修改他們的值通常不會(huì)對(duì)計(jì)算機(jī)的運(yùn)行 造成很大的影響。通用寄存器最多的用途是計(jì)算。EAX32-bit 寬通用寄存器。相對(duì)其他寄存器,在進(jìn)行運(yùn)算方面比較常用。在保護(hù)模式中,也可以作為內(nèi)存偏移指針(此時(shí),DS作為段 寄存器或選擇器)EBX32-bit 寬通用寄存器。通常作為內(nèi)存偏移指針使用(相對(duì)于EAX ECX EDX),DS是默認(rèn)的段寄存器或選擇器。在保護(hù)模式中,同樣可以起這個(gè)作用。ECX32-bit 寬通用寄存器。通常用于特定指令的計(jì)數(shù)。
15、在保護(hù)模式中,也可以作為內(nèi)存偏移指針(此時(shí),DS作為寄存器或段選擇器)。EDX32-bit 寬通用寄存器。在某些運(yùn)算中作為EAX的溢岀寄存器(例如乘、除)。在保護(hù)模式中,也可以作為內(nèi)存偏移指針(此時(shí),DS作為段 寄存器或選擇器)。上述寄存器同 EAX一樣包括對(duì)應(yīng)的16-bit和8-bit分組。用作內(nèi)存指針的特殊寄存器ESI32-bit 寬通常在內(nèi)存操作指令中作為“源地址指針”使用。當(dāng)然,ESI可以被裝入任意的數(shù)值,但通常沒(méi)有人把它當(dāng)作通用寄存器來(lái)用。DS是默認(rèn)段寄存器或選擇器。EDI32-bit 寬通常在內(nèi)存操作指令中作為“目的地址指針”使用。當(dāng)然,EDI也可以被裝入任意的數(shù)值,但通常沒(méi)有人把
16、它當(dāng)作通用寄存器來(lái)用。DS是默認(rèn)段寄存器或選擇器。EBP32-bit 寬這也是一個(gè)作為指針的寄存器。通常,它被高級(jí)語(yǔ)言編譯器用以建造堆棧幀來(lái)保存函數(shù)或過(guò)程的局部變量,不過(guò),還是那句話,你可以在其中保存你希望的任何數(shù)據(jù)。SS是它的默認(rèn)段寄存器或選擇器。注意,這三個(gè)寄存器沒(méi)有對(duì)應(yīng)的8-bit分組。換言之,你可以通過(guò)SI、DI、BP作為別名訪問(wèn)他們的低 16位,卻沒(méi)有辦法直接訪問(wèn)他們的低8位。段寄存器和選擇器實(shí)模式下的段寄存器到保護(hù)模式下?lián)u身一變就成了選擇器。不同的是,實(shí)模式下的“段寄存器”是16-bit的,而保護(hù)模式下的選擇器是32-bit的。CS代碼段,或代碼選擇器。同IP寄存器(稍后介紹)一同
17、指向當(dāng)前正在執(zhí)行的那個(gè)地址。處理器執(zhí)行時(shí)從這個(gè)寄存器指向的段(實(shí)模式)或內(nèi)存(保護(hù)模式)中獲取指令。除了跳轉(zhuǎn)或其他分支指令之外,你無(wú)法修改這個(gè)寄存器的內(nèi)容。DS數(shù)據(jù)段,或數(shù)據(jù)選擇器。這個(gè)寄存器的低16 bit連同ESI 一同指向的指令將要處理的內(nèi)存。同時(shí),所有的內(nèi)存操作指令默認(rèn)情況下都用它指定操作段(實(shí)模式)或內(nèi)存(作為選擇器,在保護(hù)模式。這個(gè)寄存器可以被裝入任意數(shù)值,然而在這么做的時(shí)候需要小心 一些。方法是,首先把數(shù)據(jù)送給AX,然后再把它從 AX傳送給DS(當(dāng)然,也可以通過(guò)堆棧來(lái)做).ES附加段,或附加選擇器。這個(gè)寄存器的低16 bit連同EDI 一同指向的指令將要處理的內(nèi)存。同樣的,這個(gè)寄
18、存器可以被裝入任意數(shù)值,方法和 DS類(lèi)似。FSf段或F選擇器(推測(cè)F可能是Free?)??梢杂眠@個(gè)寄存器作為默認(rèn)段寄存器或選擇器的一個(gè)替代品。它可以被裝入任何數(shù)值,方 法和DS類(lèi)似。GSG段或G選擇器(G的意義和F 樣,沒(méi)有在Intel的文檔中解釋?zhuān)?。它和FS幾乎完全一樣。SS堆棧段或堆棧選擇器。這個(gè)寄存器的低16bit連同ESP一同指向下一次堆棧操作(push和pop)所要使用的堆棧地址。這個(gè)寄存器也可以被裝入任意數(shù)值,你可以通過(guò)入棧和岀棧操作來(lái)給他賦值,不過(guò)由于堆棧對(duì)于很多操作有很重要的意義,因此,不正確的修改有可能造成對(duì)堆棧的破壞。*注意一定不要在初學(xué)匯編的階段把這些寄存器弄混。他們非常
19、重要,而一旦你掌握了他們,你就可以對(duì)他們做任意的操作了。段寄存器,或選 擇器,在沒(méi)有指定的情況下都是使用默認(rèn)的那個(gè)。這句話在現(xiàn)在看來(lái)可能有點(diǎn)稀里糊涂,不過(guò)你很快就會(huì)在后面知道如何去做。特殊寄存器(指向到特定段或內(nèi)存的偏移量):EIP這個(gè)寄存器非常的重要。這是一個(gè)32位寬的寄存器 ,冋CS冋指向即將執(zhí)行的那條指令的地址。不能夠直接修改這個(gè)寄存器的值,修改它的唯一方法是跳轉(zhuǎn)或分支指令。(CS是默認(rèn)的段或選擇器)ESP這個(gè)32位寄存器指向堆棧中即將被操作的那個(gè)地址。盡管可以修改它的值,然而并不提倡這樣做,因?yàn)槿绻悴皇欠浅C靼鬃约?在做什么,那么你可能造成堆棧的破壞。對(duì)于絕大多數(shù)情況而言,這對(duì)程序是
20、致命的。(SS是默認(rèn)的段或選擇器)IP: In struction Poi nter,指令指針SP: Stack Poi nter,堆棧指針(都是32位寬):好了,上面是最基本的寄存器。下面是一些其他的寄存器,你甚至可能沒(méi)有聽(tīng)說(shuō)過(guò)它們。CRO, CR2, CR3(控制寄存器)。舉一個(gè)例子,CRO的作用是切換實(shí)模式和保護(hù)模式。還有其他一些寄存器,DO, D1, D2, D3, D6 和D7(調(diào)試寄存器)。他們可以作為調(diào)試器的硬件支持來(lái)設(shè)置條件斷點(diǎn)。TR3, TR4, TR5, TR6 和TR?寄存器(測(cè)試寄存器)用于某些條件測(cè)試。1110FEDCA907654321由5AHF和LAHF躁存的部分
21、最后我們要說(shuō)的是一個(gè)在程序設(shè)計(jì)中起著非常關(guān)鍵的作用的寄存器:標(biāo)志寄存器。 I I I I I I I I I I I進(jìn)位標(biāo)擊邙奇詁驗(yàn)位 0 一輔助標(biāo)志0ZF香梶南 記符號(hào)標(biāo)恚 TF單步標(biāo)志一 IF中斷標(biāo)志DF萬(wàn)應(yīng)掠志0宙會(huì)詳細(xì)說(shuō)明)OF隘宙煤暮1ioplHT任霧眾套様志(te8&ld) 0Rf恢垠標(biāo)志(僅3胡口上)訓(xùn)虛擬方式標(biāo)志(R380U11)本節(jié)中部份表格來(lái)自David Jurgens的HelpPC 2.10快速參考手冊(cè)。在此謹(jǐn)表謝意。簡(jiǎn)明x86匯編語(yǔ)言教程(3)-使用寄存器作者:司徒彥南 來(lái)源:互聯(lián)網(wǎng)酷勤網(wǎng)收集 2008-04-11摘要酷勤網(wǎng)寄存器是處理器內(nèi)部的一些保存數(shù)據(jù)的存儲(chǔ)單元。僅
22、僅了解這些是不足以寫(xiě)出一個(gè)可用的匯編語(yǔ)言程序的,但你已經(jīng)可以大致讀懂一般匯編 語(yǔ)言程序了(不必驚訝,因?yàn)閰R編語(yǔ)言的祝記符和英文單詞非常接近),因?yàn)槟阋呀?jīng)了解了關(guān)于基本寄存器的絕大多數(shù)知識(shí)。2.2使用寄存器在前一節(jié)中的x86基本寄存器的介紹,對(duì)于一個(gè)匯編語(yǔ)言編程人員來(lái)說(shuō)是不可或缺的?,F(xiàn)在你知道,寄存器是處理器內(nèi)部的一些保存數(shù)據(jù)的存儲(chǔ) 單元。僅僅了解這些是不足以寫(xiě)出一個(gè)可用的匯編語(yǔ)言程序的,但你已經(jīng)可以大致讀懂一般匯編語(yǔ)言程序了(不必驚訝,因?yàn)閰R編語(yǔ)言的祝記符 和英文單詞非常接近),因?yàn)槟阋呀?jīng)了解了關(guān)于基本寄存器的絕大多數(shù)知識(shí)。在正式引入第一個(gè)匯編語(yǔ)言程序之前,我粗略地介紹一下匯編語(yǔ)言中不同進(jìn)制整
23、數(shù)的表示方法。如果你不了解十進(jìn)制以外的其他進(jìn)制,請(qǐng)把鼠標(biāo) 移動(dòng)到這里。匯編語(yǔ)言中的整數(shù)常量表示十進(jìn)制整數(shù)這是匯編器默認(rèn)的數(shù)制。直接用我們熟悉的表示方式表示即可。例如,1234表示十進(jìn)制的1234。不過(guò),如果你指定了使用其他數(shù)制,或者有凡事都進(jìn)行完整定義的小愛(ài)好,也可以寫(xiě)成十進(jìn)制數(shù)d或十進(jìn)制數(shù)D的形式。十六進(jìn)制數(shù)這是匯編程序中最常用的數(shù)制,我個(gè)人比較偏愛(ài)使用十六進(jìn)制表示數(shù)據(jù),至于為什么,以后我會(huì)作說(shuō)明。十六進(jìn)制數(shù)表示為0十六進(jìn)制數(shù)h或0十六進(jìn)制數(shù)H,其中,如果十六進(jìn)制數(shù)的第一位是數(shù)字,則開(kāi)頭的0可以省略。例如,7fffh, 0ffffh,等等。二進(jìn)制數(shù)這也是一種常用的數(shù)制。二進(jìn)制數(shù)表示為二進(jìn)制數(shù)
24、b或二進(jìn)制數(shù)B。一般程序中用二進(jìn)制數(shù)表示掩碼( mask code)等數(shù)據(jù)非常的直觀,但需要些很長(zhǎng)的數(shù)據(jù)(4位二進(jìn)制數(shù)相當(dāng)于一位十六進(jìn)制數(shù))。例如,1010110b。八進(jìn)制數(shù)八進(jìn)制數(shù)現(xiàn)在已經(jīng)不是很常用了(確實(shí)還在用,一個(gè)典型的例子是Unix的文件屬性)。八進(jìn)制數(shù)的形式是八進(jìn)制數(shù)q、八進(jìn)制數(shù)Q、八進(jìn)制數(shù)o、八進(jìn)制數(shù)0。例如,777Q。需要說(shuō)明的是,這些方法是針對(duì)宏匯編器(例如,MASM TASM NASM說(shuō)的,調(diào)試器默認(rèn)使用十六進(jìn)制表示整數(shù),并且不需要特別的聲明(例如,在調(diào)試器中直接用 FFFF表示十進(jìn)制的65535,用10表示十進(jìn)制的16)。現(xiàn)在我們來(lái)寫(xiě)一小段匯編程序,修改EAX EBX EC
25、X EDX 的數(shù)值。我們假定程序執(zhí)行之前,寄存器中的數(shù)值是全0:?XHLEAX00000000EBX00000000ECX00000000EDX00000000正如前面提到的,EAX的高16bit是沒(méi)有辦法直接訪問(wèn)的,而AX對(duì)應(yīng)它的低16bit,AH AL分別對(duì)應(yīng)AX的高、低8bit。mov eax, 012345678hmov ebx, 0abcdeffeh;將 012345678h 送入 eax;將 0abcdeffeh 送入 ebxmov ecx, 1mov edx, 2;將 000000001h 送入 ecx;將 000000002h 送入 edx則執(zhí)行上述程序段之后,寄存器的內(nèi)容變?yōu)?/p>
26、:?XHLEAX12345678EBXabcdeffeECX00000001EDX00000002那么,你已經(jīng)了解了 mov這個(gè)指令(mov是move的縮寫(xiě))的一種用法。它可以將數(shù)送到寄存器中。我們來(lái)看看下面的代碼:;ebx內(nèi)容送入eax;edx內(nèi)容送入ecxmov eax, ebxmov ecx, edx則寄存器內(nèi)容變?yōu)??XHLEAXabcdeffeEBXabcdeffeECX00000002EDX00000002我們可以看到,“ move之后,數(shù)據(jù)依然保存在原來(lái)的寄存器中。不妨把mov指令理解為“送入”,或“裝入”。練習(xí)題 把寄存器恢復(fù)成都為全 0的狀態(tài),然后執(zhí)行下面的代碼:mov ea
27、x, 0a1234hmov bx, axmov ah, blmov al, bh;將 0a1234h 送入 eax;將ax的內(nèi)容送入bx;將bl內(nèi)容送入ah;將bh內(nèi)容送入al思考:此時(shí),EAX的內(nèi)容將是多少?答案F面我們將介紹一些指令。在介紹指令之前,我們約定:使用Intel文檔中的寄存器表示方式reg32 32-bit 寄存器(表示 EAX、EBX 等)reg16 16-bit寄存器(在32位處理器中,這 AX、BX等)reg8? 8-bit寄存器(表示 AL、BH等)imm32 32-bit立即數(shù)(可以理解為常數(shù))imm16 16-bit 立即數(shù)imm8? 8-bit 立即數(shù)在寄存器中載
28、入另一寄存器,或立即數(shù)的值:mov reg32, (reg32 | imm8 | imm16 | imm32)mov reg32, (reg16 | imm8 | imm16)mov reg8, (reg8 | imm8)例如,mov eax, 010h表示,在eax中載入00000010h。需要注意的是,如果你希望在寄存器中裝入0,則有一種更快的方法,在后面我們將提到。交換寄存器的內(nèi)容:xchg reg32, reg32xchg reg16, reg16xchg reg8, reg8例如,xchg ebx,ecx,則ebx與ecx的數(shù)值將被交換。由于系統(tǒng)提供了這個(gè)指令,因此,采用其他方法交換
29、時(shí),速度將會(huì)較慢,并需要占用更多 的存儲(chǔ)空間,編程時(shí)要避免這種情況,即,盡量利用系統(tǒng)提供的指令,因?yàn)槎鄶?shù)情況下,這意味著更小、更快的代碼,同時(shí)也杜絕了錯(cuò)誤(如果 說(shuō)In tel的CPU在交換寄存器內(nèi)容的時(shí)候也會(huì)出錯(cuò),那么它就不用賣(mài)CPU 了。而對(duì)于你來(lái)說(shuō),檢查一行代碼的正確性也顯然比檢查更多代碼的正確性要容易)剛才的習(xí)題的程序用下面的代碼將更有效:mov eax, 0a1234hmov bx, axxchg ah, al將 0a1234h 送入 eax 將ax內(nèi)容送入bx 交換ah, al的內(nèi)容遞增或遞減寄存器的值:inc reg(8,16,32) dec reg(8,16,32)這兩個(gè)指令往
30、往用于循環(huán)中對(duì)指針的操作。需要說(shuō)明的是,某些時(shí)候我們有更好的方法來(lái)處理循環(huán),例如使用 后面的章節(jié)中介紹。將寄存器的數(shù)值與另一寄存器,或立即數(shù)的值相加,并存回此寄存器:loop指令,或rep前綴。這些將在add reg32, reg32 / imm(8,16,32)add reg16, reg16 / imm(8,16)add reg8, reg8 / imm(8)例如,add eax, edx ,將eax+edx的值存入eax。減法指令和加法類(lèi)似,只是將 add換成sub。需要說(shuō)明的是,與高級(jí)語(yǔ)言不同,匯編語(yǔ)言中,如果要計(jì)算兩數(shù)之和(差、積、商,或一般地說(shuō),運(yùn)算結(jié)果),那么必然有一個(gè)寄存器被用
31、來(lái)保存結(jié)果。在PASCAL中,我們可以用nA := nB+nC來(lái)讓nA保存nB+nC的結(jié)果,然而,匯編語(yǔ)言并不提供這種方法。如果你希望保持寄存器中的結(jié) 果,需要用另外的指令。這也從另一個(gè)側(cè)面反映了 “寄存器”這個(gè)名字的意義。數(shù)據(jù)只是“寄存”在那里。如果你需要保存數(shù)據(jù),那么需要將它 放到內(nèi)存或其他地方。類(lèi)似的指令還有 and、or、xor (與,或,異或)等等。它們進(jìn)行的是邏輯運(yùn)算。我們稱(chēng)add、mov sub、and等稱(chēng)為為指令助記符(這么叫是因?yàn)樗葯C(jī)器語(yǔ)言容易記憶,而起作用就是方便人記憶,某些資料中也稱(chēng)為指令、操 作碼、opcodeoperatio ncode 等);后面的參數(shù)成為操作數(shù),
32、一個(gè)指令可以沒(méi)有操作數(shù),也可以有一兩個(gè)操作數(shù),通常有一個(gè)操作數(shù)的指令,這 個(gè)操作數(shù)就是它的操作對(duì)象;而兩個(gè)參數(shù)的指令,前一個(gè)操作數(shù)一般是保存操作結(jié)果的地方,而后一個(gè)是附加的參數(shù)。我不打算在這份教程中用大量的篇幅介紹指令一一很多人做得比我更好,而且指令本身并不是重點(diǎn),如果你學(xué)會(huì)了如何組織語(yǔ)句,那么只要稍加學(xué)習(xí)就能輕易掌握其他指令。更多的指令可以參考Intel提供的資料。編寫(xiě)程序的時(shí)候,也可以參考一些在線參考手冊(cè)。Tech!Help和HelpPC 2.10盡管已經(jīng)很舊,但足以應(yīng)付絕大多數(shù)需要。聰明的讀者也許已經(jīng)發(fā)現(xiàn),使用sub eax, eax,或者xor eax, eax,可以得到與 mov e
33、ax,0類(lèi)似的效果。在高級(jí)語(yǔ)言中,你大概不會(huì)選擇用a=a-a來(lái)給a賦值,因?yàn)闇y(cè)試會(huì)告訴你這么做更慢,簡(jiǎn)直就是在自找麻煩,然而在匯編語(yǔ)言中,你會(huì)得到相反的結(jié)論,多數(shù)情況下,以由快到慢的速度排歹U,這三條指令將是 xor eax, eax 、sub eax, eax 禾口 mov eax, 0 。為什么呢?處理器在執(zhí)行指令時(shí),需要經(jīng)過(guò)幾個(gè)不同的階段:取指、譯碼、取數(shù)、執(zhí)行。我們反復(fù)強(qiáng)調(diào),寄存器是 CPU的一部分。從寄存器取數(shù),其速度很顯然要比從內(nèi)存中取數(shù)快。那么,不難理解,xor eax, eax 要比 mov eax, 0更快一些。那么,為什么a=a-a通常要比a=0慢一些呢?這和編譯器的優(yōu)化有
34、一定關(guān)系。多數(shù)編譯器會(huì)把ebp和偏移量來(lái)訪問(wèn)局部變量;程序中, x為a相對(duì)于本地堆的偏移量,在只包含一個(gè) 32-bita=a-a翻譯成類(lèi)似下面的代碼(通常,高級(jí)語(yǔ)言通過(guò) 整形變量的程序中,這個(gè)值通常是4):mov eax, dword ptr ebp-x sub eax, dword ptr ebp-x mov dword ptr ebp-x,eax而把a(bǔ)=0翻譯成mov dword ptr ebp-x, 0上面的翻譯只是示意性的,略去了很多必要的步驟,如保護(hù)寄存器內(nèi)容、恢復(fù)等等。如果你對(duì)與編譯程序的實(shí)現(xiàn)過(guò)程感興趣,可以參考相應(yīng)的書(shū)籍。多數(shù)編譯器(特別是C/C+編譯器,如Microsoft V
35、isualC+ )都提供了從源代碼到宏匯編語(yǔ)言程序的附加編譯輸出選項(xiàng)。這種情況下,你可以很方便地了解編譯程序執(zhí)行的輸出結(jié)果;如果編譯程序沒(méi)有提供這樣的功能也沒(méi)有關(guān)系,調(diào)試器會(huì)讓你看到編譯器的編譯結(jié)果。如果你明確地知道編譯器編譯出的結(jié)果不是最優(yōu)的,那就可以著手用匯編語(yǔ)言來(lái)重寫(xiě)那段代碼了。怎么確認(rèn)是否應(yīng)該用匯編語(yǔ)言重寫(xiě)呢?使用匯編語(yǔ)言重寫(xiě)代碼之前需要確認(rèn)的幾件事情首先,這種優(yōu)化最好有明顯的效果。比如,一段循環(huán)中的計(jì)算, 等等。一條語(yǔ)句的執(zhí)行時(shí)間是很短的,現(xiàn)在新的CPU的指令周期都在0.000000001s以下,Intel甚至已經(jīng)做岀了 4GHz主頻(主頻的倒數(shù)是時(shí)鐘周期)的CPU,如果你的代碼自始
36、至終只執(zhí)行一次,并且你只是減少了幾個(gè)時(shí)鐘周期的執(zhí)行時(shí)間,那么改變將是無(wú)法讓人察覺(jué)的;很多情況下,這種優(yōu)化”并不被提倡,盡管它確實(shí)減少了執(zhí)行時(shí)間,但為此需要付岀大量的時(shí)間、人力,多數(shù)情況下得不償失(極端情況,比如你的設(shè)備內(nèi)存價(jià)格非常昂貴的時(shí)候,這種優(yōu)化也許會(huì)有意義)。其次,確認(rèn)你已經(jīng)使用了 最好的算法,并且,你優(yōu)化的程序的實(shí)現(xiàn)是正確的。匯編語(yǔ)言能夠提供同樣算法的最快實(shí)現(xiàn),然而,它并不是萬(wàn)金油,更不是解決一切的靈丹妙藥。用高級(jí)語(yǔ)言實(shí)現(xiàn)一種好的算法,不一定會(huì)比匯編語(yǔ)言實(shí)現(xiàn)一種差的算法更慢。不過(guò)需要注意的是,時(shí)間、空間復(fù)雜 度最小的算法不一定就是解決某一特定問(wèn)題的最佳算法。舉例說(shuō),快速排序在完全逆序的
37、情況下等價(jià)于冒泡排序,這時(shí)其他方法就比它快。同時(shí),用匯編語(yǔ)言?xún)?yōu)化一個(gè)不正確的算法實(shí)現(xiàn),將給調(diào)試帶來(lái)很大的麻煩。最后,確認(rèn)你已經(jīng)將高級(jí)語(yǔ)言編譯器的性能 發(fā)揮到極致。Microsoft的編譯器在 RELEASE模式和DEBUG模式會(huì)有差異相當(dāng)大的輸岀,而對(duì)于 GNU系列的編譯器而言,不同級(jí)別的優(yōu)化也會(huì)生成幾乎完全不同的代碼。此外,在編程時(shí)對(duì)于問(wèn)題的嚴(yán)格定義,可以極大地幫助編譯器的優(yōu)化 過(guò)程。如何優(yōu)化高級(jí)語(yǔ)言代碼,使其編譯結(jié)果最優(yōu)超岀了本教程的范圍,但如果你不能確認(rèn)已經(jīng)發(fā)揮了編譯器的最大效能,用匯編語(yǔ)言往往是一 種更為費(fèi)力的方法。還有一點(diǎn)非常重要,那就是你明白自己做的是什么。好的高級(jí)語(yǔ)言編譯器有時(shí)會(huì)
38、有一些讓人難以理解的行為,比如,重新排列指令順序,等等。如果你發(fā)現(xiàn)這種情況,那么優(yōu)化的時(shí)候就應(yīng)該小心一一編譯器很可能比你擁有更多的關(guān)于處理器的知識(shí),例如,對(duì)于一個(gè)超標(biāo)量處理器,編譯器會(huì)對(duì)指令序列進(jìn)行 封包”使他們盡可能的并行執(zhí)行;此外,宏匯編器有時(shí)會(huì)自動(dòng)插入一些nop指令,其作用是將指令湊成整數(shù)字長(zhǎng)( 32-bit,對(duì)于16-bit處理器,是16-bit )這些都是提高代碼性能的必要措施,如果你不了解處理器,那么最好不要改動(dòng)編譯器生成的代碼,因?yàn)檫@種情況 下,盲目的修改往往不會(huì)得到預(yù)期的效果。曾經(jīng)在一份雜志上看到過(guò)有人用純機(jī)器語(yǔ)言編寫(xiě)程序。不清楚到底這是不是編輯的失誤,因?yàn)橐粋€(gè)頭腦正常的人恐怕
39、不會(huì)這么做程序,即使它不 長(zhǎng)、也不復(fù)雜。首先,匯編器能夠完成某些封包操作,即使不行,也可以用db偽指令來(lái)寫(xiě)指令;用匯編語(yǔ)言寫(xiě)程序可以防止很多錯(cuò)誤的發(fā)生,同時(shí),它還減輕了人的負(fù)擔(dān),很顯然,“完全用機(jī)器語(yǔ)言寫(xiě)程序”是完全沒(méi)有必要的,因?yàn)閰R編語(yǔ)言可以做出完全一樣的事情,并且你可以依賴(lài)它,因?yàn)橛?jì)算機(jī)不會(huì)出錯(cuò),而人總有出錯(cuò)的時(shí)候。此外,如前面所言,如果用高級(jí)語(yǔ)言實(shí)現(xiàn)程序的代價(jià)不大(例如,這段代碼在程序的整個(gè)執(zhí)行過(guò)程 中只執(zhí)行一遍,并且,這一遍的執(zhí)行時(shí)間也小于一秒),那么,為什么不用高級(jí)語(yǔ)言實(shí)現(xiàn)呢?一些比較狂熱的編程愛(ài)好者可能不太喜歡我的這種觀點(diǎn)。比方說(shuō),他們可能希望精益求精地優(yōu)化每一字節(jié)的代碼。但多數(shù)情
40、況下我們有更重要的 事情,例如,你的算法是最優(yōu)的嗎?你已經(jīng)把程序在高級(jí)語(yǔ)言許可的范圍內(nèi)優(yōu)化到盡頭了嗎?并不是所有的人都有資格這樣說(shuō)。匯編語(yǔ)言是這樣 一件東西,它足夠的強(qiáng)大,能夠控制計(jì)算機(jī),完成它能夠?qū)崿F(xiàn)的任何功能;同時(shí),因?yàn)樗膹?qiáng)大,也會(huì)提高開(kāi)發(fā)成本,并且,難于維護(hù)。因此, 我個(gè)人的建議是,如果在軟件開(kāi)發(fā)中使用匯編語(yǔ)言,則應(yīng)在軟件接近完成的時(shí)候使用,這樣可以減少很多不必要的投入。第二章中,我介紹了 x86系列處理器的基本寄存器。這些寄存器對(duì)于x86兼容處理器仍然是有效的,如果你偏愛(ài)AMD勺CPU那么使用這些寄存器的程序同樣也可以正常運(yùn)行。不過(guò)現(xiàn)在說(shuō)用匯編語(yǔ)言進(jìn)行優(yōu)化還為時(shí)尚早一一不可能寫(xiě)程序,
41、而只操作這些寄存器,因?yàn)檫@樣只能完成非常簡(jiǎn)單的操作,既然是簡(jiǎn)單的操作, 那可能就會(huì)讓人覺(jué)得乏味,甚至找一臺(tái)足夠快的機(jī)器窮舉它的所有結(jié)果(如果可以窮舉的話),并直接寫(xiě)程序調(diào)用,因?yàn)檫@樣通常會(huì)更快。但話 說(shuō)回來(lái),看完接下來(lái)的兩章一一內(nèi)存和堆棧操作,你就可以獨(dú)立完成幾乎所有的任務(wù)了,配合第五章中斷、第六章子程序的知識(shí),你將知道如何 駕馭處理器,并讓它為你工作。簡(jiǎn)明x86匯編語(yǔ)言教程(4)-操作內(nèi)存酷勤網(wǎng)收集 2008-04-11作者:司徒彥南 來(lái)源:互聯(lián)網(wǎng)摘要酷勤網(wǎng)作為匯編語(yǔ)言的抽象,C語(yǔ)言擁有 指針”這個(gè)數(shù)據(jù)類(lèi)型。在匯編語(yǔ)言中,幾乎所有對(duì)內(nèi)存的操作都是由對(duì)給定地址的內(nèi)存進(jìn)行訪問(wèn)來(lái)完成的。 這樣,在
42、匯編語(yǔ)言中,絕大多數(shù)操作都要和指針產(chǎn)生或多或少的聯(lián)系,因此,匯編語(yǔ)言中同樣會(huì)出現(xiàn)C程序中常見(jiàn)的緩沖區(qū)溢出問(wèn)題。第三章操作內(nèi)存在前面的章節(jié)中,我們已經(jīng)了解了寄存器的基本使用方法。而正如結(jié)尾提到的那樣,僅僅使用寄存器做一點(diǎn)運(yùn)算是沒(méi)有什么太大意義的,畢竟它 們不能保存太多的數(shù)據(jù),因此,對(duì)編程人員而言,他肯定迫切地希望訪問(wèn)內(nèi)存,以保存更多的數(shù)據(jù)。我將分別介紹如何在保護(hù)模式和實(shí)模式操作內(nèi)存,然而在此之前,我們先熟悉一下這兩種模式中內(nèi)存的結(jié)構(gòu)。3.1實(shí)模式事實(shí)上,在實(shí)模式中,內(nèi)存比保護(hù)模式中的結(jié)構(gòu)更令人困惑。內(nèi)存被分割成段,并且,操作內(nèi)存時(shí),需要指定段和偏移量。不過(guò),理解這些概念 是非常容易的事情。請(qǐng)看下
43、面的圖:內(nèi)存段段-寄存器這種格局是早期硬件電路限制留下的一個(gè)傷疤。地址總線在當(dāng)時(shí)有20-bit。然而20-bit的地址不能放到16-bit的寄存器里,這意味著有4-bit必須放到別的地方。因此,為了訪問(wèn)所有的內(nèi)存,必須使用兩個(gè)16-bit寄存器。這一設(shè)計(jì)上的折衷方案導(dǎo)致了今天的段-偏移量格局。最初的設(shè)計(jì)中,其中一個(gè)寄存器只有4-bit有效,然而為了簡(jiǎn)化程序,兩個(gè)寄存器都是16-bit有效,并在執(zhí)行時(shí)求出加權(quán)和來(lái)標(biāo)識(shí)20-bit地址。偏移量是16-bit的,因此,一個(gè)段是 64KB。下面的圖可以幫助你理解20-bit地址是如何形成的:4 hitIB hit r1段-偏移量標(biāo)識(shí)的地址通常記做 段:
44、偏移量 的形式。由于這樣的結(jié)構(gòu),一個(gè)內(nèi)存有多個(gè)對(duì)應(yīng)的地址。例如,0000:0010和0001:0000指的是同一內(nèi)存地址。又如,0000:1234 = 0123:0004 = 0120:0034 = 0100:02340001:1234 = 0124:0004 = 0120:0044 = 0100:024416,而不是一個(gè)“全新”的段。反之,在偏移量上加16也和在段上加1等價(jià)。某些時(shí)候,作為負(fù)面影響之一,在段上加 1相當(dāng)于在偏移量上加據(jù)此認(rèn)為段的“粒度”是 16字節(jié)。練習(xí)題嘗試一下將下面的地址轉(zhuǎn)化為20bit的地址:2EA8:D678 26CF:8D5F 453A:CFAD 2933:31A6
45、 5924:DCCF694E:175A 2B3C:D218 728F:6578 68E1:A7DC 57EC:AEEA稍高一些的要求是,寫(xiě)一個(gè)程序?qū)⒍螢锳X偏移量為BX的地址轉(zhuǎn)換為20bit的地址,并保存于 EAX中。上面習(xí)題的答案我們現(xiàn)在可以寫(xiě)一個(gè)真正的程序了。經(jīng)典程序:Hello, world;應(yīng)該得到一個(gè)29字節(jié)的.com文件.MODEL TINY.CODECR equ 13LF equ 10TERMINATOR equ $ORG 100hMai n PROCmov dx,offset sMessagemov ah,9int 21hmov ax,4c00hint 21hMai n END
46、PsMessage:DB Hello, World!DB CR LF,TERMINATOREND Mai n;.COM文件的內(nèi)存模型是,TINY?;代碼段開(kāi)始;回車(chē);換行;DOS字符串結(jié)束符;代碼起始地址為 CS:0100h;令 DS:DX 指向 Message;int 21h(DOS 中斷)功能 9 -;顯示字符串到標(biāo)準(zhǔn)輸出設(shè)備;int 21h 功能 4ch -;終止程序并返回 AL的錯(cuò)誤代碼程序結(jié)束的同時(shí)指定入口點(diǎn)為Main那么,我們需要解釋很多東西。首先,作為匯編語(yǔ)言的抽象,C語(yǔ)言擁有“指針”這個(gè)數(shù)據(jù)類(lèi)型。在匯編語(yǔ)言中,幾乎所有對(duì)內(nèi)存的操作都是由對(duì)給定地址的內(nèi)存進(jìn)行訪問(wèn)來(lái)完成的。這樣,在
47、匯編語(yǔ)言中,絕大多數(shù)操作都要和指針產(chǎn)生或多或少的聯(lián)系。這里我想強(qiáng)調(diào)的是,由于這一特性,匯編語(yǔ)言中同樣會(huì)出現(xiàn)C程序中常見(jiàn)的緩沖區(qū)溢出問(wèn)題。如果你正在設(shè)計(jì)一個(gè)與安全有關(guān)的系統(tǒng),那么最好是仔細(xì)檢查你用到的每一個(gè)串,例如,它們是否一定能夠以你預(yù)期的方式結(jié)束,以及(如果使用的話)你的緩沖區(qū)是否能保證實(shí)際可能輸入的數(shù) 據(jù)不被寫(xiě)入到它以外的地方。作為一個(gè)匯編語(yǔ)言程序員,你有義務(wù)檢查每一行代碼的可用性。程序中的equ偽指令是宏匯編特有的,它的意思接近于C或Pascal中的const (常量)。多數(shù)情況下,equ偽指令并不為符號(hào)分配空間。此外,匯編程序執(zhí)行一項(xiàng)操作是非常繁瑣的,通常,在對(duì)與效率要求不高的地方,我
48、們習(xí)慣使用系統(tǒng)提供的中斷服務(wù)來(lái)完成任務(wù)。例如本例中的中斷21h,它是DOS時(shí)代的中斷服務(wù),在 Windows中,它也被認(rèn)為是 Windows API的一部分(這一點(diǎn)可以在 Microsoft 的文檔中查到)。中斷可 以被理解為高級(jí)語(yǔ)言中的子程序,但又不完全一樣一一中斷使用系統(tǒng)棧來(lái)保存當(dāng)前的機(jī)器狀態(tài),可以由硬件發(fā)起,通過(guò)修改機(jī)器狀態(tài)字來(lái)反饋信 息,等等。那么,最后一段通過(guò) DB存放的數(shù)據(jù)到底保存在哪里了呢?答案是緊挨著代碼存放。在匯編語(yǔ)言中,DB和普通的指令的地位是相同的。如果你的匯編程序并不知道新的助記符(例如,新的處理器上的CPUID指令),而你很清楚,那么可以用DB機(jī)器碼的方式強(qiáng)行寫(xiě)下指令
49、。這意味著,你可以超越匯編器的能力撰寫(xiě)匯編程序,然而,直接用機(jī)器碼編程是幾乎肯定是一件費(fèi)力不討好的事一一匯編器廠商會(huì)經(jīng)常更新它所支持的指令集 以適應(yīng)市場(chǎng)需要,而且,你可以期待你的匯編其能夠產(chǎn)生正確的代碼,因?yàn)闄C(jī)器查表是不會(huì)出錯(cuò)的。既然機(jī)器能夠幫我們做將程序轉(zhuǎn)換為代碼這 件事情,那么為什么不讓它來(lái)做呢?細(xì)心的讀者不難發(fā)現(xiàn),在程序中我們沒(méi)有對(duì)DS進(jìn)行賦值。那么,這是否意味著程序的結(jié)果將是不可預(yù)測(cè)的呢?答案是否定的。DOS(或 Windows中的MS-DOS/M在加載.com文件的時(shí)候,會(huì)對(duì)寄存器進(jìn)行很多初始化。.com文件被限制為小于 64KB這樣,它的代碼段、數(shù)據(jù)段都被裝入同樣的數(shù)值(即,初始狀
50、態(tài)下DS=CS。也許會(huì)有人說(shuō),“嘿,這聽(tīng)起來(lái)不太好,一個(gè)64KB的程序能做得了什么呢?還有,你吹得天花亂墜的堆棧段在什么地方? ”那么,我們來(lái)看看下面這個(gè)新的Hello world 程序,它是一個(gè) EXE文件,在DOS實(shí)模式下運(yùn)行。;應(yīng)該得到一個(gè) 561字節(jié)的EXE文件.MODEL SMALL.STACK 200h;采用“ SMALL內(nèi)存模型;堆棧段CR equ 13LF equ 10TERMINATOR equ $;回車(chē);換行;DOS字符串結(jié)束符.DATA;定義數(shù)據(jù)段Message DB Hello, World rDB CR丄 F,TERMINATOR定義顯示串DB CR丄 F,TERMI
51、NATOR定義顯示串.CODEMai n PROCmov ax, DGROUPmov ds, axmov dx, offset Messagemov ah, 9int 21hmov ax, 4c00hint 21hMai n ENDPEND mai n定義代碼段將數(shù)據(jù)段加載到DS寄存器設(shè)置DX顯示終止程序561字節(jié)?實(shí)現(xiàn)相同功能的程序大了這么多!為什么呢?我們看到,程序擁有了完整的堆棧段、數(shù)據(jù)段、代碼段,其中堆棧段足足占掉了512字節(jié),其余的基本上沒(méi)什么變化。分成多個(gè)段有什么好處呢?首先,它讓程序顯得更加清晰一一你肯定更愿意看一個(gè)結(jié)構(gòu)清楚的程序,代碼中hard-coded的字符串、數(shù)據(jù)讓人覺(jué)得
52、費(fèi)解。比如,mov dx, 0152h肯定不如 mov dx, offset Message來(lái)的親切。此外,通過(guò)分段你可以使用更多的內(nèi)存,比如,代碼段騰出的空間可以做更多的事情。exe文件另一個(gè)吸引人的地方是它能夠?qū)崿F(xiàn)“重定位”?,F(xiàn)在你不需要指定程序入口點(diǎn)的地址了,因?yàn)橄到y(tǒng)會(huì)找到你的程序入口點(diǎn),而不是死板的 100h。程序中的符號(hào)也會(huì)在系統(tǒng)加載的時(shí)候重新賦予新的地址。exe程序能夠保證你的設(shè)計(jì)容易地被實(shí)現(xiàn),不需要考慮太多的細(xì)節(jié)。當(dāng)然,我們的主要目的是將匯編語(yǔ)言作為高級(jí)語(yǔ)言的一個(gè)有用的補(bǔ)充。如我在開(kāi)始提到的那樣,真正完全用匯編語(yǔ)言實(shí)現(xiàn)的程序不一定就好,因 為它不便于維護(hù),而且,由于結(jié)構(gòu)的原因,你
53、也不太容易確保它是正確的;匯編語(yǔ)言是一種非結(jié)構(gòu)化的語(yǔ)言,調(diào)試一個(gè)精心設(shè)計(jì)的匯編語(yǔ)言程序,即使對(duì)于一個(gè)老手來(lái)說(shuō)也不啻是一場(chǎng)惡夢(mèng),因?yàn)槟愫芸赡艿舻絼e人預(yù)設(shè)的“陷阱”中一一這些技巧確實(shí)提高了代碼性能,然而你很可能不理解它,于是你把它改掉,接著就發(fā)現(xiàn)程序徹底敗掉了。使用匯編語(yǔ)言加強(qiáng)高級(jí)語(yǔ)言程序時(shí),你要做的通常只是使用匯編指令,而不必搭建完整的匯編程 序。絕大多數(shù)(也是目前我遇到的全部)C/C+編譯器都支持內(nèi)嵌匯編,即在程序中使用匯編語(yǔ)言,而不必撰寫(xiě)單獨(dú)的匯編語(yǔ)言程序一一這可以節(jié)省你的不少精力,因?yàn)榍懊嬷v述的那些偽指令,如equ等,都可以用你熟悉的高級(jí)語(yǔ)言方式來(lái)編寫(xiě),編譯器會(huì)把它轉(zhuǎn)換為適當(dāng)?shù)男问健P枰f(shuō)
54、明的是,在高級(jí)語(yǔ)言中一定要注意編譯結(jié)果。編譯器會(huì)對(duì)你的匯編程序做一些修改,這不一定符合你的要求(附帶說(shuō)一句,有時(shí)編譯器會(huì)很聰明地調(diào)整指令順序來(lái)提高性能,這種情況下最好測(cè)試一下哪種寫(xiě)法的效果更好),此時(shí)需要做一些更深入的修改,或者用db來(lái)強(qiáng)制編碼。3.2保護(hù)模式實(shí)模式的東西說(shuō)得太多了,盡管我已經(jīng)刪掉了許多東西,并把一些原則性的問(wèn)題拿到了這一節(jié)討論。這樣做不是沒(méi)有理由的一一保護(hù)模式才是現(xiàn)在的程序(除了操作系統(tǒng)的底層啟動(dòng)代碼)最常用的CPU模式。保護(hù)模式提供了很多令人耳目一新的功能,包括內(nèi)存保護(hù)(這是保護(hù)模式這個(gè)名字的來(lái)源)、進(jìn)程支持、更大的內(nèi)存支持,等等。對(duì)于一個(gè)編程人員來(lái)說(shuō),能“偷懶”是一件令人愉快的事情。這里“偷懶”是說(shuō)把“應(yīng)該”由系統(tǒng)做的事情做的事情全都交給系統(tǒng)。為什么呢?這出自一個(gè)基本思想一一人總有犯錯(cuò)誤的時(shí)候,然而規(guī)則不會(huì),正確地了解規(guī)則之后,你可以期待它
溫馨提示
- 1. 本站所有資源如無(wú)特殊說(shuō)明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶(hù)所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒(méi)有圖紙預(yù)覽就沒(méi)有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫(kù)網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶(hù)上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶(hù)上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶(hù)因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 北京2025年北京市律師協(xié)會(huì)秘書(shū)處招聘筆試歷年參考題庫(kù)附帶答案詳解
- 危化品駕駛員聘用合同
- 離婚起草協(xié)議書(shū)格式
- 二零二五商場(chǎng)廣告租賃合同
- 二零二五住宅物業(yè)服務(wù)合同示范文本
- 二零二五版合同股權(quán)轉(zhuǎn)讓協(xié)議
- 社區(qū)醫(yī)療合作協(xié)議書(shū)二零二五年
- 委托物業(yè)管理合同標(biāo)準(zhǔn)樣本
- 變更租賃合同的三方協(xié)議
- 體育中心場(chǎng)地出租合同二零二五年
- DLT 5285-2018 輸變電工程架空導(dǎo)線(800mm以下)及地線液壓壓接工藝規(guī)程
- Unit3TravelPlansLesson3(課件)人教新起點(diǎn)版英語(yǔ)四年級(jí)下冊(cè)
- 《3-6歲兒童學(xué)習(xí)與發(fā)展指南》考試復(fù)習(xí)題庫(kù)350題(含答案)
- 小學(xué)生文明如廁
- JBT 11699-2013 高處作業(yè)吊籃安裝、拆卸、使用技術(shù)規(guī)程
- 2024年全國(guó)版圖知識(shí)競(jìng)賽(小學(xué)組)考試題庫(kù)大全(含答案)
- 專(zhuān)題08 八年級(jí)下冊(cè)易混易錯(cuò)總結(jié)-備戰(zhàn)2024年中考道德與法治一輪復(fù)習(xí)知識(shí)清單(全國(guó)通用)
- 中集集團(tuán)招聘題庫(kù)
- 贛政通管理員操作手冊(cè)
- 2024年ISTQB認(rèn)證筆試歷年真題薈萃含答案
- 2021年以工代賑項(xiàng)目實(shí)施工作指南(試行)
評(píng)論
0/150
提交評(píng)論