经典ALSA音频读取通过UDP局域网播放

经典ALSA音频读取通过UDP局域网播放

参考:https://www.jianshu.com/p/10a66697d40a

特别注意的是:

  1. alsa采集音频的帧数,与接收端接收的帧数,一定要匹配,比如设为1024,那就都设为1024.
  2. 还有,采集声卡的声道数,如果本身声卡不支持双声道,只支持单声道,使用-c 2,是采集不到声音的。可以先用arecord 命令采集试一下。播放的时候,也需要设置音频源的为单声道。
  3. 这个例子经过设置好这些参数,可以正常运行,不过会有点延时。


alsa_udp_r.c:

#include "alsa/asoundlib.h"

struct snd_pcm

{

    snd_pcm_t *handle;

    snd_pcm_uframes_t frame;

};

int init_pcm_capture(snd_pcm_t **handle,snd_pcm_uframes_t *frame)

{

    //1.定义pcm句柄

    snd_pcm_hw_params_t *params = NULL;

    //2.打开pcm设备---采集方式打开

    //(SND_PCM_STREAM_CAPTURE:打开模式)

    snd_pcm_open(handle, "default", SND_PCM_STREAM_CAPTURE, 0);

    //3 分配参数空间, 设置默认参数

    snd_pcm_hw_params_alloca(¶ms);

    //分配空间

    snd_pcm_hw_params_any(*handle, params);  //默认值

    //4.设置交错模式

    snd_pcm_hw_params_set_access(*handle, params, SND_PCM_ACCESS_RW_INTERLEAVED);

    //5设置量化

    snd_pcm_hw_params_set_format(*handle, params, SND_PCM_FORMAT_S16_LE);

    //6设置通道数据

    snd_pcm_hw_params_set_channels(*handle, params, 2);

    //7设置采样率

    int val = 44100;

    snd_pcm_hw_params_set_rate_near(*handle, params, &val,0);

    //8把设置好的参数设置到pcm设备

    snd_pcm_hw_params(*handle,params);

    //9.分配一个周期空间

    //获取一个周期的帧数

    snd_pcm_hw_params_get_period_size(params, frame, 0);

}

int init_pcm_playback(snd_pcm_t **handle,snd_pcm_uframes_t *frame)

{

    //1.定义pcm句柄

    //snd_pcm_t *handle = NULL;

    snd_pcm_hw_params_t *params = NULL;

    //2.打开pcm设备---采集方式打开

    //(SND_PCM_STREAM_CAPTURE:打开模式)

    snd_pcm_open(handle, "default", SND_PCM_STREAM_PLAYBACK, 0);

    //3 分配参数空间, 设置默认参数

    snd_pcm_hw_params_alloca(¶ms);

    //分配空间

    snd_pcm_hw_params_any(*handle, params);  //默认值

    //4.设置交错模式

    snd_pcm_hw_params_set_access(*handle, params, SND_PCM_ACCESS_RW_INTERLEAVED);

    //5设置量化

    snd_pcm_hw_params_set_format(*handle, params, SND_PCM_FORMAT_S16_LE);

    //6设置通道数据

snd_pcm_hw_params_set_channels(*handle, params, 2);

    //7设置采样率

    int val = 44100;

    snd_pcm_hw_params_set_rate_near(*handle, params, &val,0);

    //8把设置好的参数设置到pcm设备

    snd_pcm_hw_params(*handle,params);

    //9.分配一个周期空间

    //获取一个周期的帧数

    snd_pcm_hw_params_get_period_size(params, frame, 0);

}

int init_udp_sockfd(int *sockfd,struct sockaddr_in *s_addr,int argc,char **argv)

{

    if(argc < 1)

    {

    printf("input error\n");

    exit(0);

    }

    if((*sockfd = socket(AF_INET,SOCK_DGRAM,0)) < 0)

    {

perror("socket");

exit(0);

}

bzero(s_addr,sizeof(struct sockaddr_in));

s_addr->sin_family = AF_INET;

s_addr->sin_port  = htons(12345);

s_addr->sin_addr.s_addr = inet_addr(argv[1]);

int sin_size = 1;

if((setsockopt(*sockfd,SOL_SOCKET,SO_REUSEADDR,&sin_size,sizeof(int))) < 0)

{

perror("setsockopt");

exit(0);

}

return 0;

}

int main(int argc,char **argv)

{

    //初始化udp

    int sockfd;

    struct sockaddr_in s_addr;

    init_udp_sockfd(&sockfd,&s_addr,argc,argv);

    //初始化pcm

    struct snd_pcm sp_cp;

    struct snd_pcm sp_pb;

    sp_cp.handle = NULL;

    sp_cp.frame  = 0;

    sp_pb.handle = NULL;

    sp_pb.frame  = 0;    

    init_pcm_capture(&sp_cp.handle,&sp_cp.frame);

    //init_pcm_playback(&pb.handle,&sp_pb.frame);

    int size = sp_cp.frame * 4;//计算一个周期所需要的存储空间

    unsigned char *buffer = malloc(size);//分配一个周期的空间

    while(1)

    {

    snd_pcm_readi(sp_cp.handle, buffer, sp_cp.frame);//采集一个周期数据

    if(sendto(sockfd,buffer,size,0,(struct sockaddr *)&s_addr,sizeof(s_addr)) < 0)

       {

            perror("sendto");

            exit(0);

        }

    }

    //销毁pcm句柄

    snd_pcm_drain(sp_cp.handle);

    //关闭pcm设备

    snd_pcm_close(sp_cp.handle);

    //释放堆空间(存储一个周期数据)

    free(buffer);

    return 0;

}

上面代码属于嵌入式板上获取音频信息并且通过UDP发送到目标IP,使用./可执行文件名字 目标IP

alsa_net_s.c:

#include "alsa/asoundlib.h"

struct snd_pcm

{

snd_pcm_t *handle;

snd_pcm_uframes_t frame;

};

int init_pcm_playback(snd_pcm_t **handle,snd_pcm_uframes_t *frame)

{

//1.定义pcm句柄

snd_pcm_hw_params_t *params = NULL;

//2.打开pcm设备---采集方式打开

//(SND_PCM_STREAM_CAPTURE:打开模式)

snd_pcm_open(handle, "default", SND_PCM_STREAM_PLAYBACK, 0);

//3 分配参数空间, 设置默认参数

snd_pcm_hw_params_alloca(¶ms);

//分配空间

snd_pcm_hw_params_any(*handle, params);  //默认值

//4.设置交错模式

snd_pcm_hw_params_set_access(*handle, params, SND_PCM_ACCESS_RW_INTERLEAVED);

//5设置量化

snd_pcm_hw_params_set_format(*handle, params, SND_PCM_FORMAT_S16_LE);

//6设置通道数据

snd_pcm_hw_params_set_channels(*handle, params, 2);

//7设置采样率

int val = 44100;

snd_pcm_hw_params_set_rate_near(*handle, params, &val,0);

//8把设置好的参数设置到pcm设备

snd_pcm_hw_params(*handle,params);

//9.分配一个周期空间

//获取一个周期的帧数

snd_pcm_hw_params_get_period_size(params, frame, 0);

}

int init_udp_sockfd(int *sockfd,struct sockaddr_in *s_addr,int argc,char **argv)

{

if(argc < 1)

{

printf("input error\n");

exit(0);

}

if((*sockfd = socket(AF_INET,SOCK_DGRAM,0)) < 0)

{

perror("socket");

exit(0);

}

memset(s_addr,0,sizeof(struct sockaddr_in));

s_addr->sin_family = AF_INET;

s_addr->sin_port  = htons(12345);

s_addr->sin_addr.s_addr = inet_addr(argv[1]);

int sin_size = 1;

if((setsockopt(*sockfd,SOL_SOCKET,SO_REUSEADDR,&sin_size,sizeof(int))) < 0)

{

perror("setsockopt");

exit(0);

}

if(bind(*sockfd,(struct sockaddr *)s_addr,sizeof(struct sockaddr_in)) < 0)

{

perror("bind");

exit(0);

}

return 0;

}

int main(int argc,char **argv)

{

//初始化udp

int sockfd;

struct sockaddr_in s_addr,c_addr;

int len = sizeof(s_addr);

init_udp_sockfd(&sockfd,&s_addr,argc,argv);

//初始化pcm

struct snd_pcm sp_cp;

struct snd_pcm sp_pb;

sp_cp.handle = NULL;

sp_cp.frame  = 0;

sp_pb.handle = NULL;

sp_pb.frame  = 0;

//init_pcm_capture(&sp_cp.handle,&sp_cp.frame_cap);

init_pcm_playback(&sp_pb.handle,&sp_pb.frame);

int size = sp_pb.frame * 4;//计算一个周期所需要的存储空间

unsigned char *buffer = malloc(size);//分配一个周期的空间

while(1)

{

if(recvfrom(sockfd,buffer,size,0,(struct sockaddr *)&c_addr,&len) < 0)

{

perror("recvfrom");

exit(0);

}

snd_pcm_writei(sp_pb.handle, buffer, sp_pb.frame);//采集一个周期数据

}

//销毁pcm句柄

snd_pcm_drain(sp_pb.handle);

//关闭pcm设备

snd_pcm_close(sp_pb.handle);

//释放堆空间(存储一个周期数据)

free(buffer);

return 0;

}

上面代码绑定本机IP和端口接收UDP信息,把接收到的音频信息播放出来,使用./可执行文件名字 本机IP
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Linux ALSA-Lib库是用于读取和处理音频的开源库。它提供了一套API,可以让开发者通过 C/C++ 编程语言访问 Linux 系统中的音频设备。 ALSA-Lib 可以实现多种音频设备的读写,包括内置音频硬件,外部 USB 音频设备以及蓝牙音频ALSA-Lib 提供了一个叫做alsa-lib.h的头文件,这个头文件包含了常用的 ALSA-Lib API 函数。开发者可以根据具体需求来选择合适的函数,最常用的是snd_pcm_open()、snd_pcm_hw_params_set_xxx()、snd_pcm_writei()和snd_pcm_close(),这些函数分别用于打开、设置参数、写数据和关闭音频设备。 ALSA-Lib 提供的多种API函数使得开发者可以对音频进行多种高级操作。比如,开发者可以通过snd_pcm_drop()中止当前播放操作,通过snd_pcm_pause()暂停播放,通过snd_pcm_prepare()准备播放,还可以通过调用snd_pcm_avail_update()获取当前音频设备的缓冲区状态。 读取音频数据可以通过snd_pcm_readi()函数实现,这个函数会一次性从设备中读取指定数量的音频采样,并将其存储在一个指定的缓冲区中。开发者还可以选择使用snd_pcm_mmap_readi()和snd_pcm_mmap_begin()来读取音频采样,这两个函数可以实现更高效的读取。 在开发 Linux 音频应用程序时,ALSA-Lib 是非常重要的组件。通过掌握 ALSA-Lib 的 API 函数,开发者可以实现快速、高效地读取和处理音频数据。因此,熟悉 ALSA-Lib 是 Linux 音频开发工程师的必备技能之一。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值