close

搜尋到的好文章...
http://i-pogo.blogspot.com/2010/01/usrbinld-cannot-find-lxxx.html
解決 /usr/bin/ld: cannot find -lxxx 問題
問題:
在linux環境編譯應用程式或lib的source code時常常會出現如下的錯誤訊息:

* /usr/bin/ld: cannot find -lxxx


這些訊息會隨著編譯不同類型的source code 而有不同的結果出來如:

* /usr/bin/ld: cannot find -lc
* /usr/bin/ld: cannot find -lltdl
* /usr/bin/ld: cannot find -lXtst


其中xxx即表示函式庫文件名稱,如上例的:libc.so、libltdl.so、libXtst.so。
其命名規則是:lib+庫名(即xxx)+.so。


會發生這樣的原因有以下三種情形:

1 系統沒有安裝相對應的lib
2 相對應的lib版本不對
3 lib(.so檔)的symbolic link 不正確,沒有連結到正確的函式庫文件(.so)


對於上述三種原因有一篇文章寫的很棒可參考這一篇文章的第4點:
gcc命令祥解


解決方法:
(1)先判斷在/usr/lib 下的相對應的函式庫文件(.so) 的 symbolic link 是否正確
若不正確改成正確的連結目標即可解決問題。

(2)若不是 symbolic link 的問題引起,而是系統缺少相對應的lib安裝lib即可解決。

(3)如何安裝缺少的lib:
以上面三個錯誤訊息為例:

錯誤1缺少libc的LIB
錯誤2缺少libltdl的LIB
錯誤3缺少libXtst的LIB



以Ubuntu為例:
先搜尋相對應的LIB再進行安裝的作業如:

apt-cache search libc-dev
apt-cache search libltdl-dev
apt-cache search libXtst-dev


實例:
在進行輸入法gcin的 Source Code的編譯時出現以下的錯誤訊息:

/usr/bin/ld: cannot find -lXtst


經檢查後發現是:
lib(.so檔)的symbolic link 不正確

解決方法如下:

cd /usr/lib
ln -s libXtst.so.6 libXtst.so


如果在/usr/lib的目錄下找不到 libXtst.so 檔,那麼就表示系統沒有安裝libXtst的函式庫。
解法如下:

apt-get install libxtst-dev

--

http://www.51testing.com/html/24/1817.html
(這網站滿多好文的)

GCC命令行詳解
文章出處:blog 作者: 發佈時間:2006-11-02

1. gcc包含的c/c++編譯器
gcc,cc,c++,g++,gcc和cc是一樣的,c++和g++是一樣的,(沒有看太明白前面這半句是什麼意思:))一般c程序就用gcc編譯,c++程序就用g++編譯

2. gcc的基本用法
gcc test.c這樣將編譯出一個名為a.out的程序
gcc test.c -o test這樣將編譯出一個名為test的程序,-o參數用來指定生成程序的名字

3. 為什麼會出現undefined reference to 'xxxxx'錯誤?
首先這是鏈接錯誤,不是編譯錯誤,也就是說如果只有這個錯誤,說明你的程序源碼本身沒有問題,是你用編譯器編譯時參數用得不對,
你沒有指定鏈接程序要用到得庫,比如你的程序裡用到了一些數學函數,那麼你就要在編譯參數里指定程序要鏈接數學庫,方法是在編譯命令行里加入-lm。

4. -l參數和-L參數
-l參數就是用來指定程序要鏈接的庫,-l參數緊接著就是庫名,那麼庫名跟真正的庫文件名 有什麼關係呢?
就拿數學庫來說,他的庫名是m,他的庫文件名 是libm.so,很容易看出,把庫文件名 的頭lib和尾.so去掉就是庫名了。

好了現在我們知道怎麼得到庫名了,比如我們自已要用到一個第三方提供的庫名字叫lib
test.so,那麼我們只要把libtest.so拷貝到/usr/lib裡,編譯時加上-ltest參數,
我們就能用上libtest.so庫了(當然要用libtest.so庫裡的函數,我們還需要與libtest.so配套的頭文件)。

放在/lib和/usr/lib和/usr/local/lib裡的庫直接用-l參數就能鏈接了,但如果庫文件沒放在這三個目錄裡,而是放在其他目錄裡,這時我們只用-l參數的話,鏈接還是會出錯,出錯信息大概是:“/usr/bin/ld:
cannot find
-lxxx”,也就是鏈接程序ld在那3個目錄裡找不到libxxx.so,
這時另外一個參數-L就派上用場了,比如常用的X11的庫,它放在/usr/X11R6/lib目錄下,
我們編譯時就要用-L/usr/X11R6/lib -lX11參數,-L參數跟著的是庫文件所在的目錄名。
再比如我們把libtest.so放在/aaa/bbb/ccc目錄下,那鏈接參數就是-L/aaa/bbb/ccc -ltest

另外,大部分libxxxx.so只是一個鏈接,以RH9為例,比如libm.so它鏈接到/lib/libm.s
ox,/lib/libm.so.6又鏈接到/lib/libm-2.3.2.so,

如果沒有這樣的鏈接,還是會出錯,因為ld只會找libxxxx.so,所以如果你要用到xxxx庫,而只有libxxxx.so.x或者libxxxx-xxxso,做一個鏈接就可以了ln
-s libxxxx-xxxso libxxxx.so

手工來寫鏈接參數總是很麻煩的,還好很多庫開發包提供了生成鏈接參數的程序,名字一般叫xxxx-config,一般放在/usr/bin目錄下,比如

gtk1.2的鏈接參數生成程序是gtk-config,執行gtk-config --libs就能得到以下輸出"-
L/usr/lib -L/usr/X11R6/lib -lgtk -lgdk -rdynamic
-lgmodule -lglib -ldl -lXi -lXext -lX11 -lm",這就是編譯一個gtk1.2程序所需的
gtk鏈接參數,xxx-config除了--libs參數外還有一個參

數是--cflags用來生成頭文件包含目錄的,也就是-I參數,在下面我們將會講到。
你可以試試執行gtk-config --libs --cflags,看看輸出結果。
現在的問題就是怎樣用這些輸出結果了,最笨的方法就是複制粘貼或者照抄,聰明的辦法是在編譯命令行里加入這個`xxxx-config --libs
--cflags`,比如編譯一個gtk程序:gcc gtktest.c `gtk-config --libs
--cflags`這樣就差不多了。
注意`不是單引號,而是1鍵左邊那個鍵。

除了xxx-config以外,現在新的開發包一般都用pkg-config來生成鏈接參數,使用方法跟xxx-config類似,但xxx-config是針對特定的開發包
,但pkg-config包含很多開發包的鏈接參數的生成,用pkg-config
--list-all命令可以列出所支持的所有開發包,pkg-config的用法就是pkg -config pagName --libs
--cflags,其中pagName是包名,是pkg-config--list-all裡列出名單中的一個,比如gtk1.2的名字就是gtk+,pkg-

config gtk+ --libs --cflags的作用跟gtk-config --libs --cflags是一樣的。比如:
gcc gtktest.c `pkg-config gtk+ --libs --cflags`

5。 -include和-I参数 -include和-I參數
-include用來包含頭文件,但一般情況下包含頭文件都在源碼裡用#include xxxxxx實現,-include參數很少用。
-I參數是用來指定頭文件目錄 ,
/usr/include目錄一般是不用指定的,gcc知道去那裡找,但是如果頭文件不在/usr/include裡我們就要用-I參數指定了,比如頭文件放在/myinclude目錄裡,那編譯命令行就要加上-I/myinclude參數了,如果不加你會得到一個"xxxx.h:
No such file or directory"的錯誤。

-I 參數可以用相對路徑,比如頭文件在當前目錄,可以用-I.來指定。上面我們提到的--cf
lags參數就是用來生成-I參數的。

6。-O參數這是一個程序優化參數,一般用-O2就是,用來優化程序用的,比如gcc test.c
-O2,優化得到的程序比沒優化的要小,執行速度可能也有所提高(我沒有測試過)。

7。-shared參數編譯動態庫時要用到,比如gcc -shared test.c -o libtest.so

8。幾個相關的環境變量
PKG_CONFIG_PATH:用來指定pkg-config用到的pc文件的路徑,默認是/usr/lib/pkgconfig,
pc文件是文本文件,擴展名是.pc,裡面定義開發包的安裝路徑,Libs參數和Cflags參數等等。
CC:用來指定c編譯器。
CXX:用來指定cxx編譯器。
LIBS:跟上面的--libs作用差不多。
CFLAGS:跟上面的--cflags作用差不多。
CC,CXX,LIBS,CFLAGS手動編譯時一般用不上,在做configure時有時用到,一般情況下不用管。
環境變量設定方法:export ENV_NAME=xxxxxxxxxxxxxxxxx

9。關於交叉編譯交叉編譯通俗地講就是在一種平台上編譯出能運行在體系結構不同的另一種平台上,比如在我們地PC平台(X86 CPU)上編譯出能運行在sparc
CPU平台上的程序,編譯得到的程序在X86 CPU平台上是不能運行的,必須放到sparc
CPU平台上才能運行。
當然兩個平台用的都是linux。

這種方法在異平台移植和嵌入式開發時用得非常普遍。

相對與交叉編譯,我們平常做的編譯就叫本地編譯,也就是在當前平台編譯,編譯得到的程序也是在本地執行。

用來編譯這種程序的編譯器就叫交叉編譯器,相對來說,用來做本地編譯的就叫本地編譯器,一般用的都是gcc,但這種gcc跟本地的gcc編譯器是不一樣的,需要在編譯gcc時用特定的configure參數才能得到支持交叉編譯的gcc。

為了不跟本地編譯器混淆,交叉編譯器的名字一般都有前綴,比如sparc-xxxx-linux-gnu-gcc,sparc-xxxx-linux-gnu-g++等等

10。交叉編譯器的使用方法使用方法跟本地的gcc差不多,但有一點特殊的是:必須用-L和-I參數指定編譯器用sparc系統的庫和頭文件,不能用本地(X86)的庫(頭文件有時可以用本地的)。
例子:
sparc-xxxx-linux-gnu-gcc test.c -L/path/to/sparcLib -I/path/to/sparcInclude

arrow
arrow
    全站熱搜
    創作者介紹
    創作者 fvalinux 的頭像
    fvalinux

    Elegance

    fvalinux 發表在 痞客邦 留言(0) 人氣()