alsa 用户空间编程【转】

本文转载自:http://blog.csdn.net/sjin_1314/article/details/12872581

[cpp] view plain copy
 
 print?
  1. /**alsa play test 
  2. *ALSA用户空间编译,ALSA驱动的声卡在用户空间,不宜直接使用 
  3. *文件接口中,而应使用alsa-lib 
  4. *打开---->设置参数--->读写音频数据 ALSA全部使用alsa-lib中的API 
  5. *交叉编译 
  6. *export LD_LIBRARY_PATH=$PWD:$LD_LIBRARY_PATH 
  7. *arm-linux-gcc -o alsa_play alsa_play_test.c -L. -lasound 
  8. *需要交叉编译后的libasound.so库的支持 
  9. */  
  10. #include <stdio.h>  
  11. #include <stdlib.h>  
  12. #include "alsa/asoundlib.h"  
  13.   
  14. int main(int argc, char *argv[])  
  15. {  
  16.     int i;  
  17.     int ret;  
  18.     int buf[128];  
  19.     unsigned int val;  
  20.     int dir=0;  
  21.     char *buffer;  
  22.     int size;  
  23.     snd_pcm_uframes_t frames;  
  24.     snd_pcm_uframes_t periodsize;  
  25.     snd_pcm_t *playback_handle;//PCM设备句柄pcm.h  
  26.     snd_pcm_hw_params_t *hw_params;//硬件信息和PCM流配置  
  27.     if (argc != 2) {  
  28.         printf("error: alsa_play_test [music name]\n");  
  29.         exit(1);  
  30.     }  
  31.     printf("play song %s by wolf\n", argv[1]);  
  32.     FILE *fp = fopen(argv[1], "rb");  
  33.     if(fp == NULL)  
  34.     return 0;  
  35.     fseek(fp, 100, SEEK_SET);  
  36.       
  37.     //1. 打开PCM,最后一个参数为0意味着标准配置  
  38.     ret = snd_pcm_open(&playback_handle, "default", SND_PCM_STREAM_PLAYBACK, 0);  
  39.     if (ret < 0) {  
  40.         perror("snd_pcm_open");  
  41.         exit(1);  
  42.     }  
  43.       
  44.     //2. 分配snd_pcm_hw_params_t结构体  
  45.     ret = snd_pcm_hw_params_malloc(&hw_params);  
  46.     if (ret < 0) {  
  47.         perror("snd_pcm_hw_params_malloc");  
  48.         exit(1);  
  49.     }  
  50.     //3. 初始化hw_params  
  51.     ret = snd_pcm_hw_params_any(playback_handle, hw_params);  
  52.     if (ret < 0) {  
  53.         perror("snd_pcm_hw_params_any");  
  54.         exit(1);  
  55.     }  
  56.     //4. 初始化访问权限  
  57.     ret = snd_pcm_hw_params_set_access(playback_handle, hw_params, SND_PCM_ACCESS_RW_INTERLEAVED);  
  58.     if (ret < 0) {  
  59.         perror("snd_pcm_hw_params_set_access");  
  60.         exit(1);  
  61.     }  
  62.     //5. 初始化采样格式SND_PCM_FORMAT_U8,8位  
  63.     ret = snd_pcm_hw_params_set_format(playback_handle, hw_params, SND_PCM_FORMAT_U8);  
  64.     if (ret < 0) {  
  65.         perror("snd_pcm_hw_params_set_format");  
  66.         exit(1);  
  67.     }  
  68.     //6. 设置采样率,如果硬件不支持我们设置的采样率,将使用最接近的  
  69.     //val = 44100,有些录音采样频率固定为8KHz  
  70.       
  71.   
  72.     val = 8000;  
  73.     ret = snd_pcm_hw_params_set_rate_near(playback_handle, hw_params, &val, &dir);  
  74.     if (ret < 0) {  
  75.         perror("snd_pcm_hw_params_set_rate_near");  
  76.         exit(1);  
  77.     }  
  78.     //7. 设置通道数量  
  79.     ret = snd_pcm_hw_params_set_channels(playback_handle, hw_params, 2);  
  80.     if (ret < 0) {  
  81.         perror("snd_pcm_hw_params_set_channels");  
  82.         exit(1);  
  83.     }  
  84.       
  85.     /* Set period size to 32 frames. */  
  86.     frames = 32;  
  87.     periodsize = frames * 2;  
  88.     ret = snd_pcm_hw_params_set_buffer_size_near(playback_handle, hw_params, &periodsize);  
  89.     if (ret < 0)   
  90.     {  
  91.          printf("Unable to set buffer size %li : %s\n", frames * 2, snd_strerror(ret));  
  92.            
  93.     }  
  94.           periodsize /= 2;  
  95.   
  96.     ret = snd_pcm_hw_params_set_period_size_near(playback_handle, hw_params, &periodsize, 0);  
  97.     if (ret < 0)   
  98.     {  
  99.         printf("Unable to set period size %li : %s\n", periodsize,  snd_strerror(ret));  
  100.     }  
  101.                                     
  102.     //8. 设置hw_params  
  103.     ret = snd_pcm_hw_params(playback_handle, hw_params);  
  104.     if (ret < 0) {  
  105.         perror("snd_pcm_hw_params");  
  106.         exit(1);  
  107.     }  
  108.       
  109.      /* Use a buffer large enough to hold one period */  
  110.     snd_pcm_hw_params_get_period_size(hw_params, &frames, &dir);  
  111.                                   
  112.     size = frames * 2; /* 2 bytes/sample, 2 channels */  
  113.     buffer = (char *) malloc(size);  
  114.     fprintf(stderr,  
  115.             "size = %d\n",  
  116.             size);  
  117.       
  118.     while (1)   
  119.     {  
  120.         ret = fread(buffer, 1, size, fp);  
  121.         if(ret == 0)   
  122.         {  
  123.               fprintf(stderr, "end of file on input\n");  
  124.               break;  
  125.         }   
  126.         else if (ret != size)   
  127.         {  
  128.         }  
  129.         //9. 写音频数据到PCM设备  
  130.         while(ret = snd_pcm_writei(playback_handle, buffer, frames)<0)  
  131.         {  
  132.             usleep(2000);  
  133.             if (ret == -EPIPE)  
  134.             {  
  135.                   /* EPIPE means underrun */  
  136.                   fprintf(stderr, "underrun occurred\n");  
  137.                   //完成硬件参数设置,使设备准备好  
  138.                   snd_pcm_prepare(playback_handle);  
  139.             }   
  140.             else if (ret < 0)   
  141.             {  
  142.                   fprintf(stderr,  
  143.                       "error from writei: %s\n",  
  144.                       snd_strerror(ret));  
  145.             }    
  146.         }  
  147.           
  148.     }         
  149.     //10. 关闭PCM设备句柄  
  150.     snd_pcm_close(playback_handle);  
  151.       
  152.     return 0;  
  153. }  
posted @ 2017-09-29 11:05  请给我倒杯茶  阅读(856)  评论(0编辑  收藏  举报