音頻隊列-音頻播放

//
//  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
相關文章
相關標籤/搜索