91欧美超碰AV自拍|国产成年人性爱视频免费看|亚洲 日韩 欧美一厂二区入|人人看人人爽人人操aV|丝袜美腿视频一区二区在线看|人人操人人爽人人爱|婷婷五月天超碰|97色色欧美亚州A√|另类A√无码精品一级av|欧美特级日韩特级

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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

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

嵌入式匯編中go to到c代碼label最簡單的用法

Linux閱碼場 ? 來源:CSDN ? 作者:dog250 ? 2021-04-04 17:18 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

越來越多的工作現(xiàn)如今都交給了編譯器,甚至連動態(tài)代碼修改的數(shù)據(jù)組織這種事都交給了編譯器。gcc提供了一個特性用于嵌入式匯編,那就是asm goto,其實這個特性沒有什么神秘之處,就是在嵌入式匯編中go to到c代碼的label,其最簡單的用法如下(來自gcc的文檔):

d0f90b8c-8cdd-11eb-8b86-12bb97331649.png

asm goto其實就是在outputs,inputs,registers-modified之外提供了嵌入式匯編的第四個“:”,后面可以跟一系列的c語言的label,然后你可以在嵌入式匯編中go to到這些label中一個。然而使用asm goto可以巧妙地將“一個大家都能想到的點子”規(guī)范化,就是說你只需要調(diào)用一個統(tǒng)一的接口--一個宏,編譯器就將你想實現(xiàn)的東西給實現(xiàn)了,要不然代碼寫起來會很麻煩,這點上,編譯器不嫌麻煩。這一個大家都能想出的點子的由來還得從內(nèi)核的效率說起。

以下的代碼來自lwn的《Jump label》:

d11a53be-8cdd-11eb-8b86-12bb97331649.png

即使有了unlikey優(yōu)化,既然有if判斷,cpu的分支預測就有可能失敗,再者do_trace在代碼上離if這么近,即使編譯器再聰明,二進制代碼的do_trace也不會離前面的代碼太遠的,這樣由于局部性原理和cpu的預取機制,do_trace的代碼很有可能就被預取入了cpu的cache,就算我們從來不打算trace代碼也是如此。

我們需要的是如果不開啟trace,那么do_trace永遠不被欲取或者被預測,唯一的辦法就是去掉if判斷,永遠不調(diào)用goto語句,像下面這樣:

d16ccc02-8cdd-11eb-8b86-12bb97331649.png

在運行時修改載入內(nèi)存的二進制代碼就是我們大家都能想到的點子,就是說在運行的時候當我們知道trace_foo_enabled在某一時刻被設置為0的時候,我們動態(tài)的將二進制代碼修改掉,將if代碼段去掉,這樣一個分支預測就不存在了,而且trace_foo_enabled這一個變量也不需要再被訪問了(該變量在內(nèi)存中,訪問它肯定會涉及l(fā)oad/flush cache的動作,為了一個很可能沒有用的變量操作cache很不值)。提前要說的是,我們可以使用這種方式去掉所有的分支預測,然而這并不可取,因為程序是動態(tài)運行的,很多用于判斷的變量值都是根據(jù)程序的執(zhí)行瞬息萬變,正是這種根據(jù)判斷結果采取不同動作的機制給與了程序靈活性,如果每當我們確定一個值時就修改二進制代碼取消分支預測的話,其本身的開銷將會遠遠大于分支預測的開銷,更重要的是,緊接著那個值又變化了,我們不得不再次修改二進制代碼,這期間要訪問那個變量好幾次。所以,只有在我們確定不經(jīng)常變化的變量的判斷上才能用這種方式取消分支預測,而像trace與否的判斷正好符合我們的需求。

gcc編譯器提供了asm goto的機制來滿足我們的需求,使得我們可以在asm goto的基礎上構建出一個叫做jump label的東西。下面的代碼段說明了jump label的用法和原理:

d1e01e96-8cdd-11eb-8b86-12bb97331649.png

標號0僅僅執(zhí)行一個nop,不涉及cache,后面的pushsection保存現(xiàn)有的section,很多情況下當前的section就是text,然后定義一個“表”,表中有兩個元素:0b和trace#NUM,其實就是兩個標號,在asm goto機制中,標號還可以更多,它們在嵌入式匯編的最后一個“:”后面依次排布。這些標號就是供選擇的標號,執(zhí)行流將跳入其中的一個標號處,具體跳到哪一個就看當前的二進制代碼被修改成了“跳到哪一個”,因此asm goto為我們做的僅僅是提供一個地方(一個“:”)供我們將label傳入,保存了一系列的表還是需要我們的c代碼邏輯--jump label實現(xiàn),這些表(其實就是一系列的三元組)方便我們根據(jù)這些表來修改運行中的二進制代碼,最終修改二進制代碼還是要由我們自己寫代碼完成的。

有了這個asm goto以及我們jump label代碼的支持,內(nèi)核對于是否trace這種小事就再也不用愁了(使用中的kernel一般是不用trace的,只有在出了問題以后或者調(diào)試內(nèi)核時才使用trace,因此在主代碼中加入“是否trace”的判斷實在是一種沉重的負擔),如果對于某一個函數(shù)不需要trace,內(nèi)核只需要執(zhí)行一個操作將asm goto附近的代碼改掉即可,比如改稱下面這樣:

d1fcdf54-8cdd-11eb-8b86-12bb97331649.png

如果需要trace,那么就改成:

d227dd30-8cdd-11eb-8b86-12bb97331649.png

這一切在kernel中的用法如下:

d246edba-8cdd-11eb-8b86-12bb97331649.png

第一行的“1”是一個標號,該標號后的代碼執(zhí)行的內(nèi)容就是nop-第二行,第三行重新開始了一個section,這樣的意義很大,下面的三元組:[instruction address] [jump target] [tracepoint key]的二進制代碼就不會緊接著標號1(nop)了,這個三元組就是jump label機制的核心,指示了所有可能跳轉(zhuǎn)到的標號,這里的技巧在于標號1,標號1也作為一個合法的可能跳轉(zhuǎn)到的標號存在,和標號label是并列的,由于pushsection和popsection的存在,上面的代碼匯編結果看起來是下面這樣:

d262c2c4-8cdd-11eb-8b86-12bb97331649.png

如果啟用了trace,那么只需要將標號1修改成標號label就可以了:

d2b347f8-8cdd-11eb-8b86-12bb97331649.png

內(nèi)核之所以能夠找到需要修改代碼的地址,就是借助于上面說的那個三元組(instruction address,jump target,tracepoint key),其中instruction address就是這個地址,在linux的JUMP LABEL機制中,它固定為標號1,也就是nop的標號,如果不啟用trace,那么直接執(zhí)行nop,如果啟用了trace,那么將nop修改為jmp label即可,如果后來又禁用了trace,只需將它再次修改成三元組中的標號1即可,這一切過程中,三元組本身是不會改變的。注意,三元組中的tracepoint key在jump label機制中并沒有什么實質(zhì)的意義,它僅僅是為了組織kernel中“是否trace”變量用的,所有的“是否trace”變量組織成一個鏈表,鏈表的每一個節(jié)點下面掛著另一個子鏈表,該子鏈表中元素是所有使用這個“是否trace”變量的代碼環(huán)境,包括代碼的地址,標號的地址等。

下面看一下kernel對于JUMP_LABEL的實現(xiàn)框架。首先看一下三元組的數(shù)據(jù)結構:

d2f035e6-8cdd-11eb-8b86-12bb97331649.png

其次一個比較重要的數(shù)據(jù)結構是一個key節(jié)點,表示一個“是否trace”的變量:

d33b7420-8cdd-11eb-8b86-12bb97331649.png

啟用一個trace意味著需要將一個key(類似于trace_foo_enabled)設置為1,然后修改所有判斷該key的代碼附近的二進制代碼:

d3842de6-8cdd-11eb-8b86-12bb97331649.png

d3b8bc78-8cdd-11eb-8b86-12bb97331649.png

以上就是使用asm goto實現(xiàn)的jump label,在2.6.37內(nèi)核中被引入。

附:.section以及.previous

在匯編語言中使用.section和.previous指令可以將它們之間的代碼編譯到不同的section中,也就是不緊接著.section上面的代碼。linux kernel中的異常處理就是用這兩個偽指令實現(xiàn)的,定義了一個叫做fix的section和一個叫做ex_table的section,可能出現(xiàn)exception的代碼用一個標號表示,ex_table中保存了一些二元組(出現(xiàn)異常代碼的標號,異常處理程序的標號),異常處理程序在fix這個section中,這樣雖然代碼看起來是下面這樣:

d3f8c854-8cdd-11eb-8b86-12bb97331649.png

然而編譯器會將fix和ex_table放到離text很遠的地方的,這樣cpu預取時就不會將fix或者ex_table的代碼預取到執(zhí)行cache了,只有在發(fā)生異常的時候才會使用fix和ex_table,而發(fā)生異常畢竟是一種罕見現(xiàn)象,這就是一種優(yōu)化。

原文標題:asm goto與JUMP_LABEL

文章出處:【微信公眾號:Linuxer】歡迎添加關注!文章轉(zhuǎn)載請注明出處。

責任編輯:haq

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

    關注

    30

    文章

    4968

    瀏覽量

    73986
  • 編譯器
    +關注

    關注

    1

    文章

    1672

    瀏覽量

    51618

原文標題:asm goto與JUMP_LABEL

文章出處:【微信號:LinuxDev,微信公眾號:Linux閱碼場】歡迎添加關注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    嵌入式單片機開發(fā)學習路徑

    學習C語言 C語言是嵌入式開發(fā)的基礎。掌握C語言的基本語法、指針、結構體、內(nèi)存管理等內(nèi)容是進入嵌入式開發(fā)的前提。 2. 硬件知識 2
    發(fā)表于 02-09 15:42

    什么是嵌入式應用開發(fā)?

    、實現(xiàn)和部署,還包括硬件選擇、軟件設計、測試、集成和維護等流程?。 定義和背景 嵌入式應用開發(fā)是指將軟件部署嵌入式系統(tǒng),這些系統(tǒng)廣泛應用于消費電子和工業(yè)自動化
    發(fā)表于 01-12 16:13

    分析嵌入式軟件代碼的漏洞-代碼注入

    時就是對一切攻擊免疫的。 或者你可以通過設計代碼來禁止可能導致這些問題的接口。 不幸的是,在嵌入式系統(tǒng),這些選擇并不總是可行的。即使C是一種危險的語言,充斥著漏洞,但它仍然是許多組織
    發(fā)表于 12-22 12:53

    C語言單元測試在嵌入式軟件開發(fā)的作用及專業(yè)工具的應用

    語言使用率超過90%。從智能家居溫控系統(tǒng)汽車ECU控制單元,從工業(yè)機器人醫(yī)療設備,C語言仍然是嵌入式開發(fā)的首選語言。 C語言在
    發(fā)表于 12-18 11:46

    ARM嵌入式這樣學

    軟件,嵌入式應用軟件。大部分的嵌入式軟件都寄宿在ARM內(nèi)核的芯片上,三星,飛利浦等等都有ARM內(nèi)核的IC,做計算機軟件的人,很容易的就轉(zhuǎn)做嵌入式軟件,但是要做嵌入式驅(qū)動軟件的話,就有點
    發(fā)表于 12-04 07:48

    CW32嵌入式軟件開發(fā)的必備知識

    嵌入式系統(tǒng)中最常用的編程語言,因為它們提供了直接訪問硬件的能力,并且代碼執(zhí)行效率高。 了解匯編語言,用于編寫底層驅(qū)動、中斷處理程序以及性能要求極高的代碼段。 對其他編程語言如Pytho
    發(fā)表于 11-28 07:48

    C語言在嵌入式開發(fā)的應用

    C 語言在汽車電子控制系統(tǒng)開發(fā)的主導地位。 2、設備驅(qū)動程序 設備驅(qū)動程序是嵌入式系統(tǒng)連接硬件和軟件的橋梁,它負責實現(xiàn)嵌入式系統(tǒng)與
    發(fā)表于 11-21 08:09

    嵌入式C/C++回歸測試四大最佳實踐(附自動化測試工具TESSY使用教程)

    嵌入式開發(fā),一次微小的代碼改動都可能引發(fā)“蝴蝶效應”,如何守護系統(tǒng)的穩(wěn)健?推薦專業(yè)的自動化測試工具#TESSY,源自戴姆勒-奔馳,是嵌入式C
    的頭像 發(fā)表于 10-31 14:21 ?411次閱讀
    <b class='flag-5'>嵌入式</b><b class='flag-5'>C</b>/<b class='flag-5'>C</b>++回歸測試四大最佳實踐(附自動化測試工具TESSY使用教程)

    嵌入式需要掌握哪些核心技能?

    : 1)C語言與底層編程 核心地位:C語言是嵌入式開發(fā)的基石,需精通指針操作、內(nèi)存管理、位運算,直接操控硬件資源。 延伸技能:C++用于復雜項目架構設計,
    發(fā)表于 10-21 16:25

    嵌入式達到什么水平才能就業(yè)?

    工具定位問題具備嵌入式軟件模塊化設計能力:能按功能劃分代碼模塊,編寫規(guī)范的頭文件與源文件,實現(xiàn)模塊間低耦合調(diào)用會使用Git 進行版本管理:能提交代碼、解決沖突、回滾版本,熟悉敏捷開發(fā)流程
    發(fā)表于 09-15 10:20

    嵌入式從入門進階,怎么學?

    嵌入式從入門進階,怎么學? 嵌入式學習的核心是 “軟硬結合的技術壁壘”,科學分層才能高效突破。以下是從入門高階的精簡路線,幫你避開彎路: 1、基礎奠基層:構建技術底座
    發(fā)表于 09-02 09:44

    入行嵌入式應該怎么準備?

    知識: 一、C/C++編程C/C++是嵌入式系統(tǒng)開發(fā)中最常用的編程語言。熟練掌握C/
    發(fā)表于 08-06 10:34

    嵌入式系統(tǒng),F(xiàn)LASH 的程序代碼必須搬到 RAM 運行嗎?

    嵌入式系統(tǒng)里,F(xiàn)LASH 的程序代碼并非必須搬到 RAM 運行,這得由硬件配置、實際性能需求和應用場景共同決定。就像很多低端單片機,無論是依賴片內(nèi) Flash 還是外掛的 SPI
    的頭像 發(fā)表于 08-06 10:19 ?1366次閱讀
    <b class='flag-5'>嵌入式</b>系統(tǒng)<b class='flag-5'>中</b>,F(xiàn)LASH <b class='flag-5'>中</b>的程序<b class='flag-5'>代碼</b>必須搬到 RAM <b class='flag-5'>中</b>運行嗎?

    Linux嵌入式和單片機嵌入式的區(qū)別?

    : 單片機嵌入式 :開發(fā)環(huán)境相對簡單,通常使用C語言或匯編語言進行編程,開發(fā)工具包括Keil、IAR等。 Linux嵌入式 :開發(fā)環(huán)境較
    發(fā)表于 06-20 09:46

    Python在嵌入式系統(tǒng)的應用場景

    你想把你的職業(yè)生涯提升到一個新的水平?Python在嵌入式系統(tǒng)中正在成為一股不可缺少的新力量。盡管傳統(tǒng)上嵌入式開發(fā)更多地依賴于CC++語言,Python的優(yōu)勢在于其簡潔的語法、豐富的
    的頭像 發(fā)表于 03-19 14:10 ?1500次閱讀