// // audioQueuePlayer.m // live // // Created by lujunjie on 2016/11/4. // Copyright © 2016年 lujunjie. All rights reserved. // #import "AudioQueuePlayer.h" #import <AudioToolbox/AudioToolbox.h> #define QUEUE_BUFFER_SIZE 10 //隊列緩衝個數 #define MIN_SIZE_PER_FRAME 2000 //每幀最小數據長度 @interface AudioQueuePlayer() { NSLock *synlock ;//同步控制 AudioQueueRef audioQueue;//音頻播放隊列 BOOL audioQueueUsed[QUEUE_BUFFER_SIZE]; //音頻緩存是否在使用中 AudioStreamBasicDescription audioDescription;//音頻參數 AudioQueueBufferRef audioQueueBuffers[QUEUE_BUFFER_SIZE];//音頻緩衝 } @end @implementation AudioQueuePlayer - (instancetype)init { if (self=[super init]) { [self reset]; } return self; } - (void)reset { [self stop]; synlock = [[NSLock alloc] init]; ///設置音頻參數 audioDescription.mSampleRate = 8000; //採樣率 audioDescription.mFormatID = kAudioFormatLinearPCM; audioDescription.mFormatFlags = kLinearPCMFormatFlagIsSignedInteger | kAudioFormatFlagIsPacked; audioDescription.mChannelsPerFrame = 1; ///單聲道 audioDescription.mFramesPerPacket = 1; //每個packet一偵數據 audioDescription.mBitsPerChannel = 16; //每一個採樣點16bit量化 audioDescription.mBytesPerFrame = (audioDescription.mBitsPerChannel / 8) * audioDescription.mChannelsPerFrame; audioDescription.mBytesPerPacket = audioDescription.mBytesPerFrame; AudioQueueNewOutput(&audioDescription, audioPlayerAQInputCallback, (__bridge void*)self, nil, nil, 0, &audioQueue); //使用player的內部線程播放 //初始化音頻緩衝區 for (int i = 0; i < QUEUE_BUFFER_SIZE; i++) { AudioQueueAllocateBuffer(audioQueue, MIN_SIZE_PER_FRAME, &audioQueueBuffers[i]); } } - (void)stop { if (audioQueue) { AudioQueueStop(audioQueue, true); AudioQueueReset(audioQueue); audioQueue = nil; } } - (void)play:(void*)pcmData length:(unsigned int)length { if (audioQueue == nil || ![self checkBufferHasUsed]) { // 第一次使用 [self reset]; AudioQueueStart(audioQueue, NULL); } [synlock lock]; AudioQueueBufferRef audioQueueBuffer = NULL; while (true) { audioQueueBuffer = [self getNotUsedBuffer]; if (audioQueueBuffer != NULL) { break; } } audioQueueBuffer->mAudioDataByteSize = length; Byte* audiodata = (Byte*)audioQueueBuffer->mAudioData; for (int i = 0; i < length; i++) { audiodata[i] = ((Byte*)pcmData)[i]; } AudioQueueEnqueueBuffer(audioQueue, audioQueueBuffer, 0, NULL); [synlock unlock]; } static void audioPlayerAQInputCallback(void *input, AudioQueueRef audioQueue, AudioQueueBufferRef audioQueueBuffers) { AudioQueuePlayer *player = (__bridge AudioQueuePlayer*)input; [player playerCallback:audioQueueBuffers]; } // 是否是有緩衝在使用中 - (BOOL)checkBufferHasUsed { for (int i = 0; i < QUEUE_BUFFER_SIZE; i++) { if (YES == audioQueueUsed[i]) { return YES; } } return NO; } // 獲取沒有在使用的緩衝 - (AudioQueueBufferRef)getNotUsedBuffer { for (int i = 0; i < QUEUE_BUFFER_SIZE; i++) { if (NO == audioQueueUsed[i]) { audioQueueUsed[i] = YES; return audioQueueBuffers[i]; } } return NULL; } // 標誌緩衝空閒中 - (void)playerCallback:(AudioQueueBufferRef)outQB { for (int i = 0; i < QUEUE_BUFFER_SIZE; i++) { if (outQB == audioQueueBuffers[i]) { audioQueueUsed[i] = NO; } } } @end