PCM音頻設備的操做(轉)

 對音頻設備的操做主要是初始化音頻設備以及往音頻設備發送 PCM(Pulse Code Modulation)數據。爲了方便,本文使用 ALSA(Advanced Linux Sound Architecture)提供的庫和驅動。在編譯和運行本文中的 MP3 流媒體播放器的時候,必須先安裝 ALSA 相關的文件。
 本文用到的主要對 PCM 設備操做的函數分爲 PCM 設備初始化的函數以及 PCM 接口的一些操做函數。
PCM 硬件設備參數設置和初始化的函數有:html

 

  1. int  snd_pcm_hw_params_malloc (snd_pcm_hw_params_t **ptr)  
  2. int  snd_pcm_hw_params_any (snd_pcm_t *pcm, snd_pcm_hw_params_t *params)  
  3. void snd_pcm_hw_params_free (snd_pcm_hw_params_t *obj)  
  4. int  snd_pcm_hw_params_set_access ( snd_pcm_t *pcm,   
  5.                                     snd_pcm_hw_params_t *params,   
  6.                                     snd_pcm_access_t _access)  
  7. int  snd_pcm_hw_params_set_format ( snd_pcm_t *pcm,   
  8.                                     snd_pcm_hw_params_t *params,   
  9.                                     snd_pcm_format_t val)  
  10. int  snd_pcm_hw_params_set_channels(snd_pcm_t *pcm,   
  11.                                     snd_pcm_hw_params_t *params,   
  12.                                     unsigned int val)  
  13. int snd_pcm_hw_params_set_rate_near(snd_pcm_t *pcm,   
  14.                                     snd_pcm_hw_params_t *params,   
  15.                                     unsigned int *val, int *dir) 

PCM 接口函數有:linux

 

  1. int   snd_pcm_hw_params (snd_pcm_t *pcm, snd_pcm_hw_params_t *params)  
  2. int   snd_pcm_prepare (snd_pcm_t *pcm)  
  3. int   snd_pcm_open (snd_pcm_t **pcm, const char *name,   
  4.                     snd_pcm_stream_t stream, int mode)  
  5. int   snd_pcm_close (snd_pcm_t *pcm)  
  6. snd_pcm_sframes_t   snd_pcm_writei (snd_pcm_t *pcm,   
  7.                     const void *buffer, snd_pcm_uframes_t size) 

這些函數用到了 snd_pcm_hw_params_t 結構,此結構包含用來播放 PCM 數據流的硬件信息配置。在往音頻設備(聲卡)寫入音頻數據以前,必須設置訪問類型、採樣格式、採樣率、聲道數等。程序員

首先使用 snd_pcm_open () 打開 PCM 設備,在 ALSA 中,PCM 設備都有名字與之對應。好比咱們能夠定義 PCM 設備名字爲 char *pcm_name = "plughw:0,0"。 最重要的 PCM 設備接口是「plughw」以及「hw」接口。 使用「plughw」接口,程序員沒必要過多關心硬件,並且若是設置的配置參數和實際硬件支持的參數不一致,ALSA 會自動轉換數據。若是使用「hw」接口,咱們就必須檢測硬件是否支持設置的參數了。Plughw 後面的兩個數字分別表示設備號和次設備(subdevice)號。web

snd_pcm_hw_params_malloc( ) 在棧中分配 snd_pcm_hw_params_t 結構的空間,而後使用 snd_pcm_hw_params_any( ) 函數用聲卡的全配置空間參數初始化已經分配的 snd_pcm_hw_params_t 結構。snd_pcm_hw_params_set_access ( ) 設置訪問類型,經常使用訪問類型的宏定義有:網絡

  1. SND_PCM_ACCESS_RW_INTERLEAVED 

交錯訪問。在緩衝區的每一個 PCM 幀都包含全部設置的聲道的連續的採樣數據。好比聲卡要播放採樣長度是 16-bit 的 PCM 立體聲數據,表示每一個 PCM 幀中有 16-bit 的左聲道數據,而後是 16-bit 右聲道數據。函數

  1. SND_PCM_ACCESS_RW_NONINTERLEAVED 

非交錯訪問。每一個 PCM 幀只是一個聲道須要的數據,若是使用多個聲道,那麼第一幀是第一個聲道的數據,第二幀是第二個聲道的數據,依此類推。spa

函數 snd_pcm_hw_params_set_format() 設置數據格式,主要控制輸入的音頻數據的類型、無符號仍是有符號、是 little-endian 仍是 bit-endian。好比對於 16-bit 長度的採樣數據能夠設置爲:code

  1. SND_PCM_FORMAT_S16_LE      有符號16 bit Little Endian   
  2. SND_PCM_FORMAT_S16_BE      有符號16 bit Big Endian   
  3. SND_PCM_FORMAT_U16_LE      無符號16 bit Little Endian   
  4. SND_PCM_FORMAT_U16_BE      無符號 16 bit Big Endian  
  5. 好比對於 32-bit 長度的採樣數據能夠設置爲:  
  6. SND_PCM_FORMAT_S32_LE      有符號32 bit Little Endian   
  7. SND_PCM_FORMAT_S32_BE      有符號32 bit Big Endian   
  8. SND_PCM_FORMAT_U32_LE      無符號32 bit Little Endian   
  9. SND_PCM_FORMAT_U32_BE      無符號 32 bit Big Endian 

函數 snd_pcm_hw_params_set_channels() 設置音頻設備的聲道,常見的就是單聲道和立體聲,若是是立體聲,設置最後一個參數爲2。snd_pcm_hw_params_set_rate_near () 函數設置音頻數據的最接近目標的採樣率。snd_pcm_hw_params( ) 從設備配置空間選擇一個配置,讓函數 snd_pcm_prepare() 準備好 PCM 設備,以便寫入 PCM 數據。snd_pcm_writei() 用來把交錯的音頻數據寫入到音頻設備。orm

初始化 PCM 設備的例程以下:htm

初始化 PCM 設備的例程
 

  1. /* open a PCM device */ 
  2. int open_device(struct mad_header const *header)  
  3. {  
  4.    int err;  
  5.    snd_pcm_hw_params_t *hw_params;  
  6.    char  *pcm_name = "plughw:0,0";  
  7.    int rate = header->samplerate;  
  8.    int channels = 2;  
  9.  
  10.    if (header->mode == 0) {  
  11.       channels = 1;  
  12.    } else {  
  13.       channels = 2;  
  14.    }  
  15.  
  16.    if ((err = snd_pcm_open (&playback_handle,   
  17.                             pcm_name, SND_PCM_STREAM_PLAYBACK, 0)) < 0) {  
  18.       printf("cannot open audio device %s (%s)\n",  
  19.       pcm_name,  
  20.       snd_strerror (err));  
  21.       return -1;  
  22.    }  
  23.  
  24.    if ((err = snd_pcm_hw_params_malloc (&hw_params)) < 0) {  
  25.       printf("cannot allocate hardware parameter structure (%s)\n",  
  26.       snd_strerror (err));  
  27.       return -1;  
  28.    }  
  29.  
  30.    if ((err = snd_pcm_hw_params_any (playback_handle, hw_params)) < 0) {  
  31.       printf("cannot initialize hardware parameter structure (%s)\n",  
  32.       snd_strerror (err));  
  33.       return -1;  
  34.    }  
  35.  
  36.  
  37.    if ((err = snd_pcm_hw_params_set_access (playback_handle, hw_params,   
  38.               SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) {  
  39.       printf("cannot set access type (%s)\n",  
  40.       snd_strerror (err));  
  41.       return -1;  
  42.    }  
  43.  
  44.       
  45.    if ((err = snd_pcm_hw_params_set_format (playback_handle,   
  46.               hw_params, SND_PCM_FORMAT_S32_LE)) < 0) {  
  47.       printf("cannot set sample format (%s)\n",  
  48.       snd_strerror (err));  
  49.       return -1;  
  50.    }  
  51.    if ((err = snd_pcm_hw_params_set_rate_near (playback_handle,   
  52.               hw_params, &rate, 0)) < 0) {  
  53.       printf("cannot set sample rate (%s)\n",  
  54.       snd_strerror (err));  
  55.       return -1;  
  56.    }  
  57.  
  58.    if ((err = snd_pcm_hw_params_set_channels (playback_handle,   
  59.               hw_params, channels)) < 0) {  
  60.       printf("cannot set channel count (%s)\n",  
  61.       snd_strerror (err));  
  62.       return -1;  
  63.    }  
  64.  
  65.    if ((err = snd_pcm_hw_params (playback_handle,   
  66.               hw_params)) < 0) {  
  67.       printf("cannot set parameters (%s)\n",  
  68.       snd_strerror (err));  
  69.       return -1;  
  70.    }  
  71.  
  72.    snd_pcm_hw_params_free (hw_params);  
  73.    if ((err = snd_pcm_prepare (playback_handle)) < 0) {  
  74.       printf("cannot prepare audio interface for use (%s)\n",  
  75.       snd_strerror (err));  
  76.       return -1;  
  77.    }  
  78.  
  79.    return 0;  
  80. }  

這裏配置的 PCM 格式是 SND_PCM_FORMAT_S32_LE,採樣的格式是每一個採樣有 32-bit 的數據,數據按照 little-endian 存放。若是經過 mad_frame_decode() 函數獲得 PCM 數據後,要求每一個採樣數據只佔 16-bit,須要把數據進行MAD的定點類型到 signed short 類型進行轉換。那麼,PCM 數據如何寫入聲卡中呢?函數實現例程以下所示:

PCM 數據寫入聲卡函數實現例程
 

  1. while (nsamples--) {  
  2. /* nsamples 是採樣的數目 */ 
  3.        signed int sample;  
  4.  
  5.        sample = pcm->samples[0][j];  
  6.        *(OutputPtr++) = sample & 0xff;  
  7.        *(OutputPtr++) = (sample >> 8);  
  8.        *(OutputPtr++) = (sample >> 16);  
  9.        *(OutputPtr++) = (sample >> 24);  
  10.  
  11.        if (nchannels == 2) {  
  12.           sample = pcm->samples[1][j];  
  13.           *(OutputPtr++) = sample  & 0xff;  
  14.           *(OutputPtr++) = sample >> 8;  
  15.           *(OutputPtr++) = (sample >> 16);  
  16.           *(OutputPtr++) = (sample >> 24);  
  17.  
  18.        }  
  19.        j++;  
  20.  
  21.    }  
  22.    if ((err = snd_pcm_writei (playback_handle, buf, samples)) < 0) {  
  23.       err = xrun_recovery(playback_handle, err);  
  24.       if (err < 0) {  
  25.          printf("Write error: %s\n", snd_strerror(err));  
  26.          return -1;  
  27.       }  
  28.    }  

這裏用到了 http://www.alsa-project.org/ 關於 ALSA 文檔中的例子函數 xrun_recovery( )。詳細例子請參見http://www.alsa-project.org/alsa-doc/alsa-lib/_2test_2pcm_8c-example.html。使用此函數的目的是避免出現因爲網絡緣由,聲卡不能及時獲得音頻數據而使得 snd_pcm_writei() 不能正常連續工做。實際上在xrun_recovery( ) 中,又調用 snd_pcm_prepare()  snd_pcm_resume() 以實現能「恢復錯誤」的功能。-EPIPE錯誤表示應用程序沒有及時把 PCM 採樣數據送入ASLA 庫。xrun_recovery() 函數以下所示:


xrun_recovery() 函數
 

  1. int xrun_recovery(snd_pcm_t *handle, int err)  
  2. {  
  3.    if (err == -EPIPE) {    /* under-run */ 
  4.       err = snd_pcm_prepare(handle);  
  5.  
  6.    if (err < 0)  
  7.       printf("Can't recovery from underrun, prepare failed: %s\n",  
  8.          snd_strerror(err));  
  9.       return 0;  
  10.    } else if (err == -ESTRPIPE) {  
  11.       while ((err = snd_pcm_resume(handle)) == -EAGAIN)  
  12.          sleep(1);       /* wait until the suspend flag is released */ 
  13.          if (err < 0) {  
  14.             err = snd_pcm_prepare(handle);  
  15.          if (err < 0)  
  16.             printf("Can't recovery from suspend, prepare failed: %s\n",  
  17.               snd_strerror(err));  
  18.       }  
  19.       return 0;  
  20.    }  
  21.    return err;  
  22. }  

知道了具體的音頻設備操做方法,就該使用 MAD 提供的函數具體實現解碼了。函數 mp3_decode_buf( ) 提供了使用 libmad 解碼的方法。首先調用 mad_stream_buffer() 函數把 MP3 流數據和 decode_stream 關聯,而後開始循環解碼數據。若是在解碼數據過程當中,有不完整 PCM 數據幀,那麼 decode_stream.error 的值就是MAD_ERROR_BUFLEN,且 decode_stream.next_frame 不爲 NULL。這時候,把剩餘的未解碼的數據再拷貝到數據解碼緩衝區裏。 mad_frame_decode( ) 函數從 decode_stream 中獲得 PCM 數據。

mad_frame_decode( ) 函數從 decode_stream 中獲得 PCM 數據

  1. int mp3_decode_buf(char *input_buf, int size)  
  2. {  
  3.   int decode_over_flag = 0;  
  4.   int remain_bytes = 0;  
  5.   int ret_val = 0;  
  6.   mad_stream_buffer(&decode_stream, input_buf, size);  
  7.   decode_stream.error = MAD_ERROR_NONE;  
  8.   while (1)  
  9.   {  
  10.       if (decode_stream.error == MAD_ERROR_BUFLEN) {  
  11.         if (decode_stream.next_frame != NULL) {  
  12.            remain_bytes = decode_stream.bufend - decode_stream.next_frame;  
  13.            memcpy(input_buf, decode_stream.next_frame, remain_bytes);  
  14.            return remain_bytes;  
  15.         }  
  16.       }  
  17.       ret_val = mad_frame_decode(&decode_frame, &decode_stream);  
  18.      /* 省略部分代碼 */ 
  19.      ...  
  20.      if (ret_val == 0) {  
  21.          if (play_frame(&decode_frame) == -1) {  
  22.             return -1;  
  23.          }  
  24.       }  
  25.       /* 後面代碼省略 */ 
  26.       ...  
  27.    }  
  28.  
  29.    return 0;  
  30. }  

 

 
recommend from :http://www.ibm.com/developerworks/cn/linux/l-cn-libmadmp3player/index.html

 

本文出自 「驛落黃昏」 博客,請務必保留此出處http://yiluohuanghun.blog.51cto.com/3407300/868048

相關文章
相關標籤/搜索