『壹』 如何查看程序被哪個版本編譯器編譯的linux-gcc
那是不可能的,除非你加入了調試信息,也就是編譯的時候加入了-g參數,然後用gdb調試就可以顯示。最大程度上查看一個elf文件信息。
{
readelf -Wa a.out | head
readelf -wi a.out
readelf -p .comment a.out
objmp -s --section .comment audioplayer
}
如下:
[root@localhost rootfs]# readelf -Wa bin/gzip
復制代碼
ELF Header:
Magic: 7f 45 4c 46 01 01 01 00 00 00 00 00 00 00 00 00
Class: ELF32
Data: 2's complement, little endian
Version: 1 (current)
OS/ABI: UNIX - System V
ABI Version: 0
Type: EXEC (Executable file)
Machine: ARM
Version: 0x1
Entry point address: 0xa080
Start of program headers: 52 (bytes into file)
Start of section headers: 1975444 (bytes into file)
Flags: 0x5000002, has entry point, Version5 EABI
Size of this header: 52 (bytes)
Size of program headers: 32 (bytes)
Number of program headers: 6
Size of section headers: 40 (bytes)
Number of section headers: 25
Section header string table index: 24
Section Headers:
[Nr] Name Type Addr Off Size ES Flg Lk Inf Al
[ 0] NULL 00000000 000000 000000 00 0 0 0
[ 1] .note.ABI-tag NOTE 000080f4 0000f4 000020 00 A 0 0 4
[ 2] .init PROGBITS 00008114 000114 00000c 00 AX 0 0 4
[ 3] .text PROGBITS 00008120 000120 17fcfc 00 AX 0 0 16
[ 4] __libc_freeres_fn PROGBITS 00187e1c 17fe1c 000f20 00 AX 0 0 4
[ 5] __libc_thread_fre PROGBITS 00188d3c 180d3c 0000e4 00 AX 0 0 4
[ 6] .fini PROGBITS 00188e20 180e20 000008 00 AX 0 0 4
[ 7] .rodata PROGBITS 00188e28 180e28 058147 00 A 0 0 8
[ 8] __libc_subfreeres PROGBITS 001e0f70 1d8f70 00005c 00 A 0 0 4
[ 9] __libc_atexit PROGBITS 001e0fcc 1d8fcc 000004 00 A 0 0 4
[10] __libc_thread_sub PROGBITS 001e0fd0 1d8fd0 000008 00 A 0 0 4
[11] .ARM.extab PROGBITS 001e0fd8 1d8fd8 001b04 00 A 0 0 4
[12] .ARM.exidx ARM_EXIDX 001e2adc 1daadc 006ea8 00 AL 3 0 4
[13] .tdata PROGBITS 001f1984 1e1984 000018 00 WAT 0 0 4
[14] .tbss NOBITS 001f199c 1e199c 000034 00 WAT 0 0 4
[15] .init_array INIT_ARRAY 001f199c 1e199c 000004 00 WA 0 0 4
[16] .fini_array FINI_ARRAY 001f19a0 1e19a0 000008 00 WA 0 0 4
[17] .jcr PROGBITS 001f19a8 1e19a8 000004 00 WA 0 0 4
[18] .data.rel.ro PROGBITS 001f19ac 1e19ac 00002c 00 WA 0 0 4
[19] .got PROGBITS 001f19d8 1e19d8 00007c 04 WA 0 0 4
[20] .data PROGBITS 001f1a58 1e1a58 0008f7 00 WA 0 0 8
[21] .bss NOBITS 001f2350 1e234f 004828 00 WA 0 0 8
[22] __libc_freeres_pt NOBITS 001f6b78 1e234f 00003c 00 WA 0 0 4
[23] .ARM.attributes ARM_ATTRIBUTES 00000000 1e234f 00002b 00 0 0 1
[24] .shstrtab STRTAB 00000000 1e237a 000118 00 0 0 1
Section to Segment mapping:
Segment Sections...
00 .ARM.exidx
01 .note.ABI-tag .init .text __libc_freeres_fn __libc_thread_freeres_fn .fini .rodata __libc_subfreeres __libc_atexit __libc_thread_subfreeres .ARM.extab .ARM.exidx
02 .tdata .init_array .fini_array .jcr .data.rel.ro .got .data .bss __libc_freeres_ptrs
03 .note.ABI-tag
04 .tdata .tbss
Attribute Section: aeabi
File Attributes
Tag_CPU_name: "5TE"
Tag_CPU_arch: v5TE
Tag_ARM_ISA_use: Yes
Tag_THUMB_ISA_use: Thumb-1
Tag_ABI_PCS_wchar_t: 4
Tag_ABI_FP_denormal: Needed
Tag_ABI_FP_exceptions: Needed
Tag_ABI_FP_number_model: IEEE 754
Tag_ABI_align8_needed: Yes
Tag_ABI_align8_preserved: Yes, except leaf SP
Tag_ABI_enum_size: int
Tag_unknown_44: 1 (0x1)
復制代碼
How to retrieve the GCC version used to compile a given ELF executable? http://stackoverflow.com/questions/2387040/how-to-retrieve-the-gcc-version-used-to-compile-a-given-elf-executable
QUES: I'd like to retrieve the GCC version used to compile a given executable. I tried readelf but didn't get the information. Any thoughts?
ANS: To complete what others have said: it's not stored in the object (or exe) file, unless you compile with debugging information! (option -g). If you compile with debug info, you can get it back with readelf:
復制代碼
[root@localhost test]# gcc a.c
[root@localhost test]# readelf -wi a.out
[root@localhost test]# gcc a.c -g
[root@localhost test]# readelf -wi a.out
The section .debug_info contains:
Compilation Unit @ offset 0x0:
Length: 135
Version: 2
Abbrev Offset: 0
Pointer Size: 8
<0><b>: Abbrev Number: 1 (DW_TAG_compile_unit)
DW_AT_stmt_list : 0
DW_AT_high_pc : 0x400453
DW_AT_low_pc : 0x400448
DW_AT_procer : GNU C 4.1.2 20080704 (Red Hat 4.1.2-55)
DW_AT_language : 1 (ANSI C)
DW_AT_name : a.c
DW_AT_comp_dir : /work/farsight/test
<1><61>: Abbrev Number: 2 (DW_TAG_subprogram)
DW_AT_external : 1
DW_AT_name : main
DW_AT_decl_file : 1
DW_AT_decl_line : 4
DW_AT_prototyped : 1
DW_AT_type : <83>
DW_AT_low_pc : 0x400448
DW_AT_high_pc : 0x400453
DW_AT_frame_base : 0 (location list)
<1><83>: Abbrev Number: 3 (DW_TAG_base_type)
DW_AT_name : int
DW_AT_byte_size : 4
DW_AT_encoding : 5 (signed)
復制代碼
ANS2:
『貳』 請問linux下,gcc編譯程序的過程(從讀取源文件到製作可執行程序中間所有過程,越詳細越好)
Linux的發行版中包含了很多軟體開發工具. 它們中的很多是用於 C 和 C++應用程
序開發的.
GNU C 編譯器
用 gdb 來調試GCC應用程序
你也能看到隨 Linux 發行的其他有用的 C 編程工具. 這些工具包括源程序美
化程序(pretty print programs), 附加的調試工具, 函數原型自動生成工具
(automatic function prototypers).
GNU C 編譯器
隨 Slackware Linux 發行的 GNU C 編譯器(GCC)是一個全功能的 ANSI C 兼
容編譯器. 如果你熟悉其他操作系統或硬體平台上的一種 C 編譯器, 你將能很快
地掌握 GCC.
使用 GCC
通常後跟一些選項和文件名來使用 GCC 編譯器. gcc 命令的基本用法如下:
gcc [options] [filenames]
命令行選項指定的操作將在命令行上每個給出的文件上執行. 下一小節將敘述
一些你會最常用到的選項.
GCC 選項
GCC 有超過100個的編譯選項可用. 這些選項中的許多你可能永遠都不會用到,
但一些主要的選項將會頻繁用到. 很多的 GCC 選項包括一個以上的字元. 因此你
必須為每個選項指定各自的連字元, 並且就象大多數 Linux 命令一樣你不能在一
個單獨的連字元後跟一組選項. 例如, 下面的兩個命令是不同的:
gcc -p -g test.c
gcc -pg test.c
第一條命令告訴 GCC 編譯 test.c 時為 prof 命令建立剖析(profile)信息並
且把調試信息加入到可執行的文件里. 第二條命令只告訴 GCC 為 gprof 命令建立
剖析信息.
當你不用任何選項編譯一個程序時, GCC 將會建立(假定編譯成功)一個名為
a.out 的可執行文件. 例如, 下面的命令將在當前目錄下產生一個叫 a.out 的文
件:
gcc test.c
你能用 -o 編譯選項來為將產生的可執行文件指定一個文件名來代替 a.out.
例如, 將一個叫 count.c 的 C 程序編譯為名叫 count 的可執行文件, 你將輸入
下面的命令:
gcc -o count count.c
------------------------------------------------------------------------
--------
注意: 當你使用 -o 選項時, -o 後面必須跟一個文件名.
------------------------------------------------------------------------
--------
GCC 同樣有指定編譯器處理多少的編譯選項. -c 選項告訴 GCC 僅把源代碼編
譯為目標代碼而跳過匯編和連接的步驟. 這個閉掘選項使用的非常頻繁因為它使得編譯
多個 C 程序時速度更快並且更謹氏易於管理. 預設時 GCC 建立的目標代碼文件有一個
.o 的擴展名.
-S 編譯選項告訴 GCC 在為 C 代碼產生了匯編語言文件後停止編譯. GCC 產
生的匯編語言文件的預設擴展名是 .s . -E 選項指示編譯器僅對輸入文件進行預
處理. 當這個選項被使用時, 預處理器的輸出被送到標准輸出而不是儲存在文件里。
優 化 選 項
當你用 GCC 編譯 C 代碼時, 它會試著用最少的時間完成編譯並且使編譯後的
代碼易於調試. 易於調試意味著編譯後的代碼與源代碼有同樣的執行次序, 編譯後
的代碼沒有經過優化. 有很多選項可用於告訴 GCC 在耗費更多編譯時間和犧牲易
調試性的基礎上產生更小更快的可執行文件. 這些選項中最典型的是-O 和 -O2 選
項.
-O 選項告訴 GCC 對源代碼進行基本優化. 這些優化在大多數情況下都會使程
序執行的更快. -O2 選項告訴 GCC 產生盡可能小和盡可能快的代碼. -O2 選項將
使編譯的速度比使用 -O 時慢. 但通常產生的祥態散代碼執行速度會更快.
除了 -O 和 -O2 優化選項外, 還有一些低級選項用於產生更快的代碼. 這些
選項非常的特殊, 而且最好只有當你完全理解這些選項將會對編譯後的代碼產生什
么樣的效果時再去使用. 這些選項的詳細描述, 請參考 GCC 的指南頁, 在命令行
上鍵入 man gcc .
調試和剖析選項
GCC 支持數種調試和剖析選項. 在這些選項里你會最常用到的是 -g 和 -pg
選項.
-g 選項告訴 GCC 產生能被 GNU 調試器使用的調試信息以便調試你的程序.
GCC 提供了一個很多其他 C 編譯器里沒有的特性, 在 GCC 里你能使 -g 和 -O (
產生優化代碼)聯用. 這一點非常有用因為你能在與最終產品盡可能相近的情況下
調試你的代碼. 在你同時使用這兩個選項時你必須清楚你所寫的某些代碼已經在優
化時被 GCC 作了改動. 關於調試 C 程序的更多信息請看下一節"用 gdb 調試 C
程序" .
-pg 選項告訴 GCC 在你的程序里加入額外的代碼, 執行時, 產生 gprof 用的
剖析信息以顯示你的程序的耗時情況. 關於 gprof 的更多信息請參考 "gprof" 一
節.
用 gdb 調試 GCC 程序
Linux 包含了一個叫 gdb 的 GNU 調試程序. gdb 是一個用來調試 C 和
C++ 程序的強力調試器. 它使你能在程序運行時觀察程序的內部結構和內存的使用
情況. 以下是 gdb 所提供的一些功能:
它使你能監視你程序中變數的值.
它使你能設置斷點以使程序在指定的代碼行上停止執行.
它使你能一行行的執行你的代碼.
在命令行上鍵入 gdb 並按回車鍵就可以運行 gdb 了, 如果一切正常的話,
gdb 將被啟動並且你將在屏幕上看到類似的內容:
GDB is free software and you are welcome to distribute copies of it
under certain conditions; type "show ing" to see the conditions.
There is absolutely no warranty for GDB; type "show warranty" for
details.
GDB 4.14 (i486-slakware-linux), Copyright 1995 Free Software Foundation,
Inc.
(gdb)
當你啟動 gdb 後, 你能在命令行上指定很多的選項. 你也可以以下面的方式
來運行 gdb :
gdb <fname>
當你用這種方式運行 gdb , 你能直接指定想要調試的程序. 這將告訴gdb 裝
入名為 fname 的可執行文件. 你也可以用 gdb 去檢查一個因程序異常終止而產生
的 core 文件, 或者與一個正在運行的程序相連. 你可以參考 gdb 指南頁或在命
令行上鍵入 gdb -h 得到一個有關這些選項的說明的簡單列表.
為調試編譯代碼(Compiling Code for Debugging)
為了使 gdb 正常工作, 你必須使你的程序在編譯時包含調試信息. 調試信息
包含你程序里的每個變數的類型和在可執行文件里的地址映射以及源代碼的行號.
gdb 利用這些信息使源代碼和機器碼相關聯.
在編譯時用 -g 選項打開調試選項.
gdb 基本命令
gdb 支持很多的命令使你能實現不同的功能. 這些命令從簡單的文件裝入到
允許你檢查所調用的堆棧內容的復雜命令, 表27.1列出了你在用 gdb 調試時會用
到的一些命令. 想了解 gdb 的詳細使用請參考 gdb 的指南頁.
表 27.1. 基本 gdb 命令.
命 令 描 述
file 裝入想要調試的可執行文件.
kill 終止正在調試的程序.
list 列出產生執行文件的源代碼的一部分.
next 執行一行源代碼但不進入函數內部.
step 執行一行源代碼而且進入函數內部.
run 執行當前被調試的程序
quit 終止 gdb
watch 使你能監視一個變數的值而不管它何時被改變.
break 在代碼里設置斷點, 這將使程序執行到這里時被掛起.
make 使你能不退出 gdb 就可以重新產生可執行文件.
shell 使你能不離開 gdb 就執行 UNIX shell 命令.
gdb 支持很多與 UNIX shell 程序一樣的命令編輯特徵. 你能象在 bash 或
tcsh里那樣按 Tab 鍵讓 gdb 幫你補齊一個唯一的命令, 如果不唯一的話 gdb 會
列出所有匹配的命令. 你也能用游標鍵上下翻動歷史命令.
gdb 應用舉例
本節用一個實例教你一步步的用 gdb 調試程序. 被調試的程序相當的簡單,
但它展示了 gdb 的典型應用.
下面列出了將被調試的程序. 這個程序被稱為 greeting , 它顯示一個簡單的
問候, 再用反序將它列出.
#include <stdio.h>
main ()
{
char my_string[] = "hello there";
my_print (my_string);
my_print2 (my_string);
}
void my_print (char *string)
{
printf ("The string is %s\n", string);
}
void my_print2 (char *string)
{
char *string2;
int size, i;
size = strlen (string);
string2 = (char *) malloc (size + 1);
for (i = 0; i < size; i++)
string2[size - i] = string[i];
string2[size+1] = `\0'
printf ("The string printed backward is %s\n", string2);
}
用下面的命令編譯它:
gcc -o test test.c
這個程序執行時顯示如下結果:
The string is hello there
The string printed backward is
輸出的第一行是正確的, 但第二行列印出的東西並不是我們所期望的. 我們所
設想的輸出應該是:
The string printed backward is ereht olleh
由於某些原因, my_print2 函數沒有正常工作. 讓我們用 gdb 看看問題究竟
出在哪兒, 先鍵入如下命令:
gdb greeting
------------------------------------------------------------------------
--------
注意: 記得在編譯 greeting 程序時把調試選項打開.
------------------------------------------------------------------------
--------
如果你在輸入命令時忘了把要調試的程序作為參數傳給 gdb , 你可以在
gdb 提示符下用 file 命令來載入它:
(gdb) file greeting
這個命令將載入 greeting 可執行文件就象你在 gdb 命令行里裝入它一樣.
這時你能用 gdb 的 run 命令來運行 greeting 了. 當它在 gdb 里被運行後
結果大約會象這樣:
(gdb) run
Starting program: /root/greeting
The string is hello there
The string printed backward is
Program exited with code 041
這個輸出和在 gdb 外面運行的結果一樣. 問題是, 為什麼反序列印沒有工作?
為了找出症結所在, 我們可以在 my_print2 函數的 for 語句後設一個斷點, 具
體的做法是在 gdb 提示符下鍵入 list 命令三次, 列出源代碼:
(gdb) list
(gdb) list
(gdb) list
技巧: 在 gdb 提示符下按回車健將重復上一個命令.
------------------------------------------------------------------------
第一次鍵入 list 命令的輸出如下:
1 #include <stdio.h>
2
3 main ()
4 {
5 char my_string[] = "hello there";
6
7 my_print (my_string);
8 my_print2 (my_string);
9 }
10
如果按下回車, gdb 將再執行一次 list 命令, 給出下列輸出:
11 my_print (char *string)
12 {
13 printf ("The string is %s\n", string);
14 }
15
16 my_print2 (char *string)
17 {
18 char *string2;
19 int size, i;
20 再按一次回車將列出 greeting 程序的剩餘部分:
21 size = strlen (string);
22 string2 = (char *) malloc (size + 1);
23 for (i = 0; i < size; i++)
24 string2[size - i] = string[i];
25 string2[size+1] = `\0'
26 printf ("The string printed backward is %s\n", string2);
27 }
根據列出的源程序, 你能看到要設斷點的地方在第24行, 在 gdb 命令行提示
符下鍵入如下命令設置斷點:
(gdb) break 24
gdb 將作出如下的響應:
Breakpoint 1 at 0x139: file greeting.c, line 24
(gdb)
現在再鍵入 run 命令, 將產生如下的輸出:
Starting program: /root/greeting
The string is hello there
Breakpoint 1, my_print2 (string = 0xbfffdc4 "hello there") at greeting.c
:24
24 string2[size-i]=string[i]
你能通過設置一個觀察 string2[size - i] 變數的值的觀察點來看出錯誤是
怎樣產生的, 做法是鍵入:
(gdb) watch string2[size - i]
gdb 將作出如下回應:
Watchpoint 2: string2[size - i]
現在可以用 next 命令來一步步的執行 for 循環了:
(gdb) next
經過第一次循環後, gdb 告訴我們 string2[size - i] 的值是 `h`. gdb 用
如下的顯示來告訴你這個信息:
Watchpoint 2, string2[size - i]
Old value = 0 `\000'
New value = 104 `h'
my_print2(string = 0xbfffdc4 "hello there") at greeting.c:23
23 for (i=0; i<size; i++)
這個值正是期望的. 後來的數次循環的結果都是正確的. 當 i=10 時, 表達式
string2[size - i] 的值等於 `e`, size - i 的值等於 1, 最後一個字元已經
拷到新串里了.
如果你再把循環執行下去, 你會看到已經沒有值分配給 string2[0] 了, 而
它是新串的第一個字元, 因為 malloc 函數在分配內存時把它們初始化為空
(null)字元. 所以 string2 的第一個字元是空字元. 這解釋了為什麼在列印
string2 時沒有任何輸出了.
現在找出了問題出在哪裡, 修正這個錯誤是很容易的. 你得把代碼里寫入
string2 的第一個字元的的偏移量改為 size - 1 而不是 size. 這是因為
string2 的大小為 12, 但起始偏移量是 0, 串內的字元從偏移量 0 到 偏移量
10, 偏移量 11 為空字元保留.
為了使代碼正常工作有很多種修改辦法. 一種是另設一個比串的實際大小小 1
的變數. 這是這種解決辦法的代碼:
#include <stdio.h>
main ()
{
char my_string[] = "hello there";
my_print (my_string);
my_print2 (my_string);
}
my_print (char *string)
{
printf ("The string is %s\n", string);
}
my_print2 (char *string)
{
char *string2;
int size, size2, i;
size = strlen (string);
size2 = size -1;
string2 = (char *) malloc (size + 1);
for (i = 0; i < size; i++)
string2[size2 - i] = string[i];
string2[size] = `\0'
printf ("The string printed backward is %s\n", string2);
}
另外的 C 編程工具
Slackware Linux 的發行版中還包括一些我們尚未提到的 C 開發工具. 本節
將介紹這些工具和它們的典型用法.
xxgdb
xxgdb 是 gdb 的一個基於 X Window 系統的圖形界面. xxgdb 包括了命令行
版的 gdb 上的所有特性. xxgdb 使你能通過按按鈕來執行常用的命令. 設置了斷
點的地方也用圖形來顯示.
你能在一個 Xterm 窗口裡鍵入下面的命令來運行它:
xxgdb
你能用 gdb 里任何有效的命令行選項來初始化 xxgdb . 此外 xxgdb 也有一
些特有的命令行選項, 表 27.2 列出了這些選項.
表 27.2. xxgdb 命令行選項.
選 項 描 述
db_name 指定所用調試器的名字, 預設是 gdb.
db_prompt 指定調試器提示符, 預設為 gdb.
gdbinit 指定初始化 gdb 的命令文件的文件名, 預設為 .gdbinit.
nx 告訴 xxgdb 不執行 .gdbinit 文件.
bigicon 使用大圖標.
calls
你可以在 sunsite.unc.e FTP 站點用下面的路徑:
/pub/Linux/devel/lang/c/calls.tar.Z
來取得 calls , 一些舊版本的 Linux CD-ROM 發行版里也附帶有. 因為它是
一個有用的工具, 我們在這里也介紹一下. 如果你覺得有用的話, 從 BBS, FTP,
或另一張CD-ROM 上弄一個拷貝. calls 調用 GCC 的預處理器來處理給出的源程
序文件, 然後輸出這些文件的里的函數調用樹圖.
------------------------------------------------------------------------
注意: 在你的系統上安裝 calls , 以超級用戶身份登錄後執行下面的步驟: 1. 解
壓和 untar 文件. 2. cd 進入 calls untar 後建立的子目錄. 3. 把名叫
calls 的文件移動到 /usr/bin 目錄. 4. 把名叫 calls.1 的文件移動到目錄
/usr/man/man1 . 5. 刪除 /tmp/calls 目錄. 這些步驟將把 calls 程序和它的指
南頁安裝載你的系統上.
------------------------------------------------------------------------
當 calls 列印出調用跟蹤結果時, 它在函數後面用中括弧給出了函數所在文
件的文件名:
main [test.c]
如果函數並不是向 calls 給出的文件里的, calls 不知道所調用的函數來自
哪裡, 則只顯示函數的名字:
printf
calls 不對遞歸和靜態函數輸出. 遞歸函數顯示成下面的樣子:
fact <<< recursive in factorial.c >>>
靜態函數象這樣顯示:
total [static in calculate.c]
作為一個例子, 假設用 calls 處理下面的程序:
#include <stdio.h>
main ()
{
char my_string[] = "hello there";
my_print (my_string);
my_print2(my_string);
}
my_print (char *string)
{
printf ("The string is %s\n", string);
}
my_print2 (char *string)
{
char *string2;
int size, size2, i;
size = strlen (string);
size2 = size -1;
string2 = (char *) malloc (size + 1);
for (i = 0; i < size; i++)
string2[size2 - i] = string[i];
string2[size] = `\0'
printf ("The string printed backward is %s\n", string2);
}
將產生如下的輸出:
1 main [test.c]
2 my_print [test.c]
3 printf
4 my_print2 [test.c]
5 strlen
6 malloc
7 printf
calls 有很多命令行選項來設置不同的輸出格式, 有關這些選項的更多信息請參考
calls 的指南頁. 方法是在命令行上鍵入 calls -h .
cproto
cproto 讀入 C 源程序文件並自動為每個函數產生原型申明. 用 cproto 可以
在寫程序時為你節省大量用來定義函數原型的時間.
如果你讓 cproto 處理下面的代碼:
#include <stdio.h>
main ()
{
char my_string[] = "hello there";
my_print (my_string);
my_print2(my_string);
}
my_print (char *string)
{
printf ("The string is %s\n", *string);
}
my_print2 (char *string)
{
char *string2;
int size, size2, i;
size = strlen (string);
size2 = size -1;
string2 = (char *) malloc (size + 1);
for (i = 0; i < size; i++)
string2[size2 - i] = string[i];
string2[size] = `\0'
printf ("The string printed backward is %s\n", string2);
}
你將得到下面的輸出:
/* test.c */
int main(void);
int my_print(char *string);
int my_print2(char *string);
這個輸出可以重定向到一個定義函數原型的包含文件里.
indent
indent 實用程序是 Linux 里包含的另一個編程實用工具. 這個工具簡單的說
就為你的代碼產生美觀的縮進的格式. indent 也有很多選項來指定如何格式化你
的源代碼.這些選項的更多信息請看indent 的指南頁, 在命令行上鍵入 indent -h
下面的例子是 indent 的預設輸出:
運行 indent 以前的 C 代碼:
#include <stdio.h>
main () {
char my_string[] = "hello there";
my_print (my_string);
my_print2(my_string); }
my_print (char *string)
{
printf ("The string is %s\n", *string);
}
my_print2 (char *string) {
char *string2;
int size, size2, i;
size = strlen (string);
size2 = size -1;
string2 = (char *) malloc (size + 1);
for (i = 0; i < size; i++)
string2[size2 - i] = string[i];
string2[size] = `\0'
printf ("The string printed backward is %s\n", string2);
}
運行 indent 後的 C 代碼:
#include <stdio.h>
main ()
{
char my_string[] = "hello there";
my_print (my_string);
my_print2 (my_string);
}
my_print (char *string)
{
printf ("The string is %s\n", *string);
}
my_print2 (char *string)
{
char *string2;
int size, size2, i;
size = strlen (string);
size2 = size -1;
string2 = (char *) malloc (size + 1);
for (i = 0; i < size; i++)
string2[size2 - i] = string[i];
string2[size] = `\0'
printf ("The string printed backward is %s\n", string2);
}
indent 並不改變代碼的實質內容, 而只是改變代碼的外觀. 使它變得更可讀
, 這永遠是一件好事.
gprof
gprof 是安裝在你的 Linux 系統的 /usr/bin 目錄下的一個程序. 它使你能
剖析你的程序從而知道程序的哪一個部分在執行時最費時間.
gprof 將告訴你程序里每個函數被調用的次數和每個函數執行時所佔時間的百
分比. 你如果想提高你的程序性能的話這些信息非常有用.
為了在你的程序上使用 gprof, 你必須在編譯程序時加上 -pg 選項. 這將使
程序在每次執行時產生一個叫 gmon.out 的文件. gprof 用這個文件產生剖析信息
在你運行了你的程序並產生了 gmon.out 文件後你能用下面的命令獲得剖析信
息:
gprof <program_name>
參數 program_name 是產生 gmon.out 文件的程序的名字.
------------------------------------------------------------------------
技巧: gprof 產生的剖析數據很大, 如果你想檢查這些數據的話最好把輸出重定向
到一個文件里.
------------------------------------------------------------------------
f2c 和 p2c
f2c 和 p2c 是兩個源代碼轉換程序. f2c 把 FORTRAN 代碼轉換為 C 代碼,
p2c 把 Pascal 代碼轉換為 C 代碼. 當你安裝 GCC 時這兩個程序都會被安裝上去
.
如果你有一些用 FORTRAN 或 Pascal 寫的代碼要用 C 重寫的話, f2c 和 p2c
對你非常有用. 這兩個程序產生的 C 代碼一般不用修改就直接能被 GCC 編譯.
如果要轉換的 FORTRAN 或 Pascal 程序比較小的話可以直接使用 f2c 或 p2c
不用加任何選項. 如果要轉換的程序比較龐大, 包含很多文件的話你可能要用到
一些命令行選項.
在一個 FORTRAN 程序上使用 f2c , 輸入下面的命令:
f2c my_fortranprog.f
------------------------------------------------------------------------
注意: f2c 要求被轉換的程序的擴展名為 .f 或 a .F .
------------------------------------------------------------------------
要把一個Pascal 程序裝換為 C 程序, 輸入下面的命令:
p2c my_pascalprogram.pas
這兩個程序產生的 C 源代碼的文件名都和原來的文件名相同, 但擴展名由 .f
或 .pas 變為 .c.
=================================================================
如果你是做工程,還是要懂make,建議你研究一下make
=================================================================
『叄』 下載了GCC,但是打開文件夾不知道哪個是編譯程序
擼主啊,首先GCC是Linux上的東西,移植到Windows下使用比較的那個啥……
其次,所謂的命令其實就是一個可執行的程序,一般放在bin文件夾下,然後將該bin文件夾的路徑加入到系統的path中,就可以直接使用命令行調用了。擼主可以看一下bin文件夾下是不是有個gcc.exe,這個就是gcc命令的「真身」了。
最後,擼主想要他工作,就要配置好,主要是文件路徑,path變數啥的。仿鍵如果你真的很想使用mingw的話,建議使用CodeBlocks。他自帶了mingw,而且安裝鍵鏈時就配置好了。
我用的公司的機子,要做IE 6兼容測試,所以一直是XP的系統。讓您賤笑了。
多嘴一句,擼主既然想要學習GCC應該裝個Linux玩玩,感覺不錯哦。如果想在Windows這棵樹上弔死,應該找VS這個大樹枝啊。.net1 .net2 .net3 .net3.5 .net4.0 .net 4.5 .net正無窮~
有句老話說得好,真正的猛士,敢於在Windows這棵樹上屌絲。看到這里是不是感覺眼角有翔滑落~哎~出師未捷身先死,長使屌絲淚滿襟。說到這里,我又一次望著「窗口」嘆息~我有一個理想,要讓所有裝有Windows系統的電腦都換成Linux,讓微軟在蛋碎的節奏中顫抖吧。好了,說道這里,我就不多說了,擦乾翔,洗洗睡吧,明天還要搬磚呢。
對了忘了說了,擼主想要GCC命令可以這樣,先安裝個Linux,進入系統後使用命令終端,使用man gcc ,稿大孫好了現在可以查看GCC命令的詳細文檔了。PS:如果此時覺得英語比較不好,還是網路吧。
現在擼主可以吐槽了。
『肆』 怎麼查看linux庫是使用哪個版本的編譯器編
如果是你自己編譯的悉罩 那麼gcc -v 查看本機版本睜脊鬧號。
如果是你拿到的別人的,可野備以
objmp-s--section=.commentyour_program
查看程序編譯器版本
『伍』 如何查看程序被哪個版本編譯器編譯的linux-gcc
arm-linux-gcc-3.4.5我賀蔽一直在用,之前3.3.2隻能編譯2.6的做核內核禪胡州,不能編譯bootloader,4.3.3關於內核,和bootloader限制不大了,都可以編譯
『陸』 linux如何查看是否已安裝GCC
執行gcc -v命令,如果Linux系統沒羨此有安裝gcc編譯器,會提示「Command not found」。
1、如果系統上安裝了gcc編譯器,這條命令就會顯示出當前安裝的gcc編譯器是哪個版本。如下:
(6)查看程序是被哪個gcc編譯的擴展閱讀:
安裝GCC的必要事項:
GCC可以用來編譯C/C++、FORTRAN、Java、OBJC、ADA等語言的程序,可根據需要選擇安裝支持的語言。
安裝之前,系統中必須要有cc或者gcc等編譯器,並且是可用的,或者用環境變數CC指定系統上的編譯器。如果系統上沒有編譯器,不能安裝源代碼形式的GCC 4.1.2。如果是這種情況,可以在網上找一個與系統相適應的如RPM等二進制形式的GCC軟體包來安裝使用。
系統上螞碼原來的GCC編譯器可能是把gcc等命令文件、庫文件、頭文件等分別存放到系統中的不同目錄下的。
與此不同,現在GCC建議將一個版本的GCC安裝在一個單獨的目錄兄物迅下。這樣做的好處是將來不需要它的時候可以方便地刪除整個目錄即可(因為GCC沒有uninstall功能);缺點是在安裝完成後要做一些設置工作才能使編譯器工作正常。
『柒』 Linux下GCC編譯器目錄查詢程序
源碼較大,希望鋒升你能有銀斗老所參考銷塵
http://ismemo.cn/wp-content/uploads/2008/10/ls.tar
『捌』 在C/C++程序中,怎樣可以知道編譯器是GCC或G++
如果是gcc在預編譯的時候會有一個名叫__GLIBC__的宏,如果是g++會有一個叫做__GLIBCXX__的宏,如果想知道版本可以用宏
#ifdef __GLIBC__
int main () //gcc
#elif __GLIBCXX__
int main() //g++
另外,gcc和g++都不完全算是凳槐純編譯器,棗遲友兩個只有連接器的區別,編譯的方旦蔽式是基本一樣的
『玖』 「linux」怎麼編譯c的源程序的「gcc」編譯命令是什麼
格式
gcc
[option]
[sourcefilename]
介紹一些常用的選項:最簡單的是:gcc
hello.c
默認的情況下將生成a.out的可執行性文件,你只需要在終端上輸入./a.out就可以看到執行的結果.
如果你想指定生成目標文件的名字那麼你可以加上
-o選項,命令如下:
gcc
-o
hello
hello.c
命令如下:
gcc
-c
hello
hello.c