[ALSA]从零开始,使用ALSA驱动播放一个音频

前言

最近学了不少有关音频相关的,最近搞一下ALSA驱动

安装

参考Linux应用开发【第八章】ALSA应用开发 中提到的ALSA库及工具章节,本文中有比较详细的有关ALSA驱动引用程序怎么安装的,这里不再赘述。

关于ALSA,就当成一个音频有关的,建立在Linux操作系统上类似WinApi的东西就可以了,实际应用起来也比较简单,之后会举例。

第一个项目

为了进行第一个ALSA项目的开发,那么我们先来建立一个CMake工程。在安装部分我们已经安装了ALSA驱动,其实这个时候就已经将ALSA的头文件加入到我们的项目中来了,可以直接试着在main文件里面加上这个头文件试试会不会报错:

#include <alsa/asoundlib.h>

如果这个时候CMake生成没有报错,那么引入就是正常的。

1. 引入alsa依赖

在Linux上开发是需要引入alsa的依赖以及符号文件的,这里我们需要以下语句,这里我提供一个最精简的CMakeLists.txt范本:

cmake_minimum_required(VERSION 3.0)
project(alsa_example)# 获得ALSA的Package
find_package(ALSA REQUIRED)add_executable(alsa_example main.cpp)# 链接
target_link_libraries(alsa_example PRIVATE ${ALSA_LIBRARY})# 头文件引入
target_include_directories(alsa_example PRIVATE ${ALSA_INCLUDE_DIRS})

这样我们就能先将ALSA的依赖引用了。

2.写个简单的程序来将当前所有的音频设备全部列举了

在正式操作一个音频设备之前,我们当然了,需要知道我们需要的音频设备到底是哪一个,那我们现在将所有的音频设备信息全部打印出来,如下:

#include <alsa/asoundlib.h>
#include <iostream>int main() {snd_ctl_t *handle;snd_ctl_card_info_t *info;int card = -1;snd_ctl_card_info_alloca(&info);// Iterate over sound cardswhile (snd_card_next(&card) >= 0 && card >= 0) {char name[32];sprintf(name, "hw:%d", card);if (snd_ctl_open(&handle, name, 0) < 0) {std::cerr << "Cannot open control for card " << card << std::endl;continue;}if (snd_ctl_card_info(handle, info) < 0) {std::cerr << "Cannot get card info for card " << card << std::endl;snd_ctl_close(handle);continue;}std::cout << "Card " << card << ": " << snd_ctl_card_info_get_name(info) << std::endl;snd_ctl_close(handle);}return 0;
}

返回值如下:

Card 0: rockchip,dp0
Card 1: rockchip-hdmi0
Card 2: rockchip-hdmi1
Card 3: rockchip,es8388
Card 4: rockchip,hdmiin

我这里是做了点处理的,实际上返回值里面并没有告诉你什么Card 0这种,而是告诉你hw:0,也就是hardware的意思,总之你只用知道每个音频设备的唯一标识应该是类似hw:0的形式。

这里我们可以看音频设备的名称,其中dp0 hdmi hdmi1 hdmiin都是显示器接口音频设备,我们这里显然是用不上的,那么我们就找到前面板的显示音频卡设备es8388(不同的设备输出声卡应该不同,这个需要根据具体声卡而定。

我们这里知道我们需要选中的前面板声卡设备是hw:3,这个之后我们会用到

3. 向音频设备里面写数据

我们知道,Linux下一切皆文件嘛,播放音频实际上就是往某个设备句柄中写入音频信号,这里我们尝试一下向音频设备写入信号。

流程如下:

  1. 使用二进制流的方式打开mp3文件
  2. 打开声卡pcm句柄
  3. 设置pcm句柄的属性
  4. 循环将二进制流以buffer的形式写入到pcm句柄内

实际代码如下:

#include <alsa/asoundlib.h>
#include <iostream>
#include <fstream>
#include <vector>#define PCM_DEVICE "hw:3"  // Replace with your desired device, e.g., "hw:3"
#define MP3_FILE_PATH "/home/orangepi/workhsop/testProject/123.mp3"  // Replace with your MP3 file pathint main() {const char *mp3File = MP3_FILE_PATH;// Open the MP3 filestd::ifstream file(mp3File, std::ios::binary);if (!file) {std::cerr << "Error opening file: " << mp3File << std::endl;return 1;}// Initialize ALSAsnd_pcm_t *handle;if (snd_pcm_open(&handle, PCM_DEVICE, SND_PCM_STREAM_PLAYBACK, 0) < 0) {std::cerr << "Error opening PCM device " << PCM_DEVICE << std::endl;return 1;}// Set parameters: 44100 Hz, stereo, 16-bit little-endiansnd_pcm_set_params(handle,SND_PCM_FORMAT_S16_LE,SND_PCM_ACCESS_RW_INTERLEAVED,2,           // Channels (stereo)44100,       // Sample rate0,           // Soft resampling disabled500000);     // Latency in microseconds// Buffer for audio datastd::vector<char> buffer(4096);// Read and play audio datawhile (file.read(buffer.data(), buffer.size())) {int num_samples = buffer.size() / 2;  // Assuming 16-bit samples (2 bytes per sample)int err = snd_pcm_writei(handle, buffer.data(), num_samples);if (err < 0) {std::cerr << "Playback error: " << snd_strerror(err) << std::endl;break;}}// Close PCM handlesnd_pcm_close(handle);return 0;
}

这样我们就完成了向设备中写音频这件事,但是这个音频爆炸吵的同时,也全是乱码,显然这并不是我们需要的结果。这是因为我们播放的是mp3音频,是经过了一系列压缩算法的文件,并不是可以直接用于播放的有实际意义的音频信息。在我们正式播放音频之前,需要使用mpg123库对音频进行解码,得到pcm音频流然后再一点点将这个流写入到音频设备内。

4. 使用mpg123解码

如果需要使用mpg123解码mp3文件,首先用sudo apt-get install mpg123 将mpg123安装到本地之后,需要在CMakeLists.txt下引入这个mpg123库,为了方便理解,我这里还是将最小限度的mpg123的引入方式

cmake_minimum_required(VERSION 3.0)
project(MyAudioPlayer)# 设置C++标准
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD_REQUIRED ON)# 查找ALSA库
find_package(ALSA REQUIRED)
include_directories(${ALSA_INCLUDE_DIRS})# 查找mpg123库
find_package(PkgConfig REQUIRED)
pkg_search_module(MPG123 REQUIRED mpg123)
include_directories(${MPG123_INCLUDE_DIRS})# 添加可执行文件
add_executable(audio_player main.cpp)# 链接ALSA和mpg123库
target_link_libraries(audio_player ${ALSA_LIBRARIES} ${MPG123_LIBRARIES})

写出一段带解码和播放的代码,如下:

#include <iostream>
#include <vector>
#include <alsa/asoundlib.h>
#include <mpg123.h>#define PCM_DEVICE "hw:3"  // 目标音频设备
#define MP3_FILE_PATH "/home/orangepi/workhsop/testProject/123.mp3"  // 目标音频int main() {const char *mp3File = MP3_FILE_PATH;// 初始化mpg123库mpg123_handle *mh;mpg123_init();mh = mpg123_new(NULL, NULL);// 打开mp3文件if (mpg123_open(mh, mp3File) != MPG123_OK) {std::cerr << "Error opening file: " << mp3File << std::endl;return 1;}// 获取格式信息int channels, encoding;long rate;mpg123_getformat(mh, &rate, &channels, &encoding);// 初始化ALSAsnd_pcm_t *handle;if (snd_pcm_open(&handle, PCM_DEVICE, SND_PCM_STREAM_PLAYBACK, 0) < 0) {std::cerr << "Error opening PCM device " << PCM_DEVICE << std::endl;return 1;}// 设置声卡信息snd_pcm_set_params(handle,SND_PCM_FORMAT_S16_LE,SND_PCM_ACCESS_RW_INTERLEAVED,channels,rate,0,500000);// 缓冲区std::vector<short> buffer(4096);// 读取并播放音频size_t bytesRead;//使用mpg123进行音频信息解码,获得pcm音频信号while (mpg123_read(mh, reinterpret_cast<unsigned char*>(buffer.data()), buffer.size() * sizeof(short), &bytesRead) == MPG123_OK) {int num_samples = bytesRead / sizeof(short);int err = snd_pcm_writei(handle, buffer.data(), num_samples);if (err < 0) {std::cerr << "Playback error: " << snd_strerror(err) << std::endl;break;}}// Close mpg123 handlempg123_close(mh);mpg123_delete(mh);mpg123_exit();// Close PCM handlesnd_pcm_close(handle);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/38598.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深入浅出:npm常用命令详解与实践【保姆级教程】

大家好,我是CodeQi! 在我刚开始学习前端开发的时候,有一件事情让我特别头疼:管理和安装各种各样的依赖包。 那时候,我还不知道 npm 的存在,手动下载和管理这些库简直是噩梦。 后来,我终于接触到了 npm(Node Package Manager),它不仅帮我解决了依赖管理问题,还让我…

Python深度理解系列之【排序算法——冒泡排序】

读者大大们好呀&#xff01;&#xff01;!☀️☀️☀️ &#x1f440;期待大大的关注哦❗️❗️❗️ &#x1f680;欢迎收看我的主页文章➡️木道寻的主页 文章目录 &#x1f525;前言&#x1f680;冒泡排序python实现算法实现图形化算法展示 ⭐️⭐️⭐️总结 &#x1f525;前…

Apache POI、EasyPoi、EasyExcel

目录 ​编辑 &#xff08;一&#xff09;Apache PoI 使用 &#xff08;二&#xff09;EasyPoi使用 &#xff08;三&#xff09;EasyExcel使用 写 读 最简单的读​ 最简单的读的excel示例​ 最简单的读的对象​ &#xff08;一&#xff09;Apache PoI 使用 &#xff08;二&…

golang go-bindata打包配置文件嵌入到二进制文件

go-bindata打包配置文件嵌入到二进制文件 项目中难免会用到一些静态资源和配置文件&#xff0c;但是常规打包的二进制文件无法再其他目录正常运行&#xff08;静态资源和配置文件不存在&#xff09; 有类似需求的可以安装使用&#xff1a;go-bindata进行编译处理配置文件 go-bi…

train_encoder_decoder.py

train_encoder_decoder.py from __future__ import print_function #为了确保代码同时兼容Python 2和Python 3版本中的print函数# 导入标准库和第三方库 import os.path #导入了Python的os.path模块&#xff0c;用于处理文件和目录路径 from os import path #从os模块中导入了…

【场景题】数据库优化和接口优化——异步思想

理解 异步处理&#xff1a; 对于耗时的操作&#xff0c;可以考虑使用异步处理方式来提升接口的响应速度。用户可以在不阻塞当前操作的情况下&#xff0c;等待异步操作的结果。 异步处理在数据库优化中的应用 虽然数据库操作本身&#xff08;如查询、插入、更新等&#xff09…

Git 安装

目录 Git 安装 Git 安装 在使用 Git 前我们需要先安装 Git。Git 目前支持 Linux/Unix、Solaris、Mac 和 Windows 平台上运行。Git 各平台安装包下载地址为&#xff1a;http://git-scm.com/downloads 在 Linux 平台上安装&#xff08;包管理工具安装&#xff09; 首先&#xff0…

IIS在Windows上的搭建

&#x1f4d1;打牌 &#xff1a; da pai ge的个人主页 &#x1f324;️个人专栏 &#xff1a; da pai ge的博客专栏 ☁️宝剑锋从磨砺出&#xff0c;梅花香自苦寒来 目录 一 概念&#xff1a; 二网络…

深入理解C++中的锁

目录 1.基本互斥锁&#xff08;std::mutex&#xff09; 2.递归互斥锁&#xff08;std::recursive_mutex&#xff09; 3.带超时机制的互斥锁&#xff08;std::timed_mutex&#xff09; 4.带超时机制的递归互斥锁&#xff08;std::recursive_timed_mutex&#xff09; 5.共享…

【python脚本】批量检测sql延时注入

文章目录 前言批量检测sql延时注入工作原理脚本演示 前言 SQL延时注入是一种在Web应用程序中利用SQL注入漏洞的技术&#xff0c;当传统的基于错误信息或数据回显的注入方法不可行时&#xff0c;例如当Web应用进行了安全配置&#xff0c;不显示任何错误信息或敏感数据时&#x…

【TS】TypeScript 原始数据类型深度解析

&#x1f308;个人主页: 鑫宝Code &#x1f525;热门专栏: 闲话杂谈&#xff5c; 炫酷HTML | JavaScript基础 ​&#x1f4ab;个人格言: "如无必要&#xff0c;勿增实体" 文章目录 TypeScript 原始数据类型深度解析一、引言二、基础原始数据类型2.1 boolean2.2 …

苍穹外卖--sky-take-out(四)10-12

苍穹外卖--sky-take-out&#xff08;一&#xff09; 苍穹外卖--sky-take-out&#xff08;一&#xff09;-CSDN博客​编辑https://blog.csdn.net/kussm_/article/details/138614737?spm1001.2014.3001.5501https://blog.csdn.net/kussm_/article/details/138614737?spm1001.2…

Unity动画系统(2)

6.1 动画系统基础2-3_哔哩哔哩_bilibili p316 模型添加Animator组件 动画控制器 AnimatorController AnimatorController 可以通过代码控制动画速度 建立动画间的联系 bool值的设定 trigger p318 trigger点击的时候触发&#xff0c;如喊叫&#xff0c;开枪及换子弹等&#x…

在js中如何Json字符串格式不对,如何处理

如果 JSON 字符串格式不正确&#xff0c;解析它时会抛出异常&#xff0c;但我们可以尝试尽可能提取有效的信息。以下是一个方法&#xff0c;可以使用正则表达式和字符串操作来提取部分有效的 JSON 内容&#xff0c;即使整个字符串无法被 JSON.parse 完全解析。 示例代码如下&a…

错误 [WinError 10013] 以一种访问权限不允许的方式做了一个访问套接字的尝试 python ping

报错提示&#xff1a;错误 [WinError 10013] 以一种访问权限不允许的方式做了一个访问套接字的尝试 用python做了一个批量ping脚本&#xff0c;在windows专业版上没问题&#xff0c;但是到了windows服务器就出现这个报错 解决方法&#xff1a;右键 管理员身份运行 这个脚本 …

sql拉链表

1、定义&#xff1a;维护历史状态以及最新数据的一种表 2、使用场景 1、有一些表的数据量很大&#xff0c;比如一张用户表&#xff0c;大约1亿条记录&#xff0c;50个字段&#xff0c;这种表 2.表中的部分字段会被update更新操作&#xff0c;如用户联系方式&#xff0c;产品的…

compute和computeIfAbsent的区别和用法

compute和computeIfAbsent都是Map接口中的默认方法&#xff0c;用于在映射中进行键值对的计算和更新。它们的主要区别在于它们的行为和使用场景。 compute 方法 定义: V compute(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction);参数: k…

在 WebGPU 与 Vulkan 之间做出正确的选择(Making the Right Choice between WebGPU vs Vulkan)

在 WebGPU 与 Vulkan 之间做出正确的选择&#xff08;Making the Right Choice between WebGPU vs Vulkan&#xff09; WebGPU 和 Vulkan 之间的主要区别WebGPU 是什么&#xff1f;它适合谁使用&#xff1f;Vulkan 是什么&#xff1f;它适合谁使用&#xff1f;WebGPU 和 Vulkan…

修改CentOS7 yum源

修改CentOS默认yum源为阿里镜像源 备份系统自带yum源配置文件 mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup 下载ailiyun的yum源配置文件 CentOS7 yum源如下&#xff1a; wget -O /etc/yum.repos.d/CentOS-Base.repo http://mirrors.aliyun…

AI领域最需要掌握的技术是什么?

在AI领域&#xff0c;掌握一系列核心技术和相关知识是非常重要的&#xff0c;以下是AI专业人士最需要掌握的一些关键技术&#xff1a; 1. **数学基础** - 线性代数&#xff1a;用于处理向量和矩阵&#xff0c;是机器学习和深度学习的基石。 - 微积分&#xff1a;用于理解函数的…