當(dāng)前位置:首頁 > 單片機(jī) > 單片機(jī)
[導(dǎo)讀]   目前,智能手機(jī)、PDA和平板電腦等越來越多的嵌入式設(shè)備支持高清視頻采集和播放功能,高清視頻的采集或播放功能正廣泛用于游戲設(shè)備、監(jiān)控設(shè)備、視頻會(huì)議設(shè)備和數(shù)字網(wǎng)絡(luò)電視等嵌入式系統(tǒng)中。這些功能的

  目前,智能手機(jī)、PDA和平板電腦等越來越多的嵌入式設(shè)備支持高清視頻采集和播放功能,高清視頻的采集或播放功能正廣泛用于游戲設(shè)備、監(jiān)控設(shè)備、視頻會(huì)議設(shè)備和數(shù)字網(wǎng)絡(luò)電視等嵌入式系統(tǒng)中。這些功能的實(shí)現(xiàn)建立在高性能視頻硬件編解碼技術(shù)基礎(chǔ)之上。本文闡述了基于FFMpeg的H.264視頻硬件編解碼在S3C6410處理器上的實(shí)現(xiàn)方法,為數(shù)字娛樂、視頻監(jiān)控和視頻通信系統(tǒng)開發(fā)過程中的高清視頻硬件編解碼的實(shí)現(xiàn)提供參考。


  FFmpeg[1]是一個(gè)開源免費(fèi)跨平臺(tái)的視頻和音頻流方案,屬于自由軟件。它包含非常先進(jìn)的音頻/視頻編解碼庫libavcodec,提供了錄制、轉(zhuǎn)換以及流化音視頻的完整解決方案。FFmpeg支持MPEG4、FLV等40多種編碼,以及AVI、ASF等90多種解碼。目前國內(nèi)較為流行的播放器暴風(fēng)影音和國外較為流行的Mplayer在音頻/視頻編解碼方面都用到了FFmpeg。

  S3C6410[2]是三星公司推出的應(yīng)用處理器芯片,基于ARM11架構(gòu),主頻最高可達(dá)800 MHz。它具有多媒體硬件加速功能,其中包括大于30FPs的MPEG4 SP、H.264/263 BP和VC1(WMV9)多種視頻硬件編解碼,可用于手機(jī)、平板電腦和游戲機(jī)等手持移動(dòng)設(shè)備和其他高性能嵌入式設(shè)備。國產(chǎn)手機(jī)魅族M8的處理器使用的就是S3C6410。

  雖然FFmpeg提供了簡單的應(yīng)用程序編程接口(API),可以很方便地實(shí)現(xiàn)多種格式的視頻軟件編解碼[3],但是軟件編解碼在處理復(fù)雜視頻編解碼(如H.264)時(shí)無法運(yùn)用到處理速度不快、內(nèi)存空間不多的嵌入式環(huán)境中。為了在資源有限的嵌入式環(huán)境下使用FFmpeg實(shí)現(xiàn)復(fù)雜視頻編解碼,下面在分析FFmpeg視頻編碼流程和S3C6410處理器視頻編解碼方法的基礎(chǔ)上,闡述嵌入式Linux操作系統(tǒng)下基于FFmpeg的H.264硬件編解碼在S3C6410處理器上的實(shí)現(xiàn)方法。

  1 FFmpeg視頻編解碼流程

  FFmpeg主要有encode/decode、muxer/demuxer和內(nèi)存操作3個(gè)模塊。encode/decode模塊用于音視頻的編碼和解碼,存放在libavcodec子目錄中;muxer/demuxer模塊用于音頻和視頻的合并與分離(也稱混合器模塊),存放在libavformat目錄中;內(nèi)存等常用模塊存放于libavutil目錄中。下面以解碼過程為例分析FFmpeg視頻編解碼流程。

  解碼基本流程共分4步:

 ?、?注冊(cè)所有可能用到的編解碼器和混合器。av_register_all(void)函數(shù)中通過執(zhí)行 REGISTER_MUXDEMUX(X,x)和REGISTER_ENCDEC(X,x),把所有FFmpeg支持的混合器和編解碼器相關(guān)信息以鏈?zhǔn)降慕Y(jié)構(gòu)存放在內(nèi)存中。

 ?、?打開視頻文件。av_open_input_file(AVFormatContext **IC_ptr,const char *filename,AVInputFormat *fmt,int buf_size,AVFormatParameters *ap)函數(shù)中偵測文件的格式,根據(jù)文件格式從鏈?zhǔn)降幕旌掀髦姓业较鄬?duì)應(yīng)的混合器(demuxer)并分離出視頻信息。

 ?、?獲取視頻信息。通過av_find_stream_info(AVFormatContext *ic)函數(shù)獲取視頻格式。根據(jù)視頻格式,在鏈?zhǔn)降囊曨l解碼器中找到相應(yīng)的視頻解碼器,并通過avcodec_open(AVCodeCContext *avctx,AVCodec *codec)函數(shù)將解碼器打開用于下一步視頻的解碼。

  ④ 解碼一幀視頻,通過 avcodec_decode_video(AVCodecContext *avctx,AVFrame *picture,int *got_picture_ptr,const uint8_t *buf,int buf_size)函數(shù)解碼一幀視頻。

  FFmpeg的編碼過程與解碼過程類似,不同的是第3步根據(jù)要求編碼的格式在鏈?zhǔn)降囊曨l編碼器中找到相應(yīng)的視頻編碼器,并執(zhí)行編碼過程。

  通過以上對(duì)FFmpeg視頻編解碼流程分析可以知道,為了在FFmpeg中添加自定義的視頻編解碼器,并在程序運(yùn)行時(shí)使用這個(gè)編解碼器,關(guān)鍵在于如下兩點(diǎn):

  ① 根據(jù)FFmpeg對(duì)編解碼器的描述,實(shí)現(xiàn)自定義編解碼器。

 ?、?通過REGISTER_ENCDEC(X,x)函數(shù)將自定義的視頻編解碼器添加到視頻編解碼器鏈中。在獲取視頻信息時(shí),保證需要編碼或解碼的視頻能找到視頻編解碼器鏈中自定義的視頻編解碼器。

  2 S3C6410處理器視頻編解碼方法

  S3C6410視頻編解碼軟件架構(gòu)[4]如圖1所示。底層為操作系統(tǒng)空間,上層為用戶空間,視頻編解碼器通過驅(qū)動(dòng)和操作系統(tǒng)以設(shè)備文件的形式使用,使用的方法和普通文件一樣,包括文件打開和關(guān)閉、文件讀寫和輸入/輸出控制(ioctl,input/output control)。

圖1 S3C6410視頻編解碼軟件架構(gòu)

  具體操作方法如下:

 ?、?通過open函數(shù)打開編解碼器設(shè)備文件;

  ② 使用mmap方法在用戶空間和驅(qū)動(dòng)空間之間映射輸入/輸出緩存空間,這樣做的好處是可以快速進(jìn)行數(shù)據(jù)輸入/輸出;

 ?、?通過ioctl設(shè)備編解碼參數(shù),初始化編解碼器;

 ?、?輸入數(shù)據(jù),通過ioctl執(zhí)行編解碼過程,輸出數(shù)據(jù);

 ?、?通過close方法關(guān)閉編解碼器設(shè)備文件。

  值得注意的是,無論編碼還是解碼,處理的數(shù)據(jù)都是以一幀幀的形式操作的,所以第4步是一個(gè)不斷循環(huán)的過程,直到所有數(shù)據(jù)處理完成。另外,雖然編解碼器以設(shè)備文件的形式使用,但是它不能使用標(biāo)準(zhǔn)的文件讀寫操作,查看編解碼的設(shè)備驅(qū)動(dòng)可以發(fā)現(xiàn),其文件讀寫函數(shù)是空的,這一點(diǎn)三星公司的開發(fā)文檔并沒有說明。

  3 H.264硬件編解碼實(shí)現(xiàn)

  FFmpeg的H.264硬件編解碼[5]實(shí)現(xiàn)就是自定義一個(gè)視頻編解碼器,加入到FFmpeg庫中。這個(gè)視頻編解碼器使用S3C6410處理視頻硬件編解碼功能來實(shí)現(xiàn)H.264的視頻編碼和解碼過程,這樣使用FFmpeg庫的多媒體程序可以用訪問FFmpeg其他編解碼器一樣的方法使用這個(gè)自定義的編解碼器。添加自定義編解碼器的關(guān)鍵是根據(jù)FFmpeg中對(duì)編解碼的描述定義編解碼器,并實(shí)現(xiàn)定義中的相關(guān)函數(shù)。

  在libavcodec/avcodec.h中的AVCodec結(jié)構(gòu)體是定義FFmpeg編解碼器的關(guān)鍵結(jié)構(gòu)體,包括編解碼器的名字、類型(聲音/視頻)、編解碼器的識(shí)別號(hào)(CodecID)、支持格式和一些用于初始化、編碼、解碼和關(guān)閉的函數(shù)指針。

  typedef struct AVCodec {

  const char *name;

  enum CodecType type;

  enum CodecID id;

  int priv_data_size;

  int (*init)(AVCodecContext *);

  int (*encode)(AVCodecContext *,uint8_t *buf,int buf_size,void *data);

  int (*close)(AVCodecContext *);

  int (*decode)(AVCodecContext *,void *outdata,int *outdata_size,

  uint8_t *buf,int buf_size);

  int capabilities;

  struct AVCodec *next;

  void (*flush)(AVCodecContext *);

  const AVRational *supported_framerates;

  const enum PixelFormat *pix_fmts;

  } AVCodec;

  H.264硬件編解碼器定義如下:

  AVCodec s3cx264_encoder = {

  .name="s3cx264",

  .type=AVMEDIA_TYPE_VIDEO,

  .id=CODEC_ID_H264,

  .init=X264_init,

  .encode=X264_frame,

  .decode=X264_decode,

  .close=X264_close,

  …

  };

  解碼器的名字為s3cx264,類型為視頻。CodecID為H264,表示這個(gè)解碼器用于H.264視頻編解碼。初始化、編碼、解碼和關(guān)閉函數(shù)指針分別指向X264_init、X264_frame、X264_decodec和X264_close函數(shù)。

  添加s3cx264編解碼器到編解器鏈中,關(guān)鍵是通過修改libavcodec/allcodecs.c文件實(shí)現(xiàn),修改如下:

  REGISTER_ENCDEC (ASV1,asv1);

  REGISTER_ENCDEC (S3CX264,s3cx264);

  //添加s3cx264編解碼器

  REGISTER_ENCDEC (ASV2,asv2);

  這樣,在程序運(yùn)行時(shí)調(diào)用av_register_all(void)函數(shù)后,就可以把自定義的編解碼器s3cx264添加到FFmpeg存放在內(nèi)存中的解編碼器鏈中。值得提出的是,對(duì)同一個(gè)視頻格式FFmpeg有多個(gè)編解碼器與之相對(duì)應(yīng)。如H.264格式的視頻,F(xiàn)Fmpeg本身就帶有對(duì)應(yīng)的軟解碼器,現(xiàn)在添加了硬解碼器,為了避免不確定是哪一個(gè)解碼器在執(zhí)行,可以把自定義的硬件編解碼器在注冊(cè)時(shí)放在注冊(cè)過程的最前面,這樣編解碼器在添加到解編器鏈中時(shí)就會(huì)放在靠前的位置,查找時(shí)就可以優(yōu)于軟件解碼器找到硬解碼器。

  把硬件編解碼器s3cx264注冊(cè)到編解碼器鏈后,還要完成X264_init、X264_frame、X264_decodec和X264_close函數(shù),編解碼器才能正常工作。以下結(jié)合前面對(duì)S3C6410視頻編解碼過程的分析,以編碼為例詳細(xì)闡述實(shí)現(xiàn)過程。

  定義X264Context結(jié)構(gòu)體,保存設(shè)備文件描述符、編碼參數(shù)和輸入/輸出地址等信息,用于FFmpeg模塊間數(shù)據(jù)的傳遞:

  typedef struct X264Context {

  int dev_fd;

  uint8_t *addr;

  s3c_mfc_enc_init_arg_t enc_init;

  s3c_mfc_enc_exe_arg_t enc_exe;

  s3c_mfc_get_buf_addr_arg_t get_buf_addr;

  uint8_t *in_buf,*out_buf;

  AVFrame out_pic;

  } X264Context;

  X264_init實(shí)現(xiàn)的是編碼器初始化過程, 用于編碼器設(shè)備文件的打開、內(nèi)存空間的映射、編碼參數(shù)設(shè)置和獲取編解碼數(shù)據(jù)輸入/輸出地址。

  static av_cold int X264_init(AVCodecContext *avctx){

  X264Context *x4 = avctx?>priv_data;

  //打開編碼器設(shè)備文件

  x4?>dev_fd = open(MFC_DEV_NAME,O_RDWR"O_NDELAY);

  //內(nèi)存空間映射

  x4?>addr = (uint8_t *) mmap(0,BUF_SIZE,PROT_READ |PROT_WRITE,MAP_SHARED,x4?>dev_fd,0);

  //編碼參數(shù)設(shè)置

  ioctl(x4?>dev_fd,S3C_MFC_IOCTL_MFC_H264_ENC_INIT,&x4?>enc_init);

  //獲取輸入/輸出地址

  x4?>get_buf_addr.in_usr_data = (int)x4?>addr;

  ioctl(x4?>dev_fd,S3C_MFC_IOCTL_MFC_GET_YUV_BUF_ADDR,&x4?>get_buf_addr);

  x4?>in_buf = (uint8_t *)x4?>get_buf_addr.out_buf_addr;

  x4?>get_buf_addr.in_usr_data = (int)x4?>addr;

  ioctl(x4?>dev_fd,S3C_MFC_IOCTL_MFC_GET_LINE_BUF_ADDR,&x4?>get_buf_addr);

  x4?>out_buf = (uint8_t *)x4?>get_buf_addr.out_buf_addr;

  return 0;

  }

  ioctl的參數(shù)為S3C_MFC_IOCTL_MFC_H264_ENC_INIT,表示使用H.264編碼。

  X264_frame函數(shù)執(zhí)行編碼過程。需要注意的是data參數(shù)保存了需要編碼的數(shù)據(jù),是一個(gè)四維的數(shù)組,要把它轉(zhuǎn)換成一維數(shù)組用于S3C6410編碼器輸入。另外,編碼數(shù)據(jù)存在空的情況,也就是空幀。這是需要處理的,方法是返回“0”,表示沒有輸出數(shù)據(jù),否則程序運(yùn)行時(shí)會(huì)出現(xiàn)段錯(cuò)誤。

static int X264_frame(AVCodecC

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動(dòng)力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉