0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

匯編語言:一個函數(shù)的編譯器之旅

如意 ? 來源:編程技術(shù)宇宙微信公眾號 ? 作者:軒轅之風(fēng)O ? 2020-09-04 11:48 ? 次閱讀

我是一個函數(shù)

我是一個函數(shù),名叫 str_upper,我可以把輸入的字符串從小寫變成大寫。不信你看,我長這樣:

char* str_upper(char* str, int len) {charupper[256];if (len 》= 256 || len 《= 0) return nullptr;for (int i = 0; i 《 len; i++) {if (str[i] 》= ‘a(chǎn)’ && str[i] 《= ‘z’) {upper[i] = str[i] - 32;} else {upper[i] = str[i]; } }returnupper;}

上面是我的源代碼形式,聽我的好朋友 str_lower 說,一會兒我們就要一起被送到一個叫編譯器的地方加工處理了,我心里害怕極了。

編譯器之旅

沒多久,我們就來到了這里,一座很龐大到高樓,里面有好多精密的機器在不停的運轉(zhuǎn)著。

一進入大廳,好多函數(shù)代碼在這里排隊等待。

我抬頭向上望去,不知道有多少層樓,每一層都有一個指示牌,從下往上分別寫著:

預(yù)處理

詞法分析

語法分析

語義分析

···

再往上太遠(yuǎn)就看不太清楚了。

所有的函數(shù)代碼按照文件為單位排好隊,靜靜地等待著。

不過沒有等太久,就輪到了我們這一隊。

來了一個工作人員把我們帶到了一個房間,讓我們都好好躺著,一臺機器快速的從頭到尾掃描了一遍,將我們所在文件中出現(xiàn)的 #include 和 #define 全部給替換掉了。

接著,通過房間里的電梯,將我們送上了二樓。

接下來的一段時間,我們在好幾層樓都做了“體檢”,每個函數(shù)都被那些像 CT 一樣的機器照了個遍。

不一會兒,來到了編譯層,這一層有一個特別奇怪的機器,我看到一個個函數(shù)被送了進去,出來的時候都變了樣子。不僅如此,接待處的工作人員看起來很兇,我這下更加緊張了。

函數(shù)調(diào)用約定

工作人員拿到了我的資料,瞅了幾眼,問到:“請問你的調(diào)用約定是什么?”

我有些懵,不太懂他的意思,小聲問到:“不好意思,你剛問什么?”

工作人員有點不耐煩了,提高了音量,“我是問你調(diào)用約定是什么?調(diào)用約定??!”

看見我仍然一臉茫然,工作人員直接給我的資料上調(diào)用約定那一欄蓋上了一個標(biāo)記:cdecl。

我有點摸不著頭腦,同行的小伙伴 str_lower 拽了我一下說到:“他是在問你函數(shù)的調(diào)用約定,就是約定調(diào)用函數(shù)的方式,涉及怎么傳遞參數(shù),誰來恢復(fù)調(diào)用棧等”

他這一說我才反映過來,“這個調(diào)用約定都有哪些可選的呢?”

“一般有三種:”

cdcel,參數(shù)從右往左入棧,主調(diào)函數(shù)負(fù)責(zé)恢復(fù)棧平衡

stdcall,參數(shù)從右往左入棧,被調(diào)函數(shù)負(fù)責(zé)恢復(fù)棧平衡

fastcall,參數(shù)通過寄存器傳遞,寄存器不夠再用棧傳遞

“他剛才看你沒有顯式聲明,就默認(rèn)給你 cdecl 的方式了”,小伙伴繼續(xù)說到。

我點了點頭,原來調(diào)用個函數(shù)還有這么多講究吶!

Stack Canary

“別閑聊了,快進去吧!”,工作人員催我了。

我準(zhǔn)備走向那臺可怕的機器。

“唉,等一下”,正緊張著,工作人員又叫住了我。

我回頭看去,工作人員正招手讓我過去。

“你好,是我的代碼有什么問題嗎?”,我緊張的問到,生怕有錯誤被打回去,連累我們整個文件都要被遣返。

“不是,是我注意到你的函數(shù)里有一個局部數(shù)組,需要給你加一下棧溢出保護”,工作人員說到。

我看了下我的代碼,確實有一個局部字符數(shù)組:

charupper[256];

“棧溢出保護是什么啊?”,我小聲問到。

工作人員沒有搭理我,忙著給我的資料上加?xùn)|西。

旁邊的小伙伴又把我拽了過去,說到:“咱們函數(shù)里面定義的局部變量、參數(shù)是存放在線程棧里面的。線程要不斷游走在不同的函數(shù)中,調(diào)用函數(shù)后為了能回到原來的地方,調(diào)用之前把返回地址也放在了線程棧里。就像這樣,你看會不會有什么問題:”

匯編語言:一個函數(shù)的編譯器之旅

我仔細(xì)看了下,“哦,要是越界訪問我的 upper 數(shù)組,那就可以修改返回地址,那可就危險了!”

“很聰明嘛!”

“那這個怎么加保護呢?”,我問到。

“你看,函數(shù)進來之前,先在局部變量和返回地址之間設(shè)置一個數(shù)值,函數(shù)返回之前再去檢查一下,如果棧里的數(shù)據(jù)被破壞了,檢查這個數(shù)值就能發(fā)現(xiàn),提前拋出異常!”,小伙伴耐心的解釋到。

“這樣啊,那豈不是要把我打回去加上你說的這些設(shè)置和檢查代碼?”,我繼續(xù)提問。

這時,工作人員聽到了我們的閑聊,“不用,我們編譯器自動添加好了,快去吧,已經(jīng)處理好了”

我瞥了一眼,看到我的資料上增加了一個叫 Stack Canary 的標(biāo)記。

我小心翼翼的走進了那架奇怪的機器,立刻就失去了知覺,等我醒來時,我的身體已經(jīng)發(fā)生了變化,變成了一堆奇怪的代碼,現(xiàn)在我長這樣了:

匯編語言:一個函數(shù)的編譯器之旅

鏈接

沒過一會兒,我們這一隊的所有函數(shù)代碼都編譯完成,大家從原來的.c文件都搬到了新家:一個 .o 文件,我也再次見到了小伙伴 str_lower。

“咱們是不是已經(jīng)完成了編譯,可以離開這里了吧?”

“還不行,編譯雖然是完成了,還差鏈接這一步呢!”

又過了一小會兒,和我們一起過來的其他文件的函數(shù)代碼也編譯完成了,咱們一堆.o文件一起被送到了編譯器大廈的頂樓:鏈接層。

這一層也有一個巨大的機器,機器背后連接了一個管道,不知通向了哪里。

我們這一批的所有 .o 文件挨個走進了這個巨大的機器,像是一條時空隧道一般,穿行于其間,我感覺到了巨大的壓力把我們擠壓在了一起,很快我們再一次失去了意識。

醒來之后,我發(fā)現(xiàn)所有的函數(shù)們都被合在了一個文件中,這是一個可執(zhí)行文件,而我的身體也再次發(fā)生了變化,變成了一段段的二進制指令,現(xiàn)在我長這樣了:

匯編語言:一個函數(shù)的編譯器之旅

終于離開了編譯器,真是一趟難忘的旅程,不過我再也不想來了·····彩蛋:命運開了個玩笑,第一次運行就出錯!
責(zé)編AJX

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 匯編語言
    +關(guān)注

    關(guān)注

    14

    文章

    409

    瀏覽量

    35693
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4284

    瀏覽量

    62325
  • 編譯器
    +關(guān)注

    關(guān)注

    1

    文章

    1617

    瀏覽量

    49016
收藏 人收藏

    評論

    相關(guān)推薦

    使用匯編語言的地方有哪些

    的每個區(qū)域之前,這些匯編指令如何集成到代碼庫中將根據(jù)所使用的開發(fā)環(huán)境而有所不同。通常有些自定義的編譯器函數(shù)用于使編譯器
    發(fā)表于 12-01 06:33

    匯編語言的學(xué)習(xí)資料分享

    寄存 cpu有多個寄存 就是cpu中可以存儲數(shù)據(jù)的器件,cpu中有多個寄存
    發(fā)表于 12-06 07:16

    ARM匯編語言入門

    語言,只比處理的母語機器代碼高出步。 用匯編語言編寫整個程序,即使是相對簡單的程序,也很復(fù)雜。這就是為什么大多數(shù)人使用C或C++等高級語言
    發(fā)表于 08-08 07:28

    匯編語言程序設(shè)計pdf

    匯編語言程序設(shè)計的內(nèi)容:每種高級語言程序在連接為可執(zhí)行程序之前,都必須被編譯匯編語言程序,因此對于高級語言程序設(shè)計者來說,了解
    發(fā)表于 10-21 17:38 ?0次下載
    <b class='flag-5'>匯編語言</b>程序設(shè)計pdf

    匯編語言集成編譯器(軟件) 1.0版

    匯編語言集成編譯器(軟件) 1.0版
    發(fā)表于 03-25 08:57 ?0次下載

    MasmEdit匯編語言編譯器

    MasmEdit匯編語言編譯器.rar
    發(fā)表于 02-24 14:15 ?54次下載

    ARM匯編編譯器的使用

    10.4 ARM匯編編譯器的使用 armasm是ARM匯編語言的交叉編譯器,本節(jié)將詳細(xì)介紹它的使用方法。 armasm命令行語法格式如下。 armasm options inputfi
    發(fā)表于 10-19 10:14 ?5次下載

    編譯器知識拾零

    機器代碼(Machine code)。 現(xiàn)代編譯器的主要工作流程如下: 源程序(source code)預(yù)處理(preprocessor)編
    發(fā)表于 11-07 15:44 ?0次下載

    TMS320C54x匯編語言工具C/C++編譯器的功能優(yōu)化詳細(xì)概述

    該系列是套軟件開發(fā)工具的支持,其中包括優(yōu)化的C/C++編譯器、匯編器、鏈接,以及組合工具
    發(fā)表于 04-27 09:43 ?10次下載
    TMS320C54x<b class='flag-5'>匯編語言</b>工具C/C++<b class='flag-5'>編譯器</b>的功能優(yōu)化詳細(xì)概述

    既然C編譯器是C語言寫,那么第一個C編譯器是怎樣來的?

    既然C編譯器是C語言寫的,那第一個C編譯器是怎樣來的?
    的頭像 發(fā)表于 02-25 15:47 ?3090次閱讀

    編譯原理的角度看C語言如何轉(zhuǎn)換成匯編語言的?

    編譯原理的角度看C語言是如何轉(zhuǎn)換成匯編語言的?
    的頭像 發(fā)表于 02-25 15:52 ?2576次閱讀
    從<b class='flag-5'>編譯</b>原理的角度看C<b class='flag-5'>語言</b>如何轉(zhuǎn)換成<b class='flag-5'>匯編語言</b>的?

    C語言代碼中調(diào)用匯編函數(shù)需要注意什么

    起來看下混合語言工程中,就比如在匯編程序中調(diào)用C函數(shù),或者在C語言代碼中調(diào)用匯編函數(shù)需要注意什么
    的頭像 發(fā)表于 11-12 11:04 ?2783次閱讀

    單片機開發(fā):為何要避免使用匯編語言

    的每個區(qū)域之前,這些匯編指令如何集成到代碼庫中將根據(jù)所使用的開發(fā)環(huán)境而有所不同。通常有些自定義的編譯器函數(shù)用于使編譯器
    發(fā)表于 11-22 20:36 ?10次下載
    單片機開發(fā):為何要避免使用<b class='flag-5'>匯編語言</b>?

    匯編語言學(xué)習(xí)

    寄存 cpu有多個寄存 就是cpu中可以存儲數(shù)據(jù)的器件,cpu中有多個寄
    發(fā)表于 11-23 18:06 ?14次下載
    <b class='flag-5'>匯編語言</b>學(xué)習(xí)

    匯編語言的變量與編碼是什么

    匯編語言采用了助記符號編寫程序,通過編譯器轉(zhuǎn)換成能夠被計算機識別和處理的二進制代碼程序。
    的頭像 發(fā)表于 02-17 14:58 ?1601次閱讀
    <b class='flag-5'>匯編語言</b>的變量與編碼是什么