Linux下音频开发: 读取声卡PCM数据保存到文件(alsa-lib库)

简介: Linux下音频开发: 读取声卡PCM数据保存到文件(alsa-lib库)

一、环境介绍

系统: ubuntu18.04 (64位)

声卡: 电脑自带

二、功能介绍

通过alsa-lib库读取电脑声卡的PCM数据保存到文件。

在ubuntu系统上安装alsa-lib库方法:

 sudo apt-get install libasound2-dev

如果是在其他发行版linux系统上或者需要在嵌入式linux系统上使用alsa-lib库,可以下载alsa-lib源码包,自行编译。

 开源ALSA架构的官网地址:https://www.alsa-project.org/wiki/Main_Page

image.png

三、获取本机可用的声卡设备

获取声卡可以使用arecord -L命令。

在ubuntu系统下如果没有这个命令,直接根据提示安装一个即可。

   示例:   一般  hw:0 或者 default 表示默认声卡设备。

wbyq@wbyq:/mnt/hgfs/linux-share-dir/linux_c/linux_pcm_save$ arecord -L
default
    Playback/recording through the PulseAudio sound server
null
    Discard all samples (playback) or generate zero samples (capture)
pulse
    PulseAudio Sound Server
sysdefault:CARD=AudioPCI
    Ensoniq AudioPCI, ES1371 DAC2/ADC
    Default Audio Device
front:CARD=AudioPCI,DEV=0
    Ensoniq AudioPCI, ES1371 DAC2/ADC
    Front speakers
iec958:CARD=AudioPCI,DEV=0
    Ensoniq AudioPCI, ES1371 DAC2/ADC
    IEC958 (S/PDIF) Digital Audio Output
dmix:CARD=AudioPCI,DEV=0
    Ensoniq AudioPCI, ES1371 DAC2/ADC
    Direct sample mixing device
dsnoop:CARD=AudioPCI,DEV=0
    Ensoniq AudioPCI, ES1371 DAC2/ADC
    Direct sample snooping device
hw:CARD=AudioPCI,DEV=0
    Ensoniq AudioPCI, ES1371 DAC2/ADC
    Direct hardware device without any conversions
plughw:CARD=AudioPCI,DEV=0
    Ensoniq AudioPCI, ES1371 DAC2/ADC
    Hardware device with all software conversions

四、示例代码

参考文档:https://users.suse.com/~mana/alsa090_howto.html

/*
 进行音频采集,采集pcm数据并直接保存pcm数据
 音频参数: 
   声道数:   2
   采样位数:  16bit、LE格式
   采样频率:  44100Hz
*/
#include <stdio.h>
#include <stdlib.h>
#include <alsa/asoundlib.h>
#include <signal.h>
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 = 128;
  unsigned int rate = 44100;// 常用的采样频率: 44100Hz 、16000HZ、8000HZ、48000HZ、22050HZ
  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=SND_PCM_FORMAT_S16_LE; // 采样位数:16bit、LE格式
  /*打开音频采集卡硬件,并判断硬件是否打开成功,若打开失败则打印出错误提示*/
  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,2)) < 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");
  /*配置一个数据缓冲区用来缓冲数据*/
  buffer=malloc(128*snd_pcm_format_width(format)/8*2);
  printf("缓冲区分配成功.\n");
  /*开始采集音频pcm数据*/
  printf("开始采集数据...\n");
  while(1) 
  {
    /*从声卡设备读取一帧音频数据*/
    if((err=snd_pcm_readi(capture_handle,buffer,buffer_frames))!=buffer_frames) 
    {
        printf("从音频接口读取失败(%s)\n",snd_strerror(err));
        exit(1);
    }
    /*写数据到文件*/
    fwrite(buffer,(buffer_frames*2),sizeof(short),pcm_data_file);
    if(run_flag)
    {
      printf("停止采集.\n");
      break;
    }
  }
  /*释放数据缓冲区*/
  free(buffer);
  /*关闭音频采集卡硬件*/
  snd_pcm_close(capture_handle);
  /*关闭文件流*/
  fclose(pcm_data_file);
  return 0;
}

编译方法:

gcc linux_pcm_save.c -lasound

运行方法:

./a.out hw:0 123.pcm

录制过程中如果想要结束录制,直接按下Ctrl+C即可。

成功之后,会在当前目录下生成123.pcm文件。

 

五、播放PCM数据

Windows下可以使用audacity软件进行播放PCM裸流数据。

或者通过ffplay工具(安装ffmpeg会带上这个工具)。

ffplay -ar 44100 -channels 2 -f s16le -i 123.pcm

audacity软件下载地址:

https://download.csdn.net/download/xiaolong1126626497/12252685    

image.png

image.png

image.png

目录
相关文章
|
2月前
|
存储 编译器 Linux
动态链接的魔法:Linux下动态链接库机制探讨
本文将深入探讨Linux系统中的动态链接库机制,这其中包括但不限于全局符号介入、延迟绑定以及地址无关代码等内容。
838 25
|
4月前
|
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开发知识可参考相关书籍。
142 0
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
|
4月前
|
弹性计算 Linux 数据库
阿里云国际版如何迁移Linux云服务器系统盘中的数据
阿里云国际版如何迁移Linux云服务器系统盘中的数据
|
5月前
|
存储 Linux 开发工具
如何进行Linux内核开发【ChatGPT】
如何进行Linux内核开发【ChatGPT】
|
6月前
|
Java Linux API
Linux设备驱动开发详解2
Linux设备驱动开发详解
71 6
|
6月前
|
消息中间件 算法 Unix
Linux设备驱动开发详解1
Linux设备驱动开发详解
80 5
|
6月前
|
Linux 开发工具
linux下使用gcp拷贝数据的时候显示进度条
linux下使用gcp拷贝数据的时候显示进度条
43 2
|
7天前
|
Linux
Linux系统之whereis命令的基本使用
Linux系统之whereis命令的基本使用
50 23
Linux系统之whereis命令的基本使用
|
3月前
|
Linux 网络安全 数据安全/隐私保护
Linux 超级强大的十六进制 dump 工具:XXD 命令,我教你应该如何使用!
在 Linux 系统中,xxd 命令是一个强大的十六进制 dump 工具,可以将文件或数据以十六进制和 ASCII 字符形式显示,帮助用户深入了解和分析数据。本文详细介绍了 xxd 命令的基本用法、高级功能及实际应用案例,包括查看文件内容、指定输出格式、写入文件、数据比较、数据提取、数据转换和数据加密解密等。通过掌握这些技巧,用户可以更高效地处理各种数据问题。
344 8
|
3月前
|
监控 Linux
如何检查 Linux 内存使用量是否耗尽?这 5 个命令堪称绝了!
本文介绍了在Linux系统中检查内存使用情况的5个常用命令:`free`、`top`、`vmstat`、`pidstat` 和 `/proc/meminfo` 文件,帮助用户准确监控内存状态,确保系统稳定运行。
1128 6