Mach-O 的動態鏈接(Lazy Bind 機制)

来源:http://www.cnblogs.com/TingyunAPM/archive/2016/12/12/6163393.html
-Advertisement-
Play Games

➠更多技術乾貨請戳:聽雲博客 動態鏈接 要解決空間浪費和更新困難這兩個問題最簡單的方法就是把程式的模塊相互分割開來,形成獨立的文件,而不再將它們靜態的鏈接在一起。簡單地講,就是不對那些組成程式的目標文件進行鏈接,等到程式要運行時才進行鏈接。也就是說,把鏈接過程推遲到了運行時再進行,這就是 _動態鏈接 ...


更多技術乾貨請戳:聽雲博客

動態鏈接

要解決空間浪費和更新困難這兩個問題最簡單的方法就是把程式的模塊相互分割開來,形成獨立的文件,而不再將它們靜態的鏈接在一起。簡單地講,就是不對那些組成程式的目標文件進行鏈接,等到程式要運行時才進行鏈接。也就是說,把鏈接過程推遲到了運行時再進行,這就是 _動態鏈接(Dynamic Linking)_的思想。

延遲綁定(PLT)

動態鏈接比靜態鏈接性能低,主要原因是動態鏈接下對於全局和靜態的數據訪問都要進行複雜的GOT定位,間接定址;對於模塊間的調用也要先定位GOT,間接跳轉,  程式的運行速度一定會減慢。 另外因為動態鏈接是在運行時完成鏈接的工作:在程式開始執行時,動態鏈接器都要進行一次鏈接工作,動態鏈接器會 search 並 load 所需要的 共用對象,然後進行符號查詢 地址重定位,這一系列動作會減慢程式的啟動速度。 

PLT 就是為了優化動態鏈接性能而存在,基本思想就是 當函數第一次被調用到時才進行綁定(符號查找,重定位),如果沒有用到則不進行 bind。 這樣在程式執行時,模塊間的函數調用都沒有進行 bind , 而是需要調用時才由 動態鏈接器來負責 bind 。這樣可以加速程式的啟動速度。

Mach-O Lazy Bind

Mach-O 文件通過dyld 載入的時候並沒有確定每一個函數的具體地址在哪裡,而是在真正調用該函數的時候通過 過程鏈接表(procedure linkage table),簡稱 PLT,來進行一次lazybind。

結合Mach-O 文件的分析與代碼的調試簡單的分析一下。

 源代碼如下:

1.png

分別在兩個printf函數處下 斷點。

第一個調用printf函數

2.png

在0x100000f52 \<+34\>行處通過callq 0x100000f76 來調用printf。

執行callq指令 之後代碼跳轉到這裡:

2-2.png

這裡的jmpq 要跳轉到 0x0000000100000F8C ,這個地址是從 —Data , —la-symbol-ptr  中的Lazy Symbol Pointers 中獲取到的。

3.png

通過lldb 的命令 查看 0x100001010處的地址 獲取了同樣的值。

通過—stub—helper進行lazybind

在Mach-O 中每一個Symbol Stub 可能有以下兩種行為其中之一:跳轉到函數的指令,執行函數體,通過動態動態庫鏈接器查找函數的Symbol,然後執行函數體

查看 —stubs的Section 數據,發現只有一個函數就是 printf

4.png

這裡的Data 其實就是上面看到的 jmpq 的代碼。執行之後代碼跳轉到了這樣的代碼片段。

5.png

這裡就是通過 —stub-helper來調用 dyld-stubbinder函數來計算printf 函數的真是地址。 通過下麵的 信息可以看出來,jmpq 0x100000f7c ,就是在壓如入參數0x0 (函數的link 的時候給的編號) 之後跳轉到Section的起始處,調用 binder(一段彙編代碼, 作用就是計算具體的函數地址,並調用printf 函數)

6.png

第二次調用printf 函數

7.png

執行指令之後發現和第一次調用printf 已經不一樣了。

8.png

再一次查看 0x100001010 處記憶體值。已經很第一次不同了,也就是說, —Data, —la-symbol-ptr 中指向printf地址的值已經發生了變化,指向了 printf的指令。

這就證明瞭,延遲綁定只會在第一次調用的時候發生。整個流程與 linux中的PLT 和GOT 在實現邏輯上基本相同,只是實現代碼不同而已。

參考《Mac OS X  and iOS Internals》、《鏈接,裝載與庫》

 

原文鏈接:http://blog.tingyun.com/web/article/detail/1347


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在我們製作網頁時,例如導航等,會遇到很多的一些小圖標,這裡以京東的小三角(如下圖) 舉例,來介紹常用的一些方法。 方法一:也是比較簡單的一種方法。用background 來做,一般用一個行內標簽如 i ,s 等 先轉換 顯示方式 display:block;,然後設置寬高。 如果不是單一的圖片,而是 ...
  • Base64是網路上最常見的用於傳輸8Bit位元組代碼的編碼方式之一,在瞭解Base64編碼之前,先瞭解幾個基本概念:位、位元組。 位:"位(bit)"是電腦中最小的數據單位。每一位的狀態只能是0或1; 位元組:8個二進位位構成1個"位元組(Byte)",位元組是存儲空間的基本計量單位。1個位元組可以儲存1個 ...
  • <!DOCTYPE html><html> <head> <meta charset="utf-8"> <title></title> <style type="text/css"> *{ margin: 0; padding: 0; } .box{ width: 180px; height: 30 ...
  • Array.prototype.reduce() 概述 reduce()方法是數組的一個實例方法(共有方法),可以被數組的實例對象調用。reduce() 方法接收一個函數作為累加器(accumulator),數組中的每個值(從左到右)開始縮減,最終為一個值。 語法 arr.reduce(callba ...
  • 今天我們一起來探討下安卓中BroadcastReceiver組件以及詳細分析下它的兩種註冊方式。 BroadcastReceiver也就是“廣播接收者”的意思,顧名思義,它就是用來接收來自系統和應用中的廣播。在Android系統中,廣播體現在方方面面,例如當開機完成後系統會產生一條廣播,接收到這條廣 ...
  • 今日,在學習AnimatorSet時,使用play、with、after、before時,代碼書寫如下: 運行代碼時,報錯,錯誤報告如下: 然後,我將set.play(animator1).after(animator2).before(animator3).with(animator2);代碼修改 ...
  • Toast的多種使用方法 Toast其實是一個功能特別強大的組件,不僅僅可以吐司一個文本內容,還可以吐司圖片以及圖文混排的界面。具體用法如下: 第一種:簡單的純文本內容的吐司: 第二種:吐司一張圖片: 第三種:圖文混合吐司 Toast還有一些其它的屬性,知道這最基本的,然後發揮我們的想象,t就可以利 ...
  •   對於直播來說,客戶端主要做兩件事情,推流和播放。今天先講播放。 播放流程 1.拉流:伺服器已有直播內容,從指定地址進行拉取的過程。其實就是向伺服器請求數據。 2.解碼:對視屏數據進行解壓縮。 3.渲染:生成圖片的過程。視屏其實就是一張張連續的圖片進行切換,一張圖片就是一幀。幀 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...