codecamp

使用OpenSL ES开发音频录制功能

OpenSL ES全称为Open Sound Library for Embedded Systems,是一个嵌入式、跨平台、免费的音频处理库。为嵌入式移动多媒体设备上的应用开发者提供标准化、高性能、低延迟的API。HarmonyOS的Native API基于Khronos Group开发的OpenSL ES 1.0.1 API 规范实现,开发者可以通过<OpenSLES.h>和<OpenSLES_OpenHarmony.h>在HarmonyOS上使用相关API。

HarmonyOS上的OpenSL ES

OpenSL ES中提供了以下的接口,HarmonyOS当前仅实现了部分OpenSL ES接口,可以实现音频录制的基础功能。

调用未实现接口后会返回SL_RESULT_FEATURE_UNSUPPORTED,当前没有相关扩展可以使用。

以下列表列举了HarmonyOS上已实现的OpenSL ES的接口,具体说明请参考OpenSL ES规范:

  • HarmonyOS上支持的Engine接口
    • SLresult (*CreateAudioPlayer) (SLEngineItf self, SLObjectItf * pPlayer, SLDataSource *pAudioSrc, SLDataSink *pAudioSnk, SLuint32 numInterfaces, const SLInterfaceID * pInterfaceIds, const SLboolean * pInterfaceRequired)
    • SLresult (*CreateAudioRecorder) (SLEngineItf self, SLObjectItf * pRecorder, SLDataSource *pAudioSrc, SLDataSink *pAudioSnk, SLuint32 numInterfaces, const SLInterfaceID * pInterfaceIds, const SLboolean * pInterfaceRequired)
    • SLresult (*CreateOutputMix) (SLEngineItf self, SLObjectItf * pMix, SLuint32 numInterfaces, const SLInterfaceID * pInterfaceIds, const SLboolean * pInterfaceRequired)
  • HarmonyOS上支持的Object接口:
    • SLresult (*Realize) (SLObjectItf self, SLboolean async)
    • SLresult (*GetState) (SLObjectItf self, SLuint32 * pState)
    • SLresult (*GetInterface) (SLObjectItf self, const SLInterfaceID iid, void * pInterface)
    • void (*Destroy) (SLObjectItf self)
  • HarmonyOS上支持的Recorder接口:
    • SLresult (*SetRecordState) (SLRecordItf self, SLuint32 state)
    • SLresult (*GetRecordState) (SLRecordItf self,SLuint32 *pState)
  • HarmonyOS上支持的BufferQueue接口
    以下接口需引入<OpenSLES_OpenHarmony.h>使用。

    接口

    说明

    SLresult (*Enqueue) (SLOHBufferQueueItf self, const void *buffer, SLuint32 size)

    根据情况将buffer加到相应队列中。

    如果是播放操作,则将带有音频数据的buffer插入到filledBufferQ_队列中;如果是录音操作,则将录音使用后的空闲buffer插入到freeBufferQ_队列中。

    self:表示调用该函数的BufferQueue接口对象。

    buffer:播放时表示带有音频数据的buffer,录音时表示已存储完录音数据后的空闲buffer。

    size:表示buffer的大小。

    SLresult (*Clear) (SLOHBufferQueueItf self)

    释放BufferQueue接口对象。

    self:表示调用该函数的BufferQueue接口对象将被释放。

    SLresult (*GetState) (SLOHBufferQueueItf self, SLOHBufferQueueState *state)

    获取BufferQueue接口对象状态。

    self:表示调用该函数的BufferQueue接口对象。

    state:BufferQueue的当前状态。

    SLresult (*RegisterCallback) (SLOHBufferQueueItf self, SlOHBufferQueueCallback callback, void* pContext)

    注册回调函数。

    self:表示调用该函数的BufferQueue接口对象。

    callback:播放/录音时注册的回调函数。

    pContext:播放时传入待播放音频文件,录音时传入将要录制的音频文件。

    SLresult (*GetBuffer) (SLOHBufferQueueItf self, SLuint8** buffer, SLuint32* size)

    根据情况获取相应的buffer。

    如果是播放操作,则从freeBufferQ_队列中获取空闲buffer;如果是录音操作,则从filledBufferQ_队列中获取携带录音数据的buffer。

    self:表示调用该函数的BufferQueue接口对象。

    buffer:播放时表示空闲的buffer,录音时表示携带录音数据的buffer。

    size:表示buffer的大小。

完整示例

参考下列示例代码,完成音频录制。

  1. 添加头文件
    1. #include <OpenSLES.h>
    2. #include <OpenSLES_OpenHarmony.h>
    3. #include <OpenSLES_Platform.h>
  2. 使用slCreateEngine接口创建引擎对象和实例化引擎对象engine。
    1. SLObjectItf engineObject = nullptr;
    2. slCreateEngine(&engineObject, 0, nullptr, 0, nullptr, nullptr);
    3. (*engineObject)->Realize(engineObject, SL_BOOLEAN_FALSE);
  3. 获取接口SL_IID_ENGINE的引擎接口engineEngine实例。
    1. SLEngineItf engineItf = nullptr;
    2. (*engineObject)->GetInterface(engineObject, SL_IID_ENGINE, &engineItf);
  4. 配置录音器信息(配置输入源audiosource、输出源audiosink),创建录音对象pcmCapturerObject。
    1. SLDataLocator_IODevice io_device = {
    2. SL_DATALOCATOR_IODEVICE,
    3. SL_IODEVICE_AUDIOINPUT,
    4. SL_DEFAULTDEVICEID_AUDIOINPUT,
    5. NULL
    6. };
    7. SLDataSource audioSource = {
    8. &io_device,
    9. NULL
    10. };
    11. SLDataLocator_BufferQueue buffer_queue = {
    12. SL_DATALOCATOR_BUFFERQUEUE,
    13. 3
    14. };
    15. // 具体参数需要根据音频文件格式进行适配
    16. SLDataFormat_PCM format_pcm = {
    17. SL_DATAFORMAT_PCM, // 输入的音频格式
    18. 1, // 单声道
    19. SL_SAMPLINGRATE_44_1, // 采样率: 44100HZ
    20. SL_PCMSAMPLEFORMAT_FIXED_16, // 音频采样格式,小尾数,带符号的16位整数
    21. 0,
    22. 0,
    23. 0
    24. };
    25. SLDataSink audioSink = {
    26. &buffer_queue,
    27. &format_pcm
    28. };
    29. SLObjectItf pcmCapturerObject = nullptr;
    30. (*engineItf)->CreateAudioRecorder(engineItf, &pcmCapturerObject,
    31. &audioSource, &audioSink, 0, nullptr, nullptr);
    32. (*pcmCapturerObject)->Realize(pcmCapturerObject, SL_BOOLEAN_FALSE);
  5. 获取录音接口SL_IID_RECORD的recordItf接口实例。
    1. SLRecordItf recordItf;
    2. (*pcmCapturerObject)->GetInterface(pcmCapturerObject, SL_IID_RECORD, &recordItf);
  6. 获取接口 SL_IID_OH_BUFFERQUEUE 的 bufferQueueItf 实例
    1. SLOHBufferQueueItf bufferQueueItf;
    2. (*pcmCapturerObject)->GetInterface(pcmCapturerObject, SL_IID_OH_BUFFERQUEUE, &bufferQueueItf);
  7. 注册BufferQueueCallback回调。
    1. static void BufferQueueCallback(SLOHBufferQueueItf bufferQueueItf, void *pContext, SLuint32 size)
    2. {
    3. // 可从pContext获取注册时传入的使用者信息
    4. SLuint8 *buffer = nullptr;
    5. SLuint32 pSize = 0;
    6. (*bufferQueueItf)->GetBuffer(bufferQueueItf, &buffer, &pSize);
    7. if (buffer != nullptr) {
    8. // 可从buffer内读取录音数据进行后续处理
    9. (*bufferQueueItf)->Enqueue(bufferQueueItf, buffer, size);
    10. }
    11. }
    12. void *pContext; // 可传入自定义的上下文信息,会在Callback内收到
    13. (*bufferQueueItf)->RegisterCallback(bufferQueueItf, BufferQueueCallback, pContext);
  8. 开始录音。
    1. (*recordItf)->SetRecordState(recordItf, SL_RECORDSTATE_RECORDING);
  9. 结束音频录制。
    1. (*recordItf)->SetRecordState(recordItf, SL_RECORDSTATE_STOPPED);
    2. (*pcmCapturerObject)->Destroy(pcmCapturerObject);
使用AudioCapturer开发音频录制功能
管理麦克风
温馨提示
下载编程狮App,免费阅读超1000+编程语言教程
取消
确定
目录
HAR

关闭

MIP.setData({ 'pageTheme' : getCookie('pageTheme') || {'day':true, 'night':false}, 'pageFontSize' : getCookie('pageFontSize') || 20 }); MIP.watch('pageTheme', function(newValue){ setCookie('pageTheme', JSON.stringify(newValue)) }); MIP.watch('pageFontSize', function(newValue){ setCookie('pageFontSize', newValue) }); function setCookie(name, value){ var days = 1; var exp = new Date(); exp.setTime(exp.getTime() + days*24*60*60*1000); document.cookie = name + '=' + value + ';expires=' + exp.toUTCString(); } function getCookie(name){ var reg = new RegExp('(^| )' + name + '=([^;]*)(;|$)'); return document.cookie.match(reg) ? JSON.parse(document.cookie.match(reg)[2]) : null; }