/*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#define LOG_TAG "AAudioMixer"
//#define LOG_NDEBUG 0
#include <utils/Log.h>
#define ATRACE_TAG ATRACE_TAG_AUDIO
#include <cstring>
#include <utils/Trace.h>
#include "AAudioMixer.h"
#ifndef AAUDIO_MIXER_ATRACE_ENABLED
#define AAUDIO_MIXER_ATRACE_ENABLED 1
#endif
using android::WrappingBuffer;
using android::FifoBuffer;
using android::fifo_frames_t;
AAudioMixer::~AAudioMixer() {
delete[] mOutputBuffer;
}
void AAudioMixer::allocate(int32_t samplesPerFrame, int32_t framesPerBurst) {
mSamplesPerFrame = samplesPerFrame;
mFramesPerBurst = framesPerBurst;
int32_t samplesPerBuffer = samplesPerFrame * framesPerBurst;
mOutputBuffer = new float[samplesPerBuffer];
mBufferSizeInBytes = samplesPerBuffer * sizeof(float);
}
void AAudioMixer::clear() {
memset(mOutputBuffer, 0, mBufferSizeInBytes);
}
int32_t AAudioMixer::mix(int streamIndex, FifoBuffer *fifo, bool allowUnderflow) {
WrappingBuffer wrappingBuffer;
float *destination = mOutputBuffer;
#if AAUDIO_MIXER_ATRACE_ENABLED
ATRACE_BEGIN("aaMix");
#endif /* AAUDIO_MIXER_ATRACE_ENABLED */
// Gather the data from the client. May be in two parts.
fifo_frames_t fullFrames = fifo->getFullDataAvailable(&wrappingBuffer);
#if AAUDIO_MIXER_ATRACE_ENABLED
if (ATRACE_ENABLED()) {
char rdyText[] = "aaMixRdy#";
char letter = 'A' + (streamIndex % 26);
rdyText[sizeof(rdyText) - 2] = letter;
ATRACE_INT(rdyText, fullFrames);
}
#else /* MIXER_ATRACE_ENABLED */
(void) trackIndex;
#endif /* AAUDIO_MIXER_ATRACE_ENABLED */
// If allowUnderflow then always advance by one burst even if we do not have the data.
// Otherwise the stream timing will drift whenever there is an underflow.
// This actual underflow can then be detected by the client for XRun counting.
//
// Generally, allowUnderflow will be false when stopping a stream and we want to
// use up whatever data is in the queue.
fifo_frames_t framesDesired = mFramesPerBurst;
if (!allowUnderflow && fullFrames < framesDesired) {
framesDesired = fullFrames; // just use what is available then stop
}
// Mix data in one or two parts.
int partIndex = 0;
int32_t framesLeft = framesDesired;
while (framesLeft > 0 && partIndex < WrappingBuffer::SIZE) {
fifo_frames_t framesToMixFromPart = framesLeft;
fifo_frames_t framesAvailableFromPart = wrappingBuffer.numFrames[partIndex];
if (framesAvailableFromPart > 0) {
if (framesToMixFromPart > framesAvailableFromPart) {
framesToMixFromPart = framesAvailableFromPart;
}
mixPart(destination, (float *)wrappingBuffer.data[partIndex],
framesToMixFromPart);
destination += framesToMixFromPart * mSamplesPerFrame;
framesLeft -= framesToMixFromPart;
}
partIndex++;
}
fifo->getFifoControllerBase()->advanceReadIndex(framesDesired);
#if AAUDIO_MIXER_ATRACE_ENABLED
ATRACE_END();
#endif /* AAUDIO_MIXER_ATRACE_ENABLED */
return (framesDesired - framesLeft); // framesRead
}
void AAudioMixer::mixPart(float *destination, float *source, int32_t numFrames) {
int32_t numSamples = numFrames * mSamplesPerFrame;
// TODO maybe optimize using SIMD
for (int sampleIndex = 0; sampleIndex < numSamples; sampleIndex++) {
*destination++ += *source++;
}
}
float *AAudioMixer::getOutputBuffer() {
return mOutputBuffer;
}