• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            elva

            gdb使用及原理【轉(zhuǎn)】

            1、GDB對(duì)于基于GNU系統(tǒng)開發(fā)的程序員來說是最基本的東西,必須的。所以這篇學(xué)習(xí)總結(jié)中,不打算包括GDB的一般使用方法。因?yàn)檫@些東西必須是隨手拈來的。所以也就不花時(shí)間來整理,我只把一些比較高級(jí)的應(yīng)用在這里作一個(gè)整理。
             
            2、在編譯鏈接程序時(shí)需要使用"-ggdb"選項(xiàng)來生成可供GDB調(diào)試用的信息,否則GDB將失去作用,因此GDB和GCC聯(lián)系的非常緊密。并且當(dāng)-g和-O開關(guān)同時(shí)打開時(shí),調(diào)試和優(yōu)化可能會(huì)產(chǎn)生沖突,經(jīng)常會(huì)發(fā)現(xiàn)所見和事實(shí)不合的情況,所以要選擇性地開啟優(yōu)化開關(guān)。
             
            3、GDB的一些使用技巧:
             1)設(shè)置斷點(diǎn)的方法包括:函數(shù),行號(hào),if條件斷點(diǎn)express,這些前面都可以跟上文件名。另外還可以設(shè)置地址斷點(diǎn):b *0x8048424.
             2)GDB用來分析core文件,啟動(dòng)格式:gdb debugme core.xyz
             3)開啟core文件生成的方法是: ulimit -c unlimited
             4)在不同函數(shù)的調(diào)用棧上切換及查看當(dāng)前信息:bt/frame XX/up/down/info frame/args/locals
             5)調(diào)試一正運(yùn)行的進(jìn)程:gdb debugme pid或者gdb debugme + attach pid + detach,類似的應(yīng)用還有:strace/ltrace/truss
             6)如果某個(gè)線程/進(jìn)程處于死鎖狀態(tài),還可以通過gcore pid來手動(dòng)生成core文件來分析當(dāng)前線程/進(jìn)程的狀態(tài),然后利用GDB來分析, gcore使用方法:gcore pid,注意被調(diào)試的進(jìn)程會(huì)臨時(shí)性停止去生成core文件
             7)查看函數(shù)的反匯編指令:disassemble fun_name
             8)匯編指令級(jí)別的單步執(zhí)行:ni/si,顯示當(dāng)前執(zhí)行的匯編指令: x/i $pc
             9)查看寄存器的內(nèi)容:info registers/all-registers
             10)查看某地址開始的內(nèi)容:x/num 0xYYYYYYY 查看從0xYYYYYYY開始的num個(gè)單元內(nèi)容;p 輸出數(shù)組內(nèi)容
             11)在函數(shù)調(diào)試中途強(qiáng)制返回:return  <expression>;
             12)向被調(diào)試程序發(fā)送指定信號(hào):在任意一點(diǎn)ctrl+C進(jìn)入gdb調(diào)試命令行,然后:signal 1-15
             
            4、用GDB來調(diào)試多線程程序:
             1)顯示當(dāng)前可調(diào)試的所有線程:info threads,GDB按照線程啟動(dòng)順序重新安排了一個(gè)線程ID,這個(gè)ID是供GDB使用的
             2)在調(diào)試多線程的程序時(shí),默認(rèn)調(diào)試的是主線程,其他線程也同時(shí)處于暫停狀態(tài),如果想切換調(diào)試其他的線程,則只需要:thread id
             3)在對(duì)某一線程進(jìn)行next/step執(zhí)行的時(shí)候,其他線程也同時(shí)在執(zhí)行,如果要限制其他線程執(zhí)行,則可以使用:set scheduler-locking on
             4)對(duì)指定線程或者所有線程執(zhí)行同樣的操作,比如查看調(diào)用棧信息:thread apply ID1 ID2/all bt
             5)另外你也可以利用strace -p pid來顯示某個(gè)線程當(dāng)前的系統(tǒng)調(diào)用情況。或者利用gdb debugme pid來調(diào)試某個(gè)線程,但注意該方法會(huì)暫停整個(gè)進(jìn)程的執(zhí)行。對(duì)于多線程的程序gdb ./debugme相當(dāng)于默認(rèn)調(diào)試主線程,而gdb ./debugme pid則相當(dāng)于默認(rèn)調(diào)試pid線程。
             
            5、用GDB來調(diào)試多進(jìn)程程序:
             1)當(dāng)fork子進(jìn)程后,繼續(xù)調(diào)試父進(jìn)程或者調(diào)試剛產(chǎn)生的子進(jìn)程:set follow-fork-mode parent/child,注意調(diào)試的時(shí)候其他的進(jìn)程仍然在運(yùn)行。
             2)如果父進(jìn)程fork了多個(gè)子進(jìn)程,上面的這種方法也只能跟蹤調(diào)試到第一個(gè)子進(jìn)程,并且不影響其他子進(jìn)程的運(yùn)行。
             3)如果想在調(diào)試一個(gè)進(jìn)程的時(shí)候,其他進(jìn)程處于暫停狀態(tài),則可以利用:set detach-on-fork off來做到
             4)利用attach來調(diào)試子進(jìn)程。因?yàn)楦高M(jìn)程fork子進(jìn)程后,子進(jìn)程會(huì)馬上得到執(zhí)行,如果恰好執(zhí)行過了你要調(diào)試的地方,則來不及查詢pid并且attach,所以為了支持直接attach調(diào)試,一般會(huì)在子進(jìn)程的代碼開始處加上一個(gè)sleep,以使得你有時(shí)間來查詢pid,然后attach進(jìn)入來調(diào)試。
               attach pid + stop + break XXX + continue + n + n ...+ s + s + ....
             5)利用gdb ./debugme pid都可以用來調(diào)試進(jìn)程和線程,但不同的是GDB控制的范圍不一樣,前者不影響其他的并行單元(進(jìn)程),而后則會(huì)使真?zhèn)€進(jìn)程暫停。 
             
            6、調(diào)試動(dòng)態(tài)鏈接庫函數(shù):
             我們可能要調(diào)試動(dòng)態(tài)庫的函數(shù),或者通過調(diào)試來學(xué)習(xí)動(dòng)態(tài)庫函數(shù)的實(shí)現(xiàn)。這個(gè)時(shí)候,則需要GDB包括該動(dòng)態(tài)庫的debug版本,否則在GDB下面只會(huì)打印:0xXXXXXX: ??
             比如包括:glibc debug version,如下是一些glibc的debug版本的下載地址:
             http://linux.maruhn.com/sec/glibc-debug.html
             
            注:GDB的遠(yuǎn)端調(diào)試功能,暫時(shí)還沒有接觸過,現(xiàn)不做學(xué)習(xí)和總結(jié).
             GDB對(duì)于多線程,多進(jìn)程的調(diào)試支持并不強(qiáng)大,但可以利用其他專用調(diào)試器,比如TotalView:
             參考地址:http://www.totalviewtech.com/
                     http://www.total-view.com.cn/
                
               
            7、一些輔助的診斷及調(diào)試工具:
             1)strace:跟蹤系統(tǒng)調(diào)用情況
             2)ltrace:跟蹤動(dòng)態(tài)庫的調(diào)用情況
             3)mtrace,pmalloc:跟蹤內(nèi)存使用情況,需要嵌入代碼,打印內(nèi)存使用記錄。
             4)Binuitls:Toolchain的工具,參考我的上一篇總結(jié)。
             5)Valgrind:非常好的內(nèi)存泄露檢測工具,限于i386
             6)oprofile, NPTL Trace Tool等
             7)ald:匯編語言調(diào)試器
             8)Dude:另一個(gè)運(yùn)行l(wèi)inux上的調(diào)試器,未使用ptrace實(shí)現(xiàn)
             9)Linice(http://www.linice.com/)是SoftIce在Linux中的模擬軟件,用于調(diào)試沒有源代碼的二進(jìn)制文件的內(nèi)核級(jí)調(diào)試器。
             10)其他
            關(guān)于調(diào)試及診斷工具包括許多,估計(jì)可以寫一系列的文章來說明。

            其他參考資料:
            0)GDB官方網(wǎng)站:http://www.gnu.org/software/gdb/gdb.html
            1)快速參考GDB支持的所有調(diào)試命令:《GDB QUICK REFERENCE》
            2)GDB的使用手冊:《Debugging with gdb--The gnu Source-Level Debugger》
            3)《Embedded linux prime》的第13/14/15章可以作為參考。
            http://book.opensourceproject.org.cn/embedded/embeddedprime/index.html?page=opensource/0136130550/ch13lev1sec1.html
            文章出處:飛諾網(wǎng)(www.firnow.com):http://dev.firnow.com/course/6_system/linux/Linuxjs/20091209/184488.html

             

            ***********************************************************************************************

            1、GDB基本組成:
               GDB由三個(gè)部分組成:
             (1)用戶接口user interface,除支持傳統(tǒng)的CLI接口還支持mi接口(ddd等工具使用)
             (2)符號(hào)處理層symbol handling,當(dāng)gdb ./debugme后GDB會(huì)讀取文件的符號(hào)信息,之后的原代碼,變量/函數(shù)/類型的顯示都由該部分進(jìn)行(everything you can do without live process)。
             (3)目標(biāo)系統(tǒng)處理層target system handling。包括執(zhí)行控制,斷點(diǎn)設(shè)置,單步執(zhí)行,堆棧分析等操作都有該部分來進(jìn)行。
             
            2、GDB各部分的實(shí)現(xiàn):
             (1)用戶接口層(CLI)的實(shí)現(xiàn)很顯然要用到readline/history庫,而圖形界面mi則需要用到:GNU ncurses庫。
               參考資料: http://www.gnu.org/software/ncurses/
                      http://tiswww.case.edu/php/chet/readline/rltop.html
                     
             (2)符號(hào)處理層則需要使用到:BFD/Opcodes庫,分別用來讀取分析ELF/Core文件,反匯編.
              參考資料: http://www.xfocus.net/articles/200109/265.html
                 http://sourceware.org/binutils/docs/bfd/index.html#Top
                 http://www.linuxselfhelp.com/gnu/bfd/html_chapter/bfd_toc.html
                
             (3)目標(biāo)系統(tǒng)控制層:用ptrace系統(tǒng)調(diào)用來實(shí)現(xiàn)對(duì)其他進(jìn)程的執(zhí)行控制,檢查和改變其核心映像以及寄存器等操作。
             
              
            3、后端(目標(biāo)系統(tǒng)控制層)實(shí)現(xiàn):
             (1)內(nèi)核在執(zhí)行用戶請求的系統(tǒng)調(diào)用之前回檢查當(dāng)前進(jìn)程是否處于被“跟蹤”狀態(tài),如果是的話內(nèi)核暫停當(dāng)前進(jìn)程并將控制權(quán)交給調(diào)試進(jìn)程,使跟蹤調(diào)試進(jìn)程可以查看甚至修改被調(diào)試進(jìn)程的內(nèi)存,寄存器等數(shù)據(jù)。而ptrace函數(shù)的作用就是告訴內(nèi)核在執(zhí)行子進(jìn)程的系統(tǒng)調(diào)用之前做的動(dòng)作。所有的動(dòng)作都可以通過request進(jìn)行傳入。
             (2)設(shè)置斷點(diǎn)原理:通過查找輸入的斷點(diǎn)和具體代碼位置對(duì)應(yīng)起來,并在該位置替換為一條斷點(diǎn)指令,并且保存以前的指令,到目標(biāo)程序運(yùn)行到該斷點(diǎn)處時(shí),產(chǎn)生SIGTRAP信號(hào),該信號(hào)被GDB捕獲,GDB查找斷點(diǎn)列表來確定是否命中斷點(diǎn)。繼續(xù)執(zhí)行的時(shí)候則會(huì)把保存的指令重新放回并執(zhí)行。n/s/ni/si/finish/uitil也會(huì)自動(dòng)設(shè)置斷點(diǎn)。
             (3)內(nèi)核傳遞給被調(diào)試進(jìn)程所有的信號(hào),都會(huì)先傳遞給GDB再由gdb采取定義的動(dòng)作來和被調(diào)試進(jìn)程之間進(jìn)行相互協(xié)調(diào)操作。gdb暫停目標(biāo)程序運(yùn)行的方法是向其發(fā)送SIGSTOP信號(hào),GDB對(duì)于隨機(jī)信號(hào)(非GDB產(chǎn)生的)的處理包括,可以通過handle signals命令來預(yù)定義
             
            4、 ptrace函數(shù)簡單介紹:long ptrace(enum __ptrace_request request, pid_t pid, void *addr, void *data);其中第一個(gè)參數(shù)代表告訴給kernel要做的動(dòng)作。
             PTRACE_ME:設(shè)置自己的被跟蹤標(biāo)志,在被調(diào)試進(jìn)程中使用。
             PTRACE_PEEKUSER:可以得到系統(tǒng)調(diào)用號(hào)及參數(shù)信息
             PTRACE_CONT:使被跟蹤進(jìn)程繼續(xù)執(zhí)行
             PRACE_GETREGS:一次性得到所有寄存器相關(guān)的值,提供輸出參數(shù)
             PTRACE_POKEDATA:可用來改變子進(jìn)程中變量的值
             PTRACE_SINGLESTEP:會(huì)使內(nèi)核在子進(jìn)程的每一條指令執(zhí)行前先將其阻塞,然后將控制權(quán)交給父進(jìn)程
             PTRACE_ATTACH:向運(yùn)行著的子進(jìn)程置上跟蹤標(biāo)志為。
             PTRACE_DETACH:和上面的行為相反。
            很多工具strace/ltrace/stuss等工具都用到了ptrace,學(xué)習(xí)ptrace的最好的資料是這些工具的原代碼和kernel相關(guān)代碼。
             
            其他參考資料:
            1)Ptrace相關(guān)資料: http://linuxgazette.net/issue81/sandeep.html
                   http://blog.chinaunix.net/u/19651/showart_362901.html //玩轉(zhuǎn)ptrace系列
                   http://blog.chinaunix.net/u/19651/showart_362921.html
                  
            2)GDB的實(shí)現(xiàn)說明: 《gdb Internals》 

            posted on 2010-08-05 18:03 葉子 閱讀(3118) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C\C++

            无码人妻精品一区二区三区久久久| 国产精品综合久久第一页| 国内精品伊人久久久久妇| 久久久国产精华液| 国内精品久久久久影院日本| 国产精品99久久久久久董美香 | 99久久免费国产特黄| 99久久国产综合精品五月天喷水 | 日产精品久久久久久久| 日本精品久久久久中文字幕8 | 无码人妻久久一区二区三区免费丨 | 一级做a爱片久久毛片| 久久久精品久久久久久| 久久综合亚洲欧美成人| 无码8090精品久久一区| 久久精品国产只有精品2020| 久久久久亚洲国产| 久久国产综合精品五月天| 久久精品a亚洲国产v高清不卡| 精品国产91久久久久久久a| 精品永久久福利一区二区| 色综合久久中文字幕综合网| 伊人热人久久中文字幕| 日韩精品久久无码中文字幕| 久久婷婷五月综合国产尤物app| 国内精品久久久久久麻豆| 久久久久免费精品国产| 精品无码久久久久久尤物| 国产成人无码精品久久久性色| 亚洲人AV永久一区二区三区久久| 亚洲国产天堂久久综合网站| 91久久福利国产成人精品| 久久天堂电影网| 久久免费精品一区二区| 久久综合九色综合欧美狠狠| 狠色狠色狠狠色综合久久| 97热久久免费频精品99| 久久亚洲国产午夜精品理论片| 72种姿势欧美久久久久大黄蕉| 99久久免费国产精精品| 99久久精品国产一区二区|