# 在程序入口處打斷點
## 獲取程序入口
### 方法一:
$ strip a.out
$ readelf -h a.out
ELF Header:
Magic: 7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00
Class: ELF64
Data: 2's complement, little endian
Version: 1 (current)
OS/ABI: UNIX - System V
ABI Version: 0
Type: EXEC (Executable file)
Machine: Advanced Micro Devices X86-64
Version: 0x1
Entry point address: 0x400440
Start of program headers: 64 (bytes into file)
Start of section headers: 4496 (bytes into file)
Flags: 0x0
Size of this header: 64 (bytes)
Size of program headers: 56 (bytes)
Number of program headers: 9
Size of section headers: 64 (bytes)
Number of section headers: 29
Section header string table index: 28
### 方法二:
$ gdb a.out
>>> info files
Symbols from "/home/me/a.out".
Local exec file:
`/home/me/a.out', file type elf64-x86-64.
Entry point: 0x400440
0x0000000000400238 - 0x0000000000400254 is .interp
0x0000000000400254 - 0x0000000000400274 is .note.ABI-tag
0x0000000000400274 - 0x0000000000400298 is .note.gnu.build-id
0x0000000000400298 - 0x00000000004002b4 is .gnu.hash
0x00000000004002b8 - 0x0000000000400318 is .dynsym
0x0000000000400318 - 0x0000000000400355 is .dynstr
0x0000000000400356 - 0x000000000040035e is .gnu.version
0x0000000000400360 - 0x0000000000400380 is .gnu.version_r
0x0000000000400380 - 0x0000000000400398 is .rela.dyn
0x0000000000400398 - 0x00000000004003e0 is .rela.plt
0x00000000004003e0 - 0x00000000004003fa is .init
0x0000000000400400 - 0x0000000000400440 is .plt
0x0000000000400440 - 0x00000000004005c2 is .text
0x00000000004005c4 - 0x00000000004005cd is .fini
0x00000000004005d0 - 0x00000000004005e0 is .rodata
0x00000000004005e0 - 0x0000000000400614 is .eh_frame_hdr
0x0000000000400618 - 0x000000000040070c is .eh_frame
0x0000000000600e10 - 0x0000000000600e18 is .init_array
0x0000000000600e18 - 0x0000000000600e20 is .fini_array
0x0000000000600e20 - 0x0000000000600e28 is .jcr
0x0000000000600e28 - 0x0000000000600ff8 is .dynamic
0x0000000000600ff8 - 0x0000000000601000 is .got
0x0000000000601000 - 0x0000000000601030 is .got.plt
0x0000000000601030 - 0x0000000000601040 is .data
0x0000000000601040 - 0x0000000000601048 is .bss
## 技巧
當調試沒有調試信息的程序時,直接運行`start`命令是沒有效果的:
(gdb) start
Function "main" not defined.
如果不知道main在何處,那么可以在程序入口處打斷點。先通過`readelf`或者進入gdb,執行`info files`獲得入口地址,然后:
(gdb) b *0x400440
(gdb) r
## 貢獻者
* xmj
* [weekface](https://github.com/weekface)
- 介紹
- 信息顯示
- 顯示gdb版本信息
- 顯示gdb版權相關信息
- 啟動時不顯示提示信息
- 退出時不顯示提示信息
- 輸出信息多時不會暫停輸出
- 函數
- 列出函數的名字
- 是否進入帶調試信息的函數
- 進入不帶調試信息的函數
- 退出正在調試的函數
- 直接執行函數
- 打印函數堆棧幀信息
- 打印尾調用堆棧幀信息
- 選擇函數堆棧幀
- 向上或向下切換函數堆棧幀
- 斷點
- 在匿名空間設置斷點
- 在程序地址上打斷點
- 在程序入口處打斷點
- 在文件行號上打斷點
- 保存已經設置的斷點
- 設置臨時斷點
- 設置條件斷點
- 忽略斷點
- 觀察點
- 設置觀察點
- 設置觀察點只針對特定線程生效
- 設置讀觀察點
- 設置讀寫觀察點
- Catchpoint
- 讓catchpoint只觸發一次
- 為fork調用設置catchpoint
- 為vfork調用設置catchpoint
- 為exec調用設置catchpoint
- 為系統調用設置catchpoint
- 通過為ptrace調用設置catchpoint破解anti-debugging的程序
- 打印
- 打印ASCII和寬字符字符串
- 打印STL容器中的內容
- 打印大數組中的內容
- 打印數組中任意連續元素值
- 打印數組的索引下標
- 打印函數局部變量的值
- 打印進程內存信息
- 打印靜態變量的值
- 打印變量的類型和所在文件
- 打印內存的值
- 打印源代碼行
- 每行打印一個結構體成員
- 按照派生類型打印對象
- 指定程序的輸入輸出設備
- 使用“$\”和“$\_”變量
- 打印程序動態分配內存的信息
- 打印調用棧幀中變量的值
- 多進程/線程
- 調試已經運行的進程
- 調試子進程
- 同時調試父進程和子進程
- 查看線程信息
- 在Solaris上使用maintenance命令查看線程信息
- 不顯示線程啟動和退出信息
- 只允許一個線程運行
- 使用“$_thread”變量
- 一個gdb會話中同時調試多個程序
- 打印程序進程空間信息
- 使用“$_exitcode”變量
- core dump文件
- 為調試進程產生core dump文件
- 加載可執行程序和core dump文件
- 匯編
- 設置匯編指令格式
- 在函數的第一條匯編指令打斷點
- 自動反匯編后面要執行的代碼
- 將源程序和匯編指令映射起來
- 顯示將要執行的匯編指令
- 打印寄存器的值
- 顯示程序原始機器碼
- 改變程序的執行
- 改變字符串的值
- 設置變量的值
- 修改PC寄存器的值
- 跳轉到指定位置執行
- 使用斷點命令改變程序的執行
- 修改被調試程序的二進制文件
- 信號
- 查看信號處理信息
- 信號發生時是否暫停程序
- 信號發生時是否打印信號信息
- 信號發生時是否把信號丟給程序處理
- 給程序發送信號
- 使用“$_siginfo”變量
- 共享庫
- 顯示共享鏈接庫信息
- 腳本
- 配置gdb init文件
- 按何種方式解析腳本文件
- 保存歷史命令
- 源文件
- 設置源文件查找路徑
- 替換查找源文件的目錄
- 圖形化界面
- 進入和退出圖形化調試界面
- 顯示匯編代碼窗口
- 顯示寄存器窗口
- 調整窗口大小
- 其它
- 命令行選項的格式
- 支持預處理器宏信息
- 使用命令的縮寫形式
- 在gdb中執行shell命令和make
- 在gdb中執行cd和pwd命令
- 設置命令提示符
- 設置被調試程序的參數
- 設置被調試程序的環境變量
- 得到命令的幫助信息
- 記錄執行gdb的過程