? ? ? ?作為一個(gè)老的windows程序員颅筋,由于工作的原因宙暇,需要在linux 下寫一個(gè)音頻處理的服務(wù)。 接收到這個(gè)任務(wù)后议泵,第一思路還是windows 思路占贫,寫一個(gè)動(dòng)態(tài)庫,調(diào)用ffmpeg 庫先口,然后在動(dòng)態(tài)庫中寫入自己的音頻處理流程型奥。 然后分享這個(gè)動(dòng)態(tài)庫瞳收,供其他程序員使用。按照這個(gè)思路厢汹,linux 也當(dāng)然可以螟深。
? ? ? ?說干就干,而且是擼起袖子加油干烫葬,我在華為云申請(qǐng)了虛擬機(jī)界弧,安裝了ubantu.然后開始寫我的音頻處理庫了。沒想到厘灼,作為windows 老手的linux新手來說夹纫,整個(gè)過程還是很是艱辛,也遇到了很多問題设凹。下面請(qǐng)聽我娓娓道來,如果恰好您也在編譯ffmpeg茅姜, 或許能給您一些幫助闪朱。
? ? ? 如果您不聽我啰嗦,也可以到下面鏈接 查看安裝方法钻洒,如果遇到這個(gè)問題再返回看奋姿。
? ? ??linux 下的ffmpeg的編譯和編寫引用ffmpeg的動(dòng)態(tài)庫 - 簡書
? ? ? 本文參考了很多網(wǎng)頁,最重要的還是這個(gè)鏈接素标,建議您也看一下称诗。
? ? ?https://trac.ffmpeg.org/wiki/CompilationGuide/Ubuntu#RevertingChangesMadebyThisGuide
一. 前期準(zhǔn)備
1.1? linux 普通賬戶
? ? 建議不在root 下完成這個(gè)項(xiàng)目。需要建立自己的賬戶 头遭,例如zhd 寓免,然后在這個(gè)普通賬戶下編譯ffmpeg。 同時(shí)在root 下计维,將zhd 設(shè)置為sudo用戶袜香。 如果是新手,具體操作可以baidu
1.2. 創(chuàng)建ffmpeg 目錄
? ? cd ~
? ? mkdir -p ~/ffmpeg_sources ~/ffmpeg_build ~/bin
1. 3 獲取 依賴庫
sudo apt-get update -qq && sudo apt-get -y install \
? autoconf \
? automake \
? build-essential \
? cmake \
? git-core \
? libtool \
? pkg-config \
? texinfo \
? wget \
? zlib1g-dev
上面這些庫鲫惶,對(duì)于windows 程序員也是熟悉的蜈首,如果不熟悉,可以逐條查一下欠母, 這些包主要是用于下載文件欢策,編譯和鏈接。
二. 編譯和安裝
根據(jù)需要赏淌,項(xiàng)目需要盡可能多的解碼器踩寇,編碼器只需要mp3 就可以了,根據(jù)ffmpeg 網(wǎng)站的安裝提示猜敢,需要安裝ffmpeg的插件如下:
1. 安裝nasm
cd ~/ffmpeg_sources && \
wget https://www.nasm.us/pub/nasm/releasebuilds/2.14.02/nasm-2.14.02.tar.bz2 && \
tar xjvf nasm-2.14.02.tar.bz2 && \
cd nasm-2.14.02 && \
./autogen.sh && \
PATH="$HOME/bin:$PATH" ./configure --prefix="$HOME/ffmpeg_build" --bindir="$HOME/bin" && \
make && \
make install
2. 安裝yasm
cd ~/ffmpeg_sources && \
wget -O yasm-1.3.0.tar.gz https://www.tortall.net/projects/yasm/releases/yasm-1.3.0.tar.gz && \
tar xzvf yasm-1.3.0.tar.gz && \
cd yasm-1.3.0 && \
./configure --prefix="$HOME/ffmpeg_build" --bindir="$HOME/bin" && \
make && \
make install
3. 安裝 libx264
cd ~/ffmpeg_sources && \
git -C x264 pull 2> /dev/null || git clone --depth 1 https://code.videolan.org/videolan/x264.git && \
cd x264 && \
PATH="$HOME/bin:$PATH" PKG_CONFIG_PATH="$HOME/ffmpeg_build/lib/pkgconfig" ./configure --prefix="$HOME/ffmpeg_build" --bindir="$HOME/bin" --enable-static --enable-pic && \
PATH="$HOME/bin:$PATH" make && \
make install
4. 安裝libx265
sudo apt-get install mercurial libnuma-dev && \
cd ~/ffmpeg_sources && \
if cd x265 2> /dev/null; then hg pull && hg update && cd ..; else hg clone https://bitbucket.org/multicoreware/x265; fi && \
cd x265/build/linux && \
PATH="$HOME/bin:$PATH" cmake -G "Unix Makefiles" -DCMAKE_INSTALL_PREFIX="$HOME/ffmpeg_build" -DENABLE_SHARED=off ../../source && \
PATH="$HOME/bin:$PATH" make && \
make install
5 .安裝libvpx
cd ~/ffmpeg_sources && \
git -C libvpx pull 2> /dev/null || git clone --depth 1 https://chromium.googlesource.com/webm/libvpx.git && \
cd libvpx && \
PATH="$HOME/bin:$PATH" ./configure --prefix="$HOME/ffmpeg_build" --disable-examples --disable-unit-tests --enable-vp9-highbitdepth --as=yasm && \
PATH="$HOME/bin:$PATH" make && \
make install
6. libfdk-aac
cd ~/ffmpeg_sources && \
git -C fdk-aac pull 2> /dev/null || git clone --depth 1 https://github.com/mstorsjo/fdk-aac && \
cd fdk-aac && \
autoreconf -fiv && \
./configure --prefix="$HOME/ffmpeg_build" --disable-shared && \
make && \
make install
這里需要增加 --with-pic,原因后面會(huì)提到
7.libmp3lame
cd ~/ffmpeg_sources && \
wget -O lame-3.100.tar.gz https://downloads.sourceforge.net/project/lame/lame/3.100/lame-3.100.tar.gz && \
tar xzvf lame-3.100.tar.gz && \
cd lame-3.100 && \
PATH="$HOME/bin:$PATH" ./configure --prefix="$HOME/ffmpeg_build" --bindir="$HOME/bin" --disable-shared --enable-nasm? && \
PATH="$HOME/bin:$PATH" make && \
make install
這里需要增加 --with-pic,原因后面會(huì)提到姑荷,這個(gè)很重要盒延!
8. libopus
cd ~/ffmpeg_sources && \
git -C opus pull 2> /dev/null || git clone --depth 1 https://github.com/xiph/opus.git && \
cd opus && \
./autogen.sh && \
./configure --prefix="$HOME/ffmpeg_build" --disable-shared? && \
make && \
make install
這里需要增加 --with-pic,原因后面會(huì)提到
9.libogg
cd ~/ffmpeg_sources && \curl -O -L http://downloads.xiph.org/releases/ogg/libogg-1.3.3.tar.gz && \tar xzvf libogg-1.3.3.tar.gz? && \cd libogg-1.3.3 && \./configure --prefix="$HOME/ffmpeg_build" --disable-shared && \make && \make install
10.libvorbis
cd ~/ffmpeg_sources && \curl -O -L http://downloads.xiph.org/releases/vorbis/libvorbis-1.3.5.tar.gz && \tar xzvf libvorbis-1.3.5.tar.gz && \cd libvorbis-1.3.5 && \./configure --prefix="$HOME/ffmpeg_build" --with-ogg="$HOME/ffmpeg_build/build" --disable-shared && \make && \make install && \
11. FFmpeg
cd ~/ffmpeg_sources && \
cd ffmpeg && \
PATH="$HOME/bin:$PATH" PKG_CONFIG_PATH="$HOME/ffmpeg_build/lib/pkgconfig" ./configure --prefix="$HOME/ffmpeg_build" --pkg-config-flags="--static" --extra-cflags="-I$HOME/ffmpeg_build/include"? --extra-ldflags="-L$HOME/ffmpeg_build/lib" --extra-libs="-lpthread -lm" --bindir="$HOME/bin" --enable-gpl? --enable-libmp3lame --enable-libfdk-aac --enable-libopus --enable-libvorbis --enable-libvpx --enable-libx264 --enable-libx265 --disable-avdevice --disable-swscale --disable-encoders --enable-encoder=libmp3lame? --enable-nonfree
PATH="$HOME/bin:$PATH" make && \
make install && \
hash -r
如果一切順利 , FFMPEG 可以順利編譯完成鼠冕。 添寺,編譯好的ffmpeg 是一系列的靜態(tài)庫,當(dāng)然還有一些外部的靜態(tài)庫懈费,在~/ffmpeg_build/lib? 中计露,后續(xù)的工作將要引用這些靜態(tài)庫。
三憎乙、測試靜態(tài)庫
為了驗(yàn)證這些靜態(tài)庫可以使用票罐,我先寫了一個(gè)可執(zhí)行程序,調(diào)用ffmpeg 靜態(tài)庫泞边,代碼如下:
audioproc.cpp
int main(int argc, char ** argv){
? ? AVFormatContext *fmt_ctx = NULL;
????AVStream *audio_stream = NULL;
????int audio_stream_idx = -1;
????int ret=avformat_open_input(&fmt_ctx, argv[1], NULL, NULL) ;
????if(ret==0) {
? ? ? cout<<"ffmpeg is running\r\n";
????}else{
????????cout<<"ffmpeg is wrong"\r\n;
????}
????return 0;
}
Makefile 文件如下所示:
MAKE_DIR=.
SRC_DIR=$(MAKE_DIR)/src/
OBJ_DIR=$(MAKE_DIR)/obj/
OUT_DIR=$(MAKE_DIR)/bin/
##定義使用的路徑
FFMPEG_INCLUDE=-I/home/zhd/ffmpeg_build/include
FFMPEG_LIB=-L/home/zhd/ffmpeg_build/lib -lavformat -lavfilter -lavcodec? -lswresample -lavutil
EXEC=audioproc
SRCS:= $(wildcard $(SRC_DIR)*.cpp)
OBJS:= $(patsubst %.cpp,$(OBJ_DIR)%.o,$(notdir $(SRCS)))
CXX? ? =g++
CFLAGS=-g
EXEC:=$(EXEC_DIR)$(EXEC)
$(EXEC): $(EXE_OBJS)
????$(CXX) $(EXE_OBJS) $(FFMPEG_LIB)? -o? $@
$(OBJ_DIR)%.o:$(SRC_DIR)%.cpp
????$(CXX) $(CFLAGS) $(FFMPEG_INCLUDE)? -c? $< -o $@
clean :
????rm -rf ./obj/*.o
????rm -rf ./bin/*
通過make 來編譯鏈接该押。 會(huì)有很多的鏈接錯(cuò)誤,例如:
/home/zhd/ffmpeg_sources/ffmpeg/libavcodec/libfdk-aacdec.c:227:對(duì)‘a(chǎn)acDecoder_Open’未定義的引用
/home/zhd/ffmpeg_sources/ffmpeg/libavcodec/libfdk-aacdec.c:234:對(duì)‘a(chǎn)acDecoder_ConfigRaw’未定義的引用
/home/zhd/ffmpeg_sources/ffmpeg/libavcodec/libfdk-aacdec.c:241:對(duì)‘a(chǎn)acDecoder_SetParam’未定義的引用
/home/zhd/ffmpeg_sources/ffmpeg/libavcodec/libfdk-aacdec.c:265:對(duì)‘a(chǎn)acDecoder_SetParam’未定義的引用
這些錯(cuò)誤都是由于缺少引用庫造成的阵谚。這點(diǎn)和windows 不一樣蚕礼,在windows 中,我們一般把ffmpeg 編譯成動(dòng)態(tài)庫梢什,然后直接引用這些動(dòng)態(tài)庫即可奠蹬。當(dāng)然這些錯(cuò)誤也一目了然,肯定是少了一些靜態(tài)庫沒有加入嗡午。根據(jù)錯(cuò)誤提示囤躁,可以判斷是哪個(gè)庫沒有加入。例如上述錯(cuò)誤就需要加入 -lfdk-aac? 這個(gè)靜態(tài)庫荔睹。為此我們?cè)黾觙fmpeg引入的一邊外部靜態(tài)庫:
-lfdk-aac -lx264 -lx265 -lvorbis -logg? -lopus -lmp3lame
繼續(xù)make 狸演,還是出現(xiàn)問題如下:
/home/zhd/ffmpeg_sources/ffmpeg/libavformat/mov.c:5121:對(duì)‘uncompress’未定義的引用
此時(shí)需要 增加? -lz 解決 。繼續(xù) make 后应媚,錯(cuò)誤少了不少严沥,但還是有錯(cuò):
/home/zhd/ffmpeg_sources/ffmpeg/libavcodec/allcodecs.c:840:對(duì)‘pthread_once’未定義的引用
加入 -lpthread? 到鏈接庫后面≈薪 繼續(xù)make編譯消玄, 發(fā)現(xiàn)錯(cuò)誤又少了不少,但還是有錯(cuò):
/home/zhd/ffmpeg_sources/ffmpeg/libavcodec/vaapi_decode.c:87:對(duì)‘vaCreateBuffer’未定義的引用
/home/zhd/ffmpeg_sources/ffmpeg/libavutil/hwcontext_vdpau.c:471:對(duì)‘vdp_device_create_x11’未定義的引用
發(fā)現(xiàn)這些錯(cuò)誤和vaapi 有關(guān)系的丢胚。 所以引入-lX11 -lva -lvdpau -lva-drm -lva-x11 庫翩瓜。再make ,這回順利編譯成功携龟。
至此兔跌,按照本次項(xiàng)目的需要,ffmpeg 的需要引用的靜態(tài)庫
FFMPEG_LIB=-L/home/zhd/ffmpeg_build/lib -lavformat -lavfilter -lavcodec? -lswresample -lavutil? -lfdk-aac -lx264 -lx265 -lvorbis -logg? -lopus -lmp3lame? -lX11 -lva -lvdpau -lva-drm -lva-x11 -lz -lpthread
編譯成功峡蟋, 開始運(yùn)行 audioproc. 運(yùn)行正常坟桅,說明我們編譯的庫也是正確的!
四. 編寫調(diào)用ffmpeg 的動(dòng)態(tài)庫
? ? ? 不忘初心华望,回到最開始,需要做一個(gè)動(dòng)態(tài)庫調(diào)用 ffmpeg 仅乓,這樣赖舟,使用庫的時(shí)候,不需要引用長長的lib 列表了夸楣。對(duì)于引用這些庫的同學(xué)來說宾抓,會(huì)很方便。對(duì)于我這個(gè)windows 老手來說豫喧。? it‘ s easy .let's go!? 但沒想到遇到了很多問題石洗。請(qǐng)向下看:
4.1 編寫代碼
我們更改 audioproc.cpp? 如下:
int processaudio(char *filepath){
????AVFormatContext *fmt_ctx = NULL;
????AVStream *audio_stream = NULL;
????int audio_stream_idx = -1;
????int ret=avformat_open_input(&fmt_ctx, filepath, NULL, NULL) ;
????if(ret==0) {
? ? ? cout<<"ffmpeg is running\r\n";
????}else{
????????cout<<"ffmpeg is wrong\r\n";
????}
????return ret;
}
int main()? //這個(gè)需要添加,要不會(huì)報(bào)錯(cuò)
{
? ? return 0;
}
4.2 增加一個(gè)test.cpp
int main(int argc, char ** argv){
? ? processaudio((char *)argv[1]);
? ? return 0;
}
4.3 Makefile
MAKE_DIR=.
SRC_DIR=$(MAKE_DIR)/src/
OBJ_DIR=$(MAKE_DIR)/obj/
OUTPUT_DIR= $(MAKE_DIR)/bin/
##定義使用的路徑
FFMPEG_INCLUDE=-I/home/zhd/ffmpeg_build/include
FFMPEG_LIB=-L/home/zhd/ffmpeg_build/lib -lavformat -lavfilter -lavcodec? -lswresample -lavutil -lpthread? -lfdk-aac -lx264 -lx265 -lvorbis -logg? -lopus -lmp3lame? -lX11 -lva -lvdpau -lva-drm -lva-x11? -lm -lz
#定義執(zhí)行文件的名字
EXEC=test
LIBC=libaudioproc.so
#源文件紧显,自動(dòng)找所有.cpp文件讲衫,并將目標(biāo)定義為同名.o文件
SRCS:= $(wildcard $(SRC_DIR)*.cpp)
OBJS:= $(patsubst %.cpp,$(OBJ_DIR)%.o,$(notdir $(SRCS)))
EXE_OBJS=test.o
LIB_OBJS:=$(OBJ_DIR)audioextract.o
EXE_OBJS:=$(OBJ_DIR)test.o
CXX? ? =g++
CFLAGS=-g -fPIC
# 最終binary的名稱( 路徑+名稱 )
#?
EXEC:=$(OUTPUT_DIR)$(EXEC)
LIBC:=$(OUTPUT_DIR)$(LIBC)
all: $(LIBC)? $(EXEC)
dll:$(LIBC)
exe:$(EXEC)
#all:$(LIBC)
# LIB 庫 放到 鏈接命令鐘
$(LIBC):$(LIB_OBJS)
????$(CXX)? $(LIB_OBJS) $(FFMPEG_LIB) -shared -o $@?
#這里后續(xù)會(huì)增加-Wl,-Bsymbolic
$(EXEC): $(EXE_OBJS)
????$(CXX) $(EXE_OBJS) -L./bin/debug -laudioproc? -o? $@
$(OBJ_DIR)%.o:$(SRC_DIR)%.cpp
????$(CXX) $(CFLAGS) $(FFMPEG_INCLUDE)? -c? $< -o $@
程序?qū)懲炅耍屛覀內(nèi)ake吧鸟妙。
4.4 艱苦的編譯過程
首先 Make dll ....滿懷期望焦人,帶著幸福的樣子,然并卵重父, 報(bào)錯(cuò)了......
/usr/bin/ld: /home/zhd/ffmpeg_build/lib/libavcodec.a(vc1dsp_mmx.o): relocation R_X86_64_PC32 against symbol `ff_pw_9' can not be used when making a shared object; recompile with -fPIC
/usr/bin/ld: 最后的鏈結(jié)失敗: 錯(cuò)誤的值 .
這個(gè)鏈接錯(cuò)誤對(duì)于windows 程序員來說,有點(diǎn)陌生忽匈,這個(gè)咋處理房午?? 百度了半天丹允,沒有好的結(jié)果郭厌,試了半天也不行。 解決問題最好的辦法就是靜下來學(xué)習(xí)了雕蔽。 于是我就開始學(xué)習(xí)了....
4.5 -fPIC 的含義
? 在百度上搜了一個(gè)折柠,明白了fPIC 的具體含義, 是位置無關(guān)代碼,同學(xué)們可以了解一下這個(gè)鏈接批狐。
https://gcc.gnu.org/onlinedocs/gcc/Code-Gen-Options.html#Code-Gen-Options
看來出現(xiàn)這個(gè)問題是編譯方式的問題扇售,需要將對(duì)應(yīng)的庫用 fPIC 參數(shù)來編譯,libavcodec.a 是屬于ffmpeg 嚣艇,那我們需要帶這個(gè)參數(shù)重新編譯ffmpeg承冰。
那么如何用-fPIC 參數(shù)編譯ffmpeg 呢? 有一個(gè)鏈接說明了ffmpeg 編譯配置的參數(shù)食零,還是不錯(cuò)的
https://blog.csdn.net/momo0853/article/details/78043903
所以我們?cè)黾?-- enable-pic 來重新編譯ffmpeg困乒。make完成后,我們?cè)倮^續(xù)編譯調(diào)用ffmpeg 的動(dòng)態(tài)庫贰谣。錯(cuò)誤依舊娜搂。 再增加 -fPIC 到 extra-cflags="-I$HOME/ffmpeg_build/include -fPIC"
迁霎,經(jīng)過漫長的編譯還是提示一樣的錯(cuò)誤。
看來這個(gè)問題百宇,還有其它的思路......
經(jīng)過 百度的繼續(xù)查找考廉, 說 增加 在編譯自己的動(dòng)態(tài)庫時(shí)候,可以使用-Wl,-Bsymbolic 參數(shù)恳谎。
還是把ffmpeg 采用最開始的方式編譯:
PATH="$HOME/bin:$PATH" PKG_CONFIG_PATH="$HOME/ffmpeg_build/lib/pkgconfig" ./configure --prefix="$HOME/ffmpeg_build" --pkg-config-flags="--static" --extra-cflags="-I$HOME/ffmpeg_build/include"? --extra-ldflags="-L$HOME/ffmpeg_build/lib" --extra-libs="-lpthread -lm" --bindir="$HOME/bin" --enable-gpl? --enable-libmp3lame --enable-libfdk-aac --enable-libopus --enable-libvorbis --enable-libvpx --enable-libx264 --enable-libx265 --disable-avdevice --disable-swscale --disable-encoders --enable-encoder=libmp3lame? --enable-nonfree
PATH="$HOME/bin:$PATH" make && \
make install && \
hash -r
然后修改Makefile 如下:
$(LIBC):$(LIB_OBJS)
????$(CXX)? $(LIB_OBJS) $(FFMPEG_LIB) -shared -Wl,-Bsymbolic -o $@
? 這里分析 -Wl,-Bsymbolic.的含義
其中Wl表示將緊跟其后的參數(shù)芝此,傳遞給連接器ld。Bsymbolic表示強(qiáng)制采用本地的全局變量定義因痛,這樣就不會(huì)出現(xiàn)動(dòng)態(tài)鏈接庫的全局變量定義被應(yīng)用程序/動(dòng)態(tài)鏈接庫中的同名定義給覆蓋了的現(xiàn)象婚苹,難道是 “ff_pw_9” 變量重復(fù)聲明了?鸵膏?膊升? 試試看吧~~~
編譯后, 盡然發(fā)現(xiàn)原來的錯(cuò)誤沒了谭企! 但~~~出現(xiàn)了一個(gè)新的錯(cuò)誤:
/usr/bin/ld: /home/zhd/ffmpeg_build/lib/libfdk-aac.a(genericStds.o): relocation R_X86_64_PC32 against symbol `stdout@@GLIBC_2.2.5' can not be used when making a shared object; recompile with -fPIC? ?/usr/bin/ld: 最后的鏈結(jié)失敗: 錯(cuò)誤的值
還是先老辦法解決廓译,采用增加 fPIC的方式重新編譯 libfdk-aac.a
進(jìn)入fdk-aac 源碼目錄,代開 configure 腳本债查,發(fā)現(xiàn)支持 --with-pic . 那我們就用它了:
./configure --prefix="$HOME/ffmpeg_build" --disable-share? --with-pic
編譯完成 libfdk-aac 后非区,這個(gè)錯(cuò)誤就沒有了!但出來了 opus的錯(cuò)誤:
/usr/bin/ld: /home/zhd/ffmpeg_build/lib/libopus.a(celt.o): relocation R_X86_64_PC32 against symbol `stderr@@GLIBC_2.2.5' can not be used when making a shared object; recompile with -fPIC
用同樣的方法編譯盹廷,增加 --with-pic? 征绸,?opus這個(gè)錯(cuò)誤也消失了!俄占!
同樣的方式又編譯了 lameMP3 管怠,然后這個(gè)調(diào)用ffmpeg的dll就編譯完成了!缸榄!
通過 test 程序渤弛,引用audioproc,提示
./bin/test: error while loading shared libraries: libaudioproc.so: cannot open shared object file: No such file or directory
這個(gè)錯(cuò)誤就好解決了甚带∷希可以臨時(shí)通過 export LD_LIBRARY_PATH=xxx 解決
然后.......可以調(diào)用成功了!
以上的文字描述是我學(xué)習(xí)ffmpeg 在linux 下的使用過程中的一些經(jīng)驗(yàn)欲低。希望對(duì)您有用辕宏,如果那些地方不正確,歡迎留言批評(píng)指正砾莱。