如何為Audio Unit 設置特效

設置音頻特效使用的是AudioEffectUnit,我們這里實現(xiàn)的是Reverb(混響)特效。生活中表現(xiàn)的場景就是在不同的空間下有不同的音效。

本篇文章分為以下4個部分:

  1. 使用ExtAudioFile從文件中讀取音頻數(shù)據(jù)。
  2. 將數(shù)據(jù)傳遞給AudioEffectUnit處理。
  3. 使用AudioOutputUnit進行播放。
  4. 設置reverbUnit的屬性。

使用ExtAudioFile讀取文件

ExtAudioFile可以按照我們設置的數(shù)據(jù)格式讀取文件,很方便,具體參照這篇文章。
ExtAudioFile如何使用

AudioEffectUnit處理數(shù)據(jù)

數(shù)據(jù)流向圖


effect.png

創(chuàng)建AudioUnit

混響效果在iOS上是kAudioUnitSubType_Reverb2,在mac上是kAudioUnitSubType_MatrixReverb

- (void)createAudioUnits {
    AudioComponentDescription ioDesc = {0};
    ioDesc.componentType = kAudioUnitType_Output;
    ioDesc.componentSubType = kAudioUnitSubType_VoiceProcessingIO;
    ioDesc.componentManufacturer = kAudioUnitManufacturer_Apple;
    
    AudioComponentDescription reverbDesc = {0};
    reverbDesc.componentType = kAudioUnitType_Effect;
    reverbDesc.componentSubType = kAudioUnitSubType_Reverb2;
    reverbDesc.componentManufacturer = kAudioUnitManufacturer_Apple;
    
    OSStatus status;
    AudioComponent outputComp = AudioComponentFindNext(NULL, &ioDesc);
    if (outputComp == NULL) {
        printf("can't get AudioComponent");
    }
    status = AudioComponentInstanceNew(outputComp, &_ioUnit);
    CheckError(status, "creat output unit");
    
    AudioComponent reverbComp = AudioComponentFindNext(NULL, &reverbDesc);
    if (reverbComp == NULL) {
        printf("can't get AudioComponent");
    }
    status = AudioComponentInstanceNew(reverbComp, &_reverbUnit);
    CheckError(status, "creat reverb unit");
}

設置AudioUnit屬性

AudioOutputUnit的輸入和AudioEffectUnit輸出連接起來。AudioEffectUnit的輸入就是它的callback。

- (void)setupAudioUnits {
    OSStatus status;
    
    // Set the callback method
    AURenderCallbackStruct callbackStruct;
    callbackStruct.inputProc = InputRenderCallback;
    callbackStruct.inputProcRefCon = (__bridge void * _Nullable)(self);
    status = AudioUnitSetProperty(_reverbUnit,
                                  kAudioUnitProperty_SetRenderCallback,
                                  kAudioUnitScope_Input,
                                  0,
                                  &callbackStruct,
                                  sizeof(callbackStruct));
    CheckError(status, "set callback");
    
    //make connection
    AudioUnitConnection connection;
    connection.sourceAudioUnit    = _reverbUnit;
    connection.sourceOutputNumber = 0;
    connection.destInputNumber    = 0;

    status = AudioUnitSetProperty(_ioUnit,                             // connection destination
                                  kAudioUnitProperty_MakeConnection,   // property key
                                  kAudioUnitScope_Input,               // destination scope
                                  0,                           // destination element
                                  &connection,                 // connection definition
                                  sizeof(connection));
    CheckError(status, "make connection");
}

AudioUnit回調(diào)中填充數(shù)據(jù)

static OSStatus InputRenderCallback(void *inRefCon,
                                    AudioUnitRenderActionFlags *ioActionFlags,
                                    const AudioTimeStamp *inTimeStamp,
                                    UInt32 inBusNumber,
                                    UInt32 inNumberFrames,
                                    AudioBufferList *ioData) {
    ZFAudioUnitEffectPlayer *player = (__bridge ZFAudioUnitEffectPlayer *)inRefCon;

    [player.dataSource readDataToBuffer:ioData length:inNumberFrames];
    
    return noErr;
}

使用AudioOutputUnit進行播放

播放和暫停。

- (void)startPlay {
    dispatch_async(_queue, ^{
        OSStatus status;
        status = AudioUnitInitialize(self.reverbUnit);
        CheckError(status, "initialize reverb unit");
        status = AudioUnitInitialize(self.ioUnit);
        CheckError(status, "initialize output unit");
        status = AudioOutputUnitStart(self.ioUnit);
        CheckError(status, "start output unit");
    });
}
- (void)stopPlay {
    dispatch_async(_queue, ^{
        OSStatus status;
        status = AudioOutputUnitStop(self.ioUnit);
        CheckError(status, "stop output unit");
        status = AudioUnitUninitialize(self.ioUnit);
        CheckError(status, "uninitialize output unit");
        status = AudioUnitUninitialize(self.reverbUnit);
        CheckError(status, "uninitialize reverb unit");
    });
}

設置reverbUnit的屬性

reverbUnit有7個屬性可以設置,都在這里了。不太懂音律,大家可以運行demo自己嘗試一下不同的效果。

- (void)setDryWetMix:(Float32)dryWetMix {
    AudioUnitSetParameter(_reverbUnit, kReverb2Param_DryWetMix, kAudioUnitScope_Global, 0, dryWetMix, 0);
}
- (void)setGain:(Float32)gain {
    AudioUnitSetParameter(_reverbUnit, kReverb2Param_Gain, kAudioUnitScope_Global, 0, gain, 0);
}
- (void)setMinDelayTime:(Float32)minDelayTime {
    AudioUnitSetParameter(_reverbUnit, kReverb2Param_MinDelayTime, kAudioUnitScope_Global, 0, minDelayTime, 0);
}
- (void)setMaxDelayTime:(Float32)maxDelayTime {
    AudioUnitSetParameter(_reverbUnit, kReverb2Param_MaxDelayTime, kAudioUnitScope_Global, 0, maxDelayTime, 0);
}
- (void)setDecayTimeAt0Hz:(Float32)decayTimeAt0Hz {
    AudioUnitSetParameter(_reverbUnit, kReverb2Param_DecayTimeAt0Hz, kAudioUnitScope_Global, 0, decayTimeAt0Hz, 0);
}
- (void)setDecayTimeAtNyquist:(Float32)decayTimeAtNyquist {
    AudioUnitSetParameter(_reverbUnit, kReverb2Param_DecayTimeAtNyquist, kAudioUnitScope_Global, 0, decayTimeAtNyquist, 0);
}
- (void)setRandomizeReflections:(Float32)randomizeReflections {
    AudioUnitSetParameter(_reverbUnit, kReverb2Param_RandomizeReflections, kAudioUnitScope_Global, 0, randomizeReflections, 0);
}

Github地址

?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內(nèi)容

  • 在iOS平臺上,所有的音頻框架底層都是基于AudioUnit實現(xiàn)的。較高層次的音頻框架包括:MediaPlayer...
    阿凡提說AI閱讀 4,066評論 3 7
  • 前言: AudioUnit是什么?在IOS平臺下,AudioUnit是一個底層音頻處理框架,主要功能如下:1、音頻...
    仙人掌__閱讀 4,082評論 6 5
  • AudioUnit簡介 AudioUnit這個名字取得還是比較形象的,它的主體就是一系列的unit,不同unit能...
    泥孩兒0107閱讀 4,669評論 0 2
  • 久違的晴天,家長會。 家長大會開好到教室時,離放學已經(jīng)沒多少時間了。班主任說已經(jīng)安排了三個家長分享經(jīng)驗。 放學鈴聲...
    飄雪兒5閱讀 7,822評論 16 22
  • 創(chuàng)業(yè)是很多人的夢想,多少人為了理想和不甘選擇了創(chuàng)業(yè)來實現(xiàn)自我價值,我就是其中一個。 創(chuàng)業(yè)后,我由女人變成了超人,什...
    亦寶寶閱讀 2,007評論 4 1

友情鏈接更多精彩內(nèi)容