Linux下使用alsa-lib库完成音频开发: 实现放音和录音(从声卡获取PCM数据保存、向声卡写PCM数据输出)

简介: Linux下使用alsa-lib库完成音频开发: 实现放音和录音(从声卡获取PCM数据保存、向声卡写PCM数据输出)

一、环境介绍


系统: 虚拟机运行ubuntu18.04 (64位)

声卡: 电脑自带声卡

二、安装alsa-lib库

参考文章: https://blog.csdn.net/xiaolong1126626497/article/details/104916277

三、参考代码:从声卡获取PCM数据,实现录音功能

下面代码在命令行通过gcc编译运行:  读取声卡数据,保存为文件,结束录音可以按下Ctrl+C即可结束。

/*
 进行音频采集,采集pcm数据并直接保存pcm数据
 音频参数: 
   声道数:   1
   采样位数:  16bit、LE格式
   采样频率:  44100Hz
运行示例:
$ gcc linux_pcm_save.c -lasound
$ ./a.out hw:0 123.pcm
*/
#include <stdio.h>
#include <stdlib.h>
#include <alsa/asoundlib.h>
#include <signal.h>
#define AudioFormat SND_PCM_FORMAT_S16_LE  //指定音频的格式,其他常用格式:SND_PCM_FORMAT_U24_LE、SND_PCM_FORMAT_U32_LE
#define AUDIO_CHANNEL_SET   1         //1单声道   2立体声
#define AUDIO_RATE_SET 44100   //音频采样率,常用的采样频率: 44100Hz 、16000HZ、8000HZ、48000HZ、22050HZ
FILE *pcm_data_file=NULL;
int run_flag=0;
void exit_sighandler(int sig)
{
  run_flag=1;
}
int main(int argc, char *argv[])
{
  int i;
  int err;
  char *buffer;
  int buffer_frames = 1024;
  unsigned int rate = AUDIO_RATE_SET;
  snd_pcm_t *capture_handle;// 一个指向PCM设备的句柄
  snd_pcm_hw_params_t *hw_params; //此结构包含有关硬件的信息,可用于指定PCM流的配置
  /*注册信号捕获退出接口*/
  signal(2,exit_sighandler);
  /*PCM的采样格式在pcm.h文件里有定义*/
  snd_pcm_format_t format=AudioFormat; // 采样位数:16bit、LE格式
  /*打开音频采集卡硬件,并判断硬件是否打开成功,若打开失败则打印出错误提示*/
  // SND_PCM_STREAM_PLAYBACK 输出流
  // SND_PCM_STREAM_CAPTURE  输入流
  if ((err = snd_pcm_open (&capture_handle, argv[1],SND_PCM_STREAM_CAPTURE,0))<0) 
  {
    printf("无法打开音频设备: %s (%s)\n",  argv[1],snd_strerror (err));
    exit(1);
  }
  printf("音频接口打开成功.\n");
  /*创建一个保存PCM数据的文件*/
  if((pcm_data_file = fopen(argv[2], "wb")) == NULL)
  {
    printf("无法创建%s音频文件.\n",argv[2]);
    exit(1);
  } 
  printf("用于录制的音频文件已打开.\n");
  /*分配硬件参数结构对象,并判断是否分配成功*/
  if((err = snd_pcm_hw_params_malloc(&hw_params)) < 0) 
  {
    printf("无法分配硬件参数结构 (%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("硬件参数结构已分配成功.\n");
  /*按照默认设置对硬件对象进行设置,并判断是否设置成功*/
  if((err=snd_pcm_hw_params_any(capture_handle,hw_params)) < 0) 
  {
    printf("无法初始化硬件参数结构 (%s)\n", snd_strerror(err));
    exit(1);
  }
  printf("硬件参数结构初始化成功.\n");
  /*
    设置数据为交叉模式,并判断是否设置成功
    interleaved/non interleaved:交叉/非交叉模式。
    表示在多声道数据传输的过程中是采样交叉的模式还是非交叉的模式。
    对多声道数据,如果采样交叉模式,使用一块buffer即可,其中各声道的数据交叉传输;
    如果使用非交叉模式,需要为各声道分别分配一个buffer,各声道数据分别传输。
  */
  if((err = snd_pcm_hw_params_set_access (capture_handle,hw_params,SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) 
  {
    printf("无法设置访问类型(%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("访问类型设置成功.\n");
  /*设置数据编码格式,并判断是否设置成功*/
  if ((err=snd_pcm_hw_params_set_format(capture_handle, hw_params,format)) < 0) 
  {
    printf("无法设置格式 (%s)\n",snd_strerror(err));
    exit(1);
  }
  fprintf(stdout, "PCM数据格式设置成功.\n");
  /*设置采样频率,并判断是否设置成功*/
  if((err=snd_pcm_hw_params_set_rate_near(capture_handle,hw_params,&rate,0))<0) 
  {
    printf("无法设置采样率(%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("采样率设置成功\n");
  /*设置声道,并判断是否设置成功*/
  if((err = snd_pcm_hw_params_set_channels(capture_handle, hw_params,AUDIO_CHANNEL_SET)) < 0) 
  {
    printf("无法设置声道数(%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("声道数设置成功.\n");
  /*将配置写入驱动程序中,并判断是否配置成功*/
  if ((err=snd_pcm_hw_params (capture_handle,hw_params))<0) 
  {
    printf("无法向驱动程序设置参数(%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("参数设置成功.\n");
  /*使采集卡处于空闲状态*/
  snd_pcm_hw_params_free(hw_params);
  /*准备音频接口,并判断是否准备好*/
  if((err=snd_pcm_prepare(capture_handle))<0) 
  {
    printf("无法使用音频接口 (%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("音频接口准备好.\n");
  /*配置一个数据缓冲区用来缓冲数据*/
  //snd_pcm_format_width(format) 获取样本格式对应的大小(单位是:bit)
  int frame_byte=snd_pcm_format_width(format)/8;
  buffer=malloc(buffer_frames*frame_byte*AUDIO_CHANNEL_SET); //2048
  printf("缓冲区分配成功.\n");
  /*开始采集音频pcm数据*/
  printf("开始采集数据...\n");
  while(1) 
  {
    /*从声卡设备读取一帧音频数据:2048字节*/
    if((err=snd_pcm_readi(capture_handle,buffer,buffer_frames))!=buffer_frames) 
    {
        printf("从音频接口读取失败(%s)\n",snd_strerror(err));
        exit(1);
    }
    /*写数据到文件: 音频的每帧数据样本大小是16位=2个字节*/
    fwrite(buffer,(buffer_frames*AUDIO_CHANNEL_SET),frame_byte,pcm_data_file);  
    if(run_flag)
    {
      printf("停止采集.\n");
      break;
    }
  }
  /*释放数据缓冲区*/
  free(buffer);
  /*关闭音频采集卡硬件*/
  snd_pcm_close(capture_handle);
  /*关闭文件流*/
  fclose(pcm_data_file);
  return 0;
}

四、参考代码:从文件读取PCM数据,再写入到声卡设备,实现声音播放功能

下面代码在命令行通过gcc编译运行:  读取文件PCM音频数据,写入到声卡进行播放,结束播放可以按下Ctrl+C即可结束。

/*
 进行音频采集,读取存放pcm数据的文件通过声卡进行播放
 音频参数: 
   声道数:   1
   采样位数:  16bit、LE格式
   采样频率:  44100Hz
运行示例:
$ gcc linux_pcm_save.c -lasound
$ ./a.out hw:0 123.pcm
*/
#include <stdio.h>
#include <stdlib.h>
#include <alsa/asoundlib.h>
#include <signal.h>
#define AudioFormat SND_PCM_FORMAT_S16_LE  //指定音频的格式,其他常用格式:SND_PCM_FORMAT_U24_LE、SND_PCM_FORMAT_U32_LE
#define AUDIO_CHANNEL_SET   1         //1单声道   2立体声
#define AUDIO_RATE_SET 44100   //音频采样率,常用的采样频率: 44100Hz 、16000HZ、8000HZ、48000HZ、22050HZ
FILE *pcm_data_file=NULL;
int run_flag=0;
void exit_sighandler(int sig)
{
  run_flag=1;
}
//
int main(int argc, char *argv[])
{
  int i;
  int err;
  char *buffer;
  int buffer_frames = 1024;
  unsigned int rate = AUDIO_RATE_SET;
  snd_pcm_t *capture_handle;// 一个指向PCM设备的句柄
  snd_pcm_hw_params_t *hw_params; //此结构包含有关硬件的信息,可用于指定PCM流的配置
  /*注册信号捕获退出接口*/
  signal(2,exit_sighandler);
  /*PCM的采样格式在pcm.h文件里有定义*/
  snd_pcm_format_t format=AudioFormat; // 采样位数:16bit、LE格式
  /*打开音频采集卡硬件,并判断硬件是否打开成功,若打开失败则打印出错误提示*/
  // SND_PCM_STREAM_PLAYBACK 输出流
  // SND_PCM_STREAM_CAPTURE  输入流
  if ((err = snd_pcm_open (&capture_handle, argv[1],SND_PCM_STREAM_PLAYBACK,0))<0) 
  {
    printf("无法打开音频设备: %s (%s)\n",  argv[1],snd_strerror (err));
    exit(1);
  }
  printf("音频接口打开成功.\n");
  /*打开存放PCM数据的文件*/
  if((pcm_data_file = fopen(argv[2], "rb")) == NULL)
  {
    printf("无法打开%s音频文件.\n",argv[2]);
    exit(1);
  } 
  printf("用于播放的音频文件已打开.\n");
  /*分配硬件参数结构对象,并判断是否分配成功*/
  if((err = snd_pcm_hw_params_malloc(&hw_params)) < 0) 
  {
    printf("无法分配硬件参数结构 (%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("硬件参数结构已分配成功.\n");
  /*按照默认设置对硬件对象进行设置,并判断是否设置成功*/
  if((err=snd_pcm_hw_params_any(capture_handle,hw_params)) < 0) 
  {
    printf("无法初始化硬件参数结构 (%s)\n", snd_strerror(err));
    exit(1);
  }
  printf("硬件参数结构初始化成功.\n");
  /*
    设置数据为交叉模式,并判断是否设置成功
    interleaved/non interleaved:交叉/非交叉模式。
    表示在多声道数据传输的过程中是采样交叉的模式还是非交叉的模式。
    对多声道数据,如果采样交叉模式,使用一块buffer即可,其中各声道的数据交叉传输;
    如果使用非交叉模式,需要为各声道分别分配一个buffer,各声道数据分别传输。
  */
  if((err = snd_pcm_hw_params_set_access (capture_handle,hw_params,SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) 
  {
    printf("无法设置访问类型(%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("访问类型设置成功.\n");
  /*设置数据编码格式,并判断是否设置成功*/
  if ((err=snd_pcm_hw_params_set_format(capture_handle, hw_params,format)) < 0) 
  {
    printf("无法设置格式 (%s)\n",snd_strerror(err));
    exit(1);
  }
  fprintf(stdout, "PCM数据格式设置成功.\n");
  /*设置采样频率,并判断是否设置成功*/
  if((err=snd_pcm_hw_params_set_rate_near(capture_handle,hw_params,&rate,0))<0) 
  {
    printf("无法设置采样率(%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("采样率设置成功\n");
  /*设置声道,并判断是否设置成功*/
  if((err = snd_pcm_hw_params_set_channels(capture_handle, hw_params,AUDIO_CHANNEL_SET)) < 0) 
  {
    printf("无法设置声道数(%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("声道数设置成功.\n");
  /*将配置写入驱动程序中,并判断是否配置成功*/
  if ((err=snd_pcm_hw_params (capture_handle,hw_params))<0) 
  {
    printf("无法向驱动程序设置参数(%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("参数设置成功.\n");
  /*使采集卡处于空闲状态*/
  snd_pcm_hw_params_free(hw_params);
  /*准备音频接口,并判断是否准备好*/
  if((err=snd_pcm_prepare(capture_handle))<0) 
  {
    printf("无法使用音频接口 (%s)\n",snd_strerror(err));
    exit(1);
  }
  printf("音频接口准备好.\n");
  /*配置一个数据缓冲区用来缓冲数据*/
  //snd_pcm_format_width(format) 获取样本格式对应的大小(单位是:bit)
  int frame_byte=snd_pcm_format_width(format)/8;
  buffer=malloc(buffer_frames*frame_byte*AUDIO_CHANNEL_SET);
  printf("缓冲区分配成功.\n");
  /*开始采集音频pcm数据*/
  printf("开始播放音频数据...\n");
  int read_cnt;
  while(1) 
  {
    /*从文件读取数据: 音频的每帧数据样本大小是16位-2个字节*/
    read_cnt=fread(buffer,1,frame_byte*(buffer_frames*AUDIO_CHANNEL_SET),pcm_data_file);
    if(read_cnt<=0)break;
    /*向声卡设备写一帧音频数据:2048字节*/
    if((err=snd_pcm_writei(capture_handle,buffer,buffer_frames))!=buffer_frames) 
    {
        printf("向音频接口写数据失败(%s)\n",snd_strerror(err));
        exit(1);
    }
    if(run_flag)
    {
      printf("停止播放.\n");
      break;
    }
  }
  printf("播放完成.\n");
  /*释放数据缓冲区*/
  free(buffer);
  /*关闭音频采集卡硬件*/
  snd_pcm_close(capture_handle);
  /*关闭文件流*/
  fclose(pcm_data_file);
  return 0;
}


目录
相关文章
|
6月前
|
Ubuntu 搜索推荐 Linux
详解Ubuntu的strings与grep命令:Linux开发的实用工具。
这就是Ubuntu中的strings和grep命令,透明且强大。我希望你喜欢这个神奇的世界,并能在你的Linux开发旅程上,通过它们找到你的方向。记住,你的电脑是你的舞台,在上面你可以做任何你想做的事,只要你敢于尝试。
375 32
|
8月前
|
JavaScript Linux 网络安全
Termux安卓终端美化与开发实战:从下载到插件优化,小白也能玩转Linux
Termux是一款安卓平台上的开源终端模拟器,支持apt包管理、SSH连接及Python/Node.js/C++开发环境搭建,被誉为“手机上的Linux系统”。其特点包括零ROOT权限、跨平台开发和强大扩展性。本文详细介绍其安装准备、基础与高级环境配置、必备插件推荐、常见问题解决方法以及延伸学习资源,帮助用户充分利用Termux进行开发与学习。适用于Android 7+设备,原创内容转载请注明来源。
2034 77
|
7月前
|
关系型数据库 MySQL Linux
在Linux环境下备份Docker中的MySQL数据并传输到其他服务器以实现数据级别的容灾
以上就是在Linux环境下备份Docker中的MySQL数据并传输到其他服务器以实现数据级别的容灾的步骤。这个过程就像是一场接力赛,数据从MySQL数据库中接力棒一样传递到备份文件,再从备份文件传递到其他服务器,最后再传递回MySQL数据库。这样,即使在灾难发生时,我们也可以快速恢复数据,保证业务的正常运行。
357 28
|
7月前
|
Ubuntu Linux PHP
利用PHP压缩音频:Linux环境下的ffmpeg简易安装指南
希望这个指南能为你的编程之旅提供帮助。只需记住,每一行代码都像音乐的音符,组合在一起,创造出美妙的旋律。祝你编程愉快!
256 6
|
Linux API 开发工具
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
ijkplayer是由B站研发的移动端播放器,基于FFmpeg 3.4,支持Android和iOS。其源码托管于GitHub,截至2024年9月15日,获得了3.24万星标和0.81万分支,尽管已停止更新6年。本文档介绍了如何在Linux环境下编译ijkplayer的so库,以便在较新的开发环境中使用。首先需安装编译工具并调整/tmp分区大小,接着下载并安装Android SDK和NDK,最后下载ijkplayer源码并编译。详细步骤包括环境准备、工具安装及库编译等。更多FFmpeg开发知识可参考相关书籍。
448 0
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
|
弹性计算 Linux 数据库
阿里云国际版如何迁移Linux云服务器系统盘中的数据
阿里云国际版如何迁移Linux云服务器系统盘中的数据
|
Java Linux API
Linux设备驱动开发详解2
Linux设备驱动开发详解
207 6
|
存储 Linux 开发工具
如何进行Linux内核开发【ChatGPT】
如何进行Linux内核开发【ChatGPT】
|
Linux 开发工具
linux下使用gcp拷贝数据的时候显示进度条
linux下使用gcp拷贝数据的时候显示进度条
170 2
|
安全 关系型数据库 MySQL
linux服务器MySQL数据从磁盘拷贝以及恢复
偶有感触:遇到这个问题,经过一个下午的排查, 终于解决。
353 0