? ? ? ?作為一個(gè)老的windows程序員,由于工作的原因,需要在linux 下寫(xiě)一個(gè)音頻處理的服務(wù)。 接收到這個(gè)任務(wù)后,第一思路還是windows 思路,寫(xiě)一個(gè)動(dòng)態(tài)庫(kù),調(diào)用ffmpeg 庫(kù),然后在動(dòng)態(tài)庫(kù)中寫(xiě)入自己的音頻處理流程。 然后分享這個(gè)動(dòng)態(tài)庫(kù),供其他程序員使用。按照這個(gè)思路,linux 也當(dāng)然可以。
? ? ? ?說(shuō)干就干,而且是擼起袖子加油干,我在華為云申請(qǐng)了虛擬機(jī),安裝了ubantu.然后開(kāi)始寫(xiě)我的音頻處理庫(kù)了。沒(méi)想到,作為windows 老手的linux新手來(lái)說(shuō),整個(gè)過(guò)程還是很是艱辛,也遇到了很多問(wèn)題。下面請(qǐng)聽(tīng)我娓娓道來(lái),如果恰好您也在編譯ffmpeg, 或許能給您一些幫助。
? ? ? 如果您不聽(tīng)我啰嗦,也可以到下面鏈接 查看安裝方法,如果遇到這個(gè)問(wèn)題再返回看。
? ? ??linux 下的ffmpeg的編譯和編寫(xiě)引用ffmpeg的動(dòng)態(tài)庫(kù) - 簡(jiǎn)書(shū)
? ? ? 本文參考了很多網(wǎng)頁(yè),最重要的還是這個(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 獲取 依賴庫(kù)
sudo apt-get update -qq && sudo apt-get -y install \
? autoconf \
? automake \
? build-essential \
? cmake \
? git-core \
? libtool \
? pkg-config \
? texinfo \
? wget \
? zlib1g-dev
上面這些庫(kù),對(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)庫(kù),當(dāng)然還有一些外部的靜態(tài)庫(kù),在~/ffmpeg_build/lib? 中,后續(xù)的工作將要引用這些靜態(tài)庫(kù)。
三、測(cè)試靜態(tài)庫(kù)
為了驗(yàn)證這些靜態(tài)庫(kù)可以使用,我先寫(xiě)了一個(gè)可執(zhí)行程序,調(diào)用ffmpeg 靜態(tài)庫(kù),代碼如下:
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/*
通過(guò)make 來(lái)編譯鏈接。 會(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ò)誤都是由于缺少引用庫(kù)造成的。這點(diǎn)和windows 不一樣,在windows 中,我們一般把ffmpeg 編譯成動(dòng)態(tài)庫(kù),然后直接引用這些動(dòng)態(tài)庫(kù)即可。當(dāng)然這些錯(cuò)誤也一目了然,肯定是少了一些靜態(tài)庫(kù)沒(méi)有加入。根據(jù)錯(cuò)誤提示,可以判斷是哪個(gè)庫(kù)沒(méi)有加入。例如上述錯(cuò)誤就需要加入 -lfdk-aac? 這個(gè)靜態(tài)庫(kù)。為此我們?cè)黾觙fmpeg引入的一邊外部靜態(tài)庫(kù):
-lfdk-aac -lx264 -lx265 -lvorbis -logg? -lopus -lmp3lame
繼續(xù)make ,還是出現(xiàn)問(wè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? 到鏈接庫(kù)后面。 繼續(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 庫(kù)。再make ,這回順利編譯成功。
至此,按照本次項(xiàng)目的需要,ffmpeg 的需要引用的靜態(tài)庫(kù)
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
編譯成功, 開(kāi)始運(yùn)行 audioproc. 運(yùn)行正常,說(shuō)明我們編譯的庫(kù)也是正確的!
四. 編寫(xiě)調(diào)用ffmpeg 的動(dòng)態(tài)庫(kù)
? ? ? 不忘初心,回到最開(kāi)始,需要做一個(gè)動(dòng)態(tài)庫(kù)調(diào)用 ffmpeg ,這樣,使用庫(kù)的時(shí)候,不需要引用長(zhǎng)長(zhǎng)的lib 列表了。對(duì)于引用這些庫(kù)的同學(xué)來(lái)說(shuō),會(huì)很方便。對(duì)于我這個(gè)windows 老手來(lái)說(shuō)。? it‘ s easy .let's go!? 但沒(méi)想到遇到了很多問(wèn)題。請(qǐng)向下看:
4.1 編寫(xiě)代碼
我們更改 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 庫(kù) 放到 鏈接命令鐘
$(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 艱苦的編譯過(guò)程
首先 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 程序員來(lái)說(shuō),有點(diǎn)陌生,這個(gè)咋處理?? 百度了半天,沒(méi)有好的結(jié)果,試了半天也不行。 解決問(wèn)題最好的辦法就是靜下來(lái)學(xué)習(xí)了。 于是我就開(kāi)始學(xué)習(xí)了....
4.5 -fPIC 的含義
? 在百度上搜了一個(gè),明白了fPIC 的具體含義, 是位置無(wú)關(guān)代碼,同學(xué)們可以了解一下這個(gè)鏈接。
https://gcc.gnu.org/onlinedocs/gcc/Code-Gen-Options.html#Code-Gen-Options
看來(lái)出現(xiàn)這個(gè)問(wèn)題是編譯方式的問(wèn)題,需要將對(duì)應(yīng)的庫(kù)用 fPIC 參數(shù)來(lái)編譯,libavcodec.a 是屬于ffmpeg ,那我們需要帶這個(gè)參數(shù)重新編譯ffmpeg。
那么如何用-fPIC 參數(shù)編譯ffmpeg 呢? 有一個(gè)鏈接說(shuō)明了ffmpeg 編譯配置的參數(shù),還是不錯(cuò)的
https://blog.csdn.net/momo0853/article/details/78043903
所以我們?cè)黾?-- enable-pic 來(lái)重新編譯ffmpeg。make完成后,我們?cè)倮^續(xù)編譯調(diào)用ffmpeg 的動(dòng)態(tài)庫(kù)。錯(cuò)誤依舊。 再增加 -fPIC 到 extra-cflags="-I$HOME/ffmpeg_build/include -fPIC"
,經(jīng)過(guò)漫長(zhǎng)的編譯還是提示一樣的錯(cuò)誤。
看來(lái)這個(gè)問(wèn)題,還有其它的思路......
經(jīng)過(guò) 百度的繼續(xù)查找, 說(shuō) 增加 在編譯自己的動(dòng)態(tài)庫(kù)時(shí)候,可以使用-Wl,-Bsymbolic 參數(shù)。
還是把ffmpeg 采用最開(kāi)始的方式編譯:
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)鏈接庫(kù)的全局變量定義被應(yīng)用程序/動(dòng)態(tài)鏈接庫(kù)中的同名定義給覆蓋了的現(xiàn)象,難道是 “ff_pw_9” 變量重復(fù)聲明了??? 試試看吧~~~
編譯后, 盡然發(fā)現(xiàn)原來(lái)的錯(cuò)誤沒(méi)了! 但~~~出現(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 源碼目錄,代開(kāi) configure 腳本,發(fā)現(xiàn)支持 --with-pic . 那我們就用它了:
./configure --prefix="$HOME/ffmpeg_build" --disable-share? --with-pic
編譯完成 libfdk-aac 后,這個(gè)錯(cuò)誤就沒(méi)有了!但出來(lái)了 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就編譯完成了?。?/p>
通過(guò) test 程序,引用audioproc,提示
./bin/test: error while loading shared libraries: libaudioproc.so: cannot open shared object file: No such file or directory
這個(gè)錯(cuò)誤就好解決了??梢耘R時(shí)通過(guò) export LD_LIBRARY_PATH=xxx 解決
然后.......可以調(diào)用成功了!
以上的文字描述是我學(xué)習(xí)ffmpeg 在linux 下的使用過(guò)程中的一些經(jīng)驗(yàn)。希望對(duì)您有用,如果那些地方不正確,歡迎留言批評(píng)指正。