参考文档

COM Coding Practices

Audio File Format Specifications

Core Audio APIs

Loopback Recording

#include <iostream>
#include <fstream>
#include <vector> #include <mmdeviceapi.h>
#include <combaseapi.h>
#include <atlbase.h>
#include <Functiondiscoverykeys_devpkey.h>
#include <Audioclient.h>
#include <Audiopolicy.h> // 利用RAII手法,自动调用 CoUninitialize
class CoInitializeGuard {
public:
CoInitializeGuard()
{
_hr = CoInitializeEx(nullptr, COINIT::COINIT_MULTITHREADED);
} ~CoInitializeGuard()
{
if (_hr == S_OK || _hr == S_FALSE) {
CoUninitialize();
}
} HRESULT result() const { return _hr; } private:
HRESULT _hr;
}; constexpr inline void exit_on_failed(HRESULT hr);
void printEndpoints(CComPtr<IMMDeviceCollection> pColletion);
std::string wchars_to_mbs(const wchar_t* s); int main()
{
HRESULT hr{}; CoInitializeGuard coInitializeGuard;
exit_on_failed(coInitializeGuard.result()); // COM 对象都用 CComPtr 包装,会自动调用 Release
// COM 接口分配的堆变量用 CComHeapPtr 包装,会自动调用 CoTaskMemFree
CComPtr<IMMDeviceEnumerator> pEnumerator;
hr = pEnumerator.CoCreateInstance(__uuidof(MMDeviceEnumerator));
exit_on_failed(hr); // 打印所有可用的音频设备
//CComPtr<IMMDeviceCollection> pColletion;
//hr = pEnumerator->EnumAudioEndpoints(eRender, DEVICE_STATE_ACTIVE, &pColletion);
//exit_on_failed(hr);
//printEndpoints(pColletion); // 使用默认的 Audio Endpoint,eRender 表示音频播放设备,而不是录音设备
CComPtr<IMMDevice> pEndpoint;
hr = pEnumerator->GetDefaultAudioEndpoint(eRender, eConsole, &pEndpoint);
exit_on_failed(hr); // 打印出播放设备的名字,可能包含中文
CComPtr<IPropertyStore> pProps;
hr = pEndpoint->OpenPropertyStore(STGM_READ, &pProps);
exit_on_failed(hr);
PROPVARIANT varName;
PropVariantInit(&varName);
hr = pProps->GetValue(PKEY_Device_FriendlyName, &varName);
exit_on_failed(hr);
std::cout << "select audio endpoint: " << wchars_to_mbs(varName.pwszVal) << std::endl;
PropVariantClear(&varName); // 由 IMMDevice 对象 得到 IAudioClient 对象
CComPtr<IAudioClient> pAudioClient;
hr = pEndpoint->Activate(__uuidof(IAudioClient), CLSCTX_ALL, nullptr, (void**)&pAudioClient);
exit_on_failed(hr); // 获得音频播放设备格式信息
CComHeapPtr<WAVEFORMATEX> pDeviceFormat;
pAudioClient->GetMixFormat(&pDeviceFormat); constexpr int REFTIMES_PER_SEC = 10000000; // 1 reference_time = 100ns
constexpr int REFTIMES_PER_MILLISEC = 10000; // 初始化 IAudioClient 对象
const REFERENCE_TIME hnsRequestedDuration = 2 * REFTIMES_PER_SEC; // 1s
hr = pAudioClient->Initialize(AUDCLNT_SHAREMODE_SHARED, AUDCLNT_STREAMFLAGS_LOOPBACK, hnsRequestedDuration, 0, pDeviceFormat, nullptr);
exit_on_failed(hr); // 获得缓冲区大小
UINT32 bufferFrameCount{};
hr = pAudioClient->GetBufferSize(&bufferFrameCount);
exit_on_failed(hr); // 由 IAudioClient 对象 得到 IAudioCaptureClient 对象,也就是将音频播放设备视为录音设备
CComPtr<IAudioCaptureClient> pCaptureClient;
hr = pAudioClient->GetService(__uuidof(IAudioCaptureClient), (void**)&pCaptureClient);
exit_on_failed(hr); // 开始录音
hr = pAudioClient->Start();
exit_on_failed(hr); const REFERENCE_TIME hnsActualDuration = (long long)REFTIMES_PER_SEC * bufferFrameCount / pDeviceFormat->nSamplesPerSec; std::ofstream ofile("./out.wav", std::ios::binary);
if (!ofile) {
exit(-1);
} // 写入各种 header 信息 constexpr UINT32 sizePlaceholder{};
// master RIFF chunk
ofile.write("RIFF", 4);
ofile.write((const char*)&sizePlaceholder, 4);
ofile.write("WAVE", 4);
// 12 // fmt chunk
ofile.write("fmt ", 4);
UINT32 fmt_ckSize = sizeof(WAVEFORMATEX) + pDeviceFormat->cbSize;
ofile.write((const char*)&fmt_ckSize, 4);
{
auto p = pDeviceFormat.Detach();
ofile.write((const char*)p, fmt_ckSize);
pDeviceFormat.Attach(p);
}
// 8 + fmt_ckSize // fact chunk
bool has_fact_chunt = pDeviceFormat->wFormatTag != WAVE_FORMAT_PCM;
if (has_fact_chunt) {
ofile.write("fact", 4);
UINT32 fact_ckSize = 4;
ofile.write((const char*)&fact_ckSize, 4);
DWORD dwSampleLength{};
ofile.write((const char*)&dwSampleLength, 4);
}
// 12 // data chunk
ofile.write("data", 4);
ofile.write((const char*)&sizePlaceholder, 4); UINT32 data_ckSize = 0; // samples data 的大小
UINT32 frame_count = 0; // 帧数 constexpr int max_duration = 60; // 录制 60s
int seconds{}; // 已经录制的时间 time_t t_begin = time(NULL); //UINT32
do {
// 睡眠一定时间,防止CPU占用率高
Sleep(9); BYTE* pData{}; // samples 数据
UINT32 numFramesAvailable{}; // 缓冲区有多少帧
DWORD dwFlags{}; hr = pCaptureClient->GetBuffer(&pData, &numFramesAvailable, &dwFlags, NULL, NULL);
exit_on_failed(hr); int frame_bytes = pDeviceFormat->nChannels * pDeviceFormat->wBitsPerSample / 8;
int count = numFramesAvailable * frame_bytes;
ofile.write((const char*)pData, count);
data_ckSize += count;
frame_count += numFramesAvailable;
seconds = frame_count / pDeviceFormat->nSamplesPerSec;
std::cout << "numFramesAvailable: " << numFramesAvailable << " seconds: " << seconds << std::endl; hr = pCaptureClient->ReleaseBuffer(numFramesAvailable);
exit_on_failed(hr); } while (seconds < max_duration); // 检测实际花了多久,实际时间 - max_duration = 延迟
time_t t_end = time(NULL);
std::cout << "use wall clock: " << t_end - t_begin << "s" << std::endl; if (data_ckSize % 2) {
ofile.put(0);
++data_ckSize;
} UINT32 wave_ckSize = 4 + (8 + fmt_ckSize) + (8 + data_ckSize);
ofile.seekp(4);
ofile.write((const char*)&wave_ckSize, 4); if (has_fact_chunt) {
ofile.seekp(12 + (8 + fmt_ckSize) + 8);
ofile.write((const char*)&frame_count, 4);
} ofile.seekp(12 + (8 + fmt_ckSize) + 12 + 4);
ofile.write((const char*)&data_ckSize, 4); ofile.close(); //所有 COM 对象和 Heap 都会自动释放
} void printEndpoints(CComPtr<IMMDeviceCollection> pColletion)
{
HRESULT hr{}; UINT count{};
hr = pColletion->GetCount(&count);
exit_on_failed(hr); for (UINT i = 0; i < count; ++i) {
CComPtr<IMMDevice> pEndpoint;
hr = pColletion->Item(i, &pEndpoint);
exit_on_failed(hr); CComHeapPtr<WCHAR> pwszID;
hr = pEndpoint->GetId(&pwszID);
exit_on_failed(hr); CComPtr<IPropertyStore> pProps;
hr = pEndpoint->OpenPropertyStore(STGM_READ, &pProps);
exit_on_failed(hr); PROPVARIANT varName;
PropVariantInit(&varName);
hr = pProps->GetValue(PKEY_Device_FriendlyName, &varName);
exit_on_failed(hr); std::cout << wchars_to_mbs(varName.pwszVal) << std::endl; PropVariantClear(&varName);
}
} constexpr inline void exit_on_failed(HRESULT hr) {
if (FAILED(hr)) {
exit(-1);
}
} // 汉字会有编码问题,全部转成窄字符
std::string wchars_to_mbs(const wchar_t* src)
{
UINT cp = GetACP();
int ccWideChar = (int)wcslen(src);
int n = WideCharToMultiByte(cp, 0, src, ccWideChar, 0, 0, 0, 0); std::vector<char> buf(n);
WideCharToMultiByte(cp, 0, src, ccWideChar, buf.data(), (int)buf.size(), 0, 0);
std::string dst(buf.data(), buf.size());
return dst;
}

使用 Windows Core Audio APs 进行 Loopback Recording 并生成 WAV 文件的更多相关文章

  1. windows core audio apis

    这个播放流程有一次当初不是很理解,做个记录,代码中的中文部分,原文档是有解释的:To move a stream of rendering data through the endpoint buff ...

  2. windows&lunix下node.js实现模板化生成word文件

    最近在做了一个小程序!里面有个功能就是根据用户提交的数据,自动生成一份word文档返回给用户.我也是第一次做这功能,大概思路就是先自己弄一份word模板,后台接受小程序发过来的数据,再根据这些数据将相 ...

  3. Core Audio(一)

    Core Audio APIs core audio apis是vista之后引入的,不使用与之前的windows版本:core audio apis提供访问endpoint devices,比如耳机 ...

  4. Core Audio 在Vista/Win7上实现

    应用范围:Vista / win7, 不支持XP 1. 关于Windows Core Auido APIs 在Windowss Vista及Windows 7操作系统下,微软为应用程序提供了一套新的音 ...

  5. Core Audio(二)

    用户模式音频组件 在windows vista中,core audio apis充当用户模式音频子系统的基础,core audio apis作为用户模式系统组件的一个thin layer,它用来将用户 ...

  6. Windows 7上安装Microsoft Loopback Adapter(微软环回网卡)

    Oracle 安装过程中,先决条件检查遇到如下错误: 正在检查网络配置要求...  检查完成.此次检查的总体结果为: 失败 <<<<  问题: 安装检测到系统的主 IP 地址是 ...

  7. 使用Core Audio实现VoIP通用音频模块

    最近一直在做iOS音频技术相关的项目,由于单项直播SDK,互动直播SDK(iOS/Mac),短视频SDK,都会用到音频技术,因此在这里收集三个SDK的音频技术需求,开发一个通用的音频模块用于三个SDK ...

  8. 重新想象 Windows 8 Store Apps (24) - 文件系统: Application Data 中的文件操作, Package 中的文件操作, 可移动存储中的文件操作

    原文:重新想象 Windows 8 Store Apps (24) - 文件系统: Application Data 中的文件操作, Package 中的文件操作, 可移动存储中的文件操作 [源码下载 ...

  9. windows下Android利用ant自动编译、修改配置文件、批量多渠道,打包生成apk文件

    原创文章,转载请注明:http://www.cnblogs.com/ycxyyzw/p/4535459.html android 程序打包成apk,如果在是命令行方式,一般都要经过如下步骤: 1.用a ...

随机推荐

  1. mui 登录跳转到首页之后顶部选项卡不灵敏问题

    前段时间开发一个用mui开发app的时候遇到了登录跳转到首页之后顶部选项卡会失灵的问题,多次尝试之后终于解决了,趁现在还有点印象记录一下吧. 一开始我是用mui.openWindow来新开首页的,出了 ...

  2. 使用 Java 操作 Redis

    Jedis 1. 概述 Jedis 是一款使用 Java 操作 Redis 的工具,有点类似于 JDBC 2. 引入依赖 <dependency> <groupId>redis ...

  3. 开发H5程序或者小程序的时候,后端Web API项目在IISExpress调试中使用IP地址,便于开发调试

    在我们开发开发H5程序或者小程序的时候,有时候需要基于内置浏览器或者微信开发者工具进行测试,这个时候可以采用默认的localhost进行访问后端接口,一般来说没什么问题,如果我们需要通过USB基座方式 ...

  4. C++大数据的读写

    当一个文件1G以上的这种,使用内存文件映射会提高读写效率: 下边时段出自<windows核心编程>,读取一个大文件,然后统计里边字符出现次数的函数: __int64 CountOs(voi ...

  5. 【LOJ#3197】【eJOI2019】T形覆盖 - (图论、简单推导)

    题面 题解 (题目中说的四种摆放方式实际上是分别旋转0°,90°,180°,270°后的图形) 题目中关于摆放方式的描述听起来很臭,我们把它转换一下,每个拼版先覆盖"上下左右中"五 ...

  6. 自定义View3-水波纹扩散(仿支付宝咻一咻)实现代码、思想

    PS:自定义view篇-水波纹实现 效果:水波纹扩散 场景:雷达.按钮点击效果.搜索等 实现:先上效果图,之前记得支付宝有一个咻一咻,当时就是水波纹效果,实现起来一共两步,第一画内圆,第二画多个外圆, ...

  7. Linux虚拟机启动报错挂载点丢失

    fstab 挂载失败 实验准备 1) 准备:vim /etc/fstab /mnt1/cdrom 挂载点不在 2) 系统启动报错截图 修复步骤 /etc/fstab 中的错误和损坏的文件系统可能会阻止 ...

  8. Linux虚拟机快捷键大全

    转发请注明原作者! 图形化命令框快捷键 Ctrl-Shift-t 创建标签页 Ctrl-Shift-w 关闭标签页 Ctrl-Shift-n 创建新窗口 Ctrl-Shift-q 关闭新窗口 Ctrl ...

  9. KingbaseES TOAST存储方式

    KingbaseES为"大字段"的物理存储提供了TOAST功能,通过合适的配置策略能够减少IO次数和扫描块数,进而提升查询速度. TOAST:The Oversized-Attri ...

  10. immutable 与 stable 函数的差异

    Stable 函数不能修改数据库,单个Query中所有行给定同样的参数确保返回相同的结果.这种稳定级别允许优化器将多次函数调用转换为一次.在索引扫描的条件中使用这种函数是可行的,因为索引扫描只计算一次 ...