MAC OS 錄音工具類

原文地址 : https://blog.lm1024.club/archives/2020121928

public class LYAudioRecord {
    private let KInputBus: AudioUnitElement = 1
    private let KOutputBus: AudioUnitElement = 0
    
    private let sampleRate: Int = 44100
    private let channel: Int = 1
    
    private var audioSamples: UInt32?
    
    private var isRecording: Bool = false
    
    private var remoteIOUnit: AudioUnit?
    public var mAudioFormat: AudioStreamBasicDescription?
    
    public var delegate: WJAudioSourceOutputDelegate?

    init() {
        configRecorder()
    }

    private func initAudioComponent() {
        var remoteIODesc = AudioComponentDescription(componentType: kAudioUnitType_Output,
                                                     componentSubType: kAudioUnitSubType_HALOutput,
                                                     componentManufacturer: kAudioUnitManufacturer_Apple,
                                                     componentFlags: 0,
                                                     componentFlagsMask: 0)
        
        let remoteIOComponent = AudioComponentFindNext(nil, &remoteIODesc)
        checkError(status: AudioComponentInstanceNew(remoteIOComponent!, &remoteIOUnit), msg: "AudioComponentInstanceNew")
        checkError(status: AudioUnitInitialize(remoteIOUnit!), msg: "AudioUnitInitialize failure")
    }
    
    private func configAudioFormat() {
        var one: UInt32 = 1
        var status = AudioUnitSetProperty(remoteIOUnit!,
                                          kAudioOutputUnitProperty_EnableIO,
                                          kAudioUnitScope_Input,
                                          KInputBus,
                                          &one,
                                          UInt32(MemoryLayout.size(ofValue: one)))
        checkError(status: status, msg: "couldn't kAudioOutputUnitProperty_EnableIO with kAudioUnitScope_Input")
        
        // Attention! set kAudioOutputUnitProperty_EnableIO, kAudioUnitScope_Output, disable
        var disableFlag: UInt32 = 0
        status = AudioUnitSetProperty(remoteIOUnit!,
                                      kAudioOutputUnitProperty_EnableIO,
                                      kAudioUnitScope_Output,
                                      KOutputBus,
                                      &disableFlag,
                                      UInt32(MemoryLayout.size(ofValue: disableFlag)))
        checkError(status: status, msg: "couldn't kAudioOutputUnitProperty_EnableIO with kAudioUnitScope_Output")
        
        var defaultDevice: AudioDeviceID = kAudioDeviceUnknown
        var propertySize = UInt32(MemoryLayout.size(ofValue: defaultDevice))
        var defaultDeviceProperty = AudioObjectPropertyAddress(mSelector: kAudioHardwarePropertyDefaultInputDevice,
                                                               mScope: kAudioObjectPropertyScopeInput,
                                                               mElement: kAudioObjectPropertyElementMaster)
        
        status = AudioObjectGetPropertyData(AudioObjectID(kAudioObjectSystemObject),
                                            &defaultDeviceProperty,
                                            0,
                                            nil,
                                            &propertySize,
                                            &defaultDevice)
        checkError(status: status, msg: "AudioObjectGetPropertyData failure")
        
        var temp = Float64(sampleRate)
        defaultDeviceProperty.mSelector = kAudioDevicePropertyNominalSampleRate
        status = AudioObjectSetPropertyData(defaultDevice, &defaultDeviceProperty, 0, nil, UInt32(MemoryLayout<Float64>.size), &temp)
        checkError(status: status, msg: "AudioObjectSetPropertyData failure")
        
        status = AudioUnitSetProperty(remoteIOUnit!,
                                      kAudioOutputUnitProperty_CurrentDevice,
                                      kAudioUnitScope_Global,
                                      KInputBus,
                                      &defaultDevice,
                                      propertySize)
        
        checkError(status: status, msg: " kAudioOutputUnitProperty_CurrentDevice error")
        
        let mBitsPerChannel: UInt32 = 16
        let mChannelsPerFrame = UInt32(channel)
        let mFramesPerPacket: UInt32 = 1
        let mBytesPerFrame: UInt32 = (mBitsPerChannel / 8) * mChannelsPerFrame
        let mBytesPerPacket: UInt32 = mBytesPerFrame * mFramesPerPacket
        let mAudioFormat = AudioStreamBasicDescription(mSampleRate: Float64(sampleRate),
                                                       mFormatID: kAudioFormatLinearPCM,
                                                       mFormatFlags: kAudioFormatFlagIsSignedInteger | kAudioFormatFlagsNativeEndian | kAudioFormatFlagIsPacked,
                                                       mBytesPerPacket: mBytesPerPacket,
                                                       mFramesPerPacket: mFramesPerPacket,
                                                       mBytesPerFrame: mBytesPerFrame,
                                                       mChannelsPerFrame: mChannelsPerFrame,
                                                       mBitsPerChannel: mBitsPerChannel,
                                                       mReserved: 0)
        self.mAudioFormat = mAudioFormat
        status = AudioUnitSetProperty(remoteIOUnit!,
                                      kAudioUnitProperty_StreamFormat,
                                      kAudioUnitScope_Output,
                                      KInputBus,
                                      &(self.mAudioFormat!),
                                      UInt32(MemoryLayout.size(ofValue: self.mAudioFormat!)))
        
        checkError(status: status, msg: "couldn't set kAudioUnitProperty_StreamFormat with kAudioUnitScope_Output")
        
        var param:UInt32 = UInt32(MemoryLayout<UInt32>.size)
        status =  AudioUnitGetProperty(remoteIOUnit!,
                                       kAudioDevicePropertyBufferFrameSize,
                                       kAudioUnitScope_Global,
                                       0,
                                       &audioSamples,
                                       &param);
        checkError(status: status, msg: "kAudioDevicePropertyBufferFrameSize failure")
        
    }
    
    private func initInputCallBack() {
        var inputCallback = AURenderCallbackStruct()
        
        inputCallback.inputProc = { inRefCon, ioActionFlags, inTimeStamp, inBusNumber, inNumberFrames, ioData in
            
            let record = Unmanaged<WJAudioRecord>.fromOpaque(inRefCon).takeUnretainedValue()
            
            return record.captureCallBack(inRefCon: inRefCon,
                                          ioActionFlags: ioActionFlags,
                                          inTimeStamp: inTimeStamp,
                                          inBusNumber: inBusNumber,
                                          inNumberFrames: inNumberFrames,
                                          ioData: ioData)
        }
        inputCallback.inputProcRefCon = UnsafeMutableRawPointer(Unmanaged.passUnretained(self).toOpaque())
        let statusCode = AudioUnitSetProperty(remoteIOUnit!,
                                              kAudioOutputUnitProperty_SetInputCallback,
                                              kAudioUnitScope_Global,
                                              KInputBus,
                                              &inputCallback,
                                              UInt32(MemoryLayout.size(ofValue: inputCallback)))
        
        checkError(status: statusCode, msg: "Could not set input callback for I/O node \(statusCode)")
    }

    private func captureCallBack(inRefCon: UnsafeMutableRawPointer?,
                                 ioActionFlags: UnsafeMutablePointer<AudioUnitRenderActionFlags>,
                                 inTimeStamp: UnsafePointer<AudioTimeStamp>,
                                 inBusNumber: UInt32,
                                 inNumberFrames: UInt32,
                                 ioData: UnsafeMutablePointer<AudioBufferList>?) -> OSStatus
    {
        guard let _ = inRefCon else {
            return noErr
        }
        
        var buffer = AudioBuffer()
        buffer.mData = nil
        buffer.mDataByteSize = 0
        buffer.mNumberChannels = UInt32(channel)
        
        var bufferList = AudioBufferList()
        bufferList.mNumberBuffers = 1
        bufferList.mBuffers = buffer
        let status = AudioUnitRender(remoteIOUnit!, ioActionFlags, inTimeStamp, inBusNumber, inNumberFrames, &bufferList)
        checkError(status: status, msg: "AudioUnitRender error")
        
        let byteSize: UInt32 = bufferList.mBuffers.mDataByteSize
        
        let dst: UnsafeMutableRawPointer = malloc(Int(byteSize))
        memcpy(dst, bufferList.mBuffers.mData, Int(byteSize))
        guard let audioDelegate = delegate else {
            free(dst)
            return noErr
        }
        audioDelegate.output(data: dst, lenght: byteSize, sampleRate: sampleRate, channel: channel,numSamples: inNumberFrames)
        free(dst)
        return noErr
    }

    private func checkError(status: OSStatus, msg: String) {
        assert(status == noErr, msg)
    }
    
    private func configRecorder() {
        initAudioComponent()
        configAudioFormat()
        initInputCallBack()
    }

    public func startRecord() {
        if remoteIOUnit == nil {
            configRecorder()
        }
        if !isRecording {
            checkError(status: AudioOutputUnitStart(remoteIOUnit!), msg: "AudioOutputUnitStart(_remoteIOUnit) failure")
            isRecording = true
        }
    }

    public func stopRecord() {
        if isRecording {
            checkError(status: AudioOutputUnitStop(remoteIOUnit!), msg: "AudioOutputUnitStop failure")
            isRecording = false
        }
    }
    
    public func getAudioSamples()->UInt32 {
        return self.audioSamples ?? 0
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市滩字,隨后出現(xiàn)的幾起案子粱甫,更是在濱河造成了極大的恐慌光戈,老刑警劉巖讳侨,帶你破解...
    沈念sama閱讀 217,406評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件掺栅,死亡現(xiàn)場(chǎng)離奇詭異陕习,居然都是意外死亡膊升,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門秧荆,熙熙樓的掌柜王于貴愁眉苦臉地迎上來倔毙,“玉大人,你說我怎么就攤上這事乙濒∩略撸” “怎么了?”我有些...
    開封第一講書人閱讀 163,711評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵颁股,是天一觀的道長(zhǎng)凯正。 經(jīng)常有香客問我,道長(zhǎng)豌蟋,這世上最難降的妖魔是什么廊散? 我笑而不...
    開封第一講書人閱讀 58,380評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮梧疲,結(jié)果婚禮上允睹,老公的妹妹穿的比我還像新娘。我一直安慰自己幌氮,他們只是感情好缭受,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,432評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著该互,像睡著了一般米者。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上宇智,一...
    開封第一講書人閱讀 51,301評(píng)論 1 301
  • 那天蔓搞,我揣著相機(jī)與錄音,去河邊找鬼随橘。 笑死喂分,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的机蔗。 我是一名探鬼主播蒲祈,決...
    沈念sama閱讀 40,145評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼甘萧,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了梆掸?” 一聲冷哼從身側(cè)響起扬卷,我...
    開封第一講書人閱讀 39,008評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎酸钦,沒想到半個(gè)月后怪得,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,443評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡钝鸽,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,649評(píng)論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了庞钢。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片拔恰。...
    茶點(diǎn)故事閱讀 39,795評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖基括,靈堂內(nèi)的尸體忽然破棺而出颜懊,到底是詐尸還是另有隱情,我是刑警寧澤风皿,帶...
    沈念sama閱讀 35,501評(píng)論 5 345
  • 正文 年R本政府宣布河爹,位于F島的核電站,受9級(jí)特大地震影響桐款,放射性物質(zhì)發(fā)生泄漏咸这。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,119評(píng)論 3 328
  • 文/蒙蒙 一魔眨、第九天 我趴在偏房一處隱蔽的房頂上張望媳维。 院中可真熱鬧,春花似錦遏暴、人聲如沸侄刽。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)州丹。三九已至,卻和暖如春杂彭,著一層夾襖步出監(jiān)牢的瞬間墓毒,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工亲怠, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蚁鳖,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,899評(píng)論 2 370
  • 正文 我出身青樓赁炎,卻偏偏與公主長(zhǎng)得像醉箕,于是被迫代替她去往敵國(guó)和親钾腺。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,724評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容