/* * Copyright 2016 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. */ #include #include "aaudio/AAudioExtensions.h" #include "aaudio/AudioStreamAAudio.h" #include "FilterAudioStream.h" #include "OboeDebug.h" #include "oboe/Oboe.h" #include "oboe/AudioStreamBuilder.h" #include "opensles/AudioInputStreamOpenSLES.h" #include "opensles/AudioOutputStreamOpenSLES.h" #include "opensles/AudioStreamOpenSLES.h" #include "QuirksManager.h" bool oboe::OboeGlobals::mWorkaroundsEnabled = true; namespace oboe { /** * The following default values are used when oboe does not have any better way of determining the optimal values * for an audio stream. This can happen when: * * - Client is creating a stream on API < 26 (OpenSLES) but has not supplied the optimal sample * rate and/or frames per burst * - Client is creating a stream on API 16 (OpenSLES) where AudioManager.PROPERTY_OUTPUT_* values * are not available */ int32_t DefaultStreamValues::SampleRate = 48000; // Common rate for mobile audio and video int32_t DefaultStreamValues::FramesPerBurst = 192; // 4 msec at 48000 Hz int32_t DefaultStreamValues::ChannelCount = 2; // Stereo constexpr int kBufferSizeInBurstsForLowLatencyStreams = 2; #ifndef OBOE_ENABLE_AAUDIO // Set OBOE_ENABLE_AAUDIO to 0 if you want to disable the AAudio API. // This might be useful if you want to force all the unit tests to use OpenSL ES. #define OBOE_ENABLE_AAUDIO 1 #endif bool AudioStreamBuilder::isAAudioSupported() { return AudioStreamAAudio::isSupported() && OBOE_ENABLE_AAUDIO; } bool AudioStreamBuilder::isAAudioRecommended() { // See https://github.com/google/oboe/issues/40, // AAudio may not be stable on Android O, depending on how it is used. // To be safe, use AAudio only on O_MR1 and above. return (getSdkVersion() >= __ANDROID_API_O_MR1__) && isAAudioSupported(); } AudioStream *AudioStreamBuilder::build() { AudioStream *stream = nullptr; if (isAAudioRecommended() && mAudioApi != AudioApi::OpenSLES) { stream = new AudioStreamAAudio(*this); } else if (isAAudioSupported() && mAudioApi == AudioApi::AAudio) { stream = new AudioStreamAAudio(*this); LOGE("Creating AAudio stream on 8.0 because it was specified. This is error prone."); } else { if (getDirection() == oboe::Direction::Output) { stream = new AudioOutputStreamOpenSLES(*this); } else if (getDirection() == oboe::Direction::Input) { stream = new AudioInputStreamOpenSLES(*this); } } return stream; } bool AudioStreamBuilder::isCompatible(AudioStreamBase &other) { return (getSampleRate() == oboe::Unspecified || getSampleRate() == other.getSampleRate()) && (getFormat() == (AudioFormat)oboe::Unspecified || getFormat() == other.getFormat()) && (getFramesPerDataCallback() == oboe::Unspecified || getFramesPerDataCallback() == other.getFramesPerDataCallback()) && (getChannelCount() == oboe::Unspecified || getChannelCount() == other.getChannelCount()); } Result AudioStreamBuilder::openStream(AudioStream **streamPP) { auto result = isValidConfig(); if (result != Result::OK) { return result; } LOGI("%s() %s -------- %s --------", __func__, getDirection() == Direction::Input ? "INPUT" : "OUTPUT", getVersionText()); if (streamPP == nullptr) { return Result::ErrorNull; } *streamPP = nullptr; AudioStream *streamP = nullptr; // Maybe make a FilterInputStream. AudioStreamBuilder childBuilder(*this); // Check need for conversion and modify childBuilder for optimal stream. bool conversionNeeded = QuirksManager::getInstance().isConversionNeeded(*this, childBuilder); // Do we need to make a child stream and convert. if (conversionNeeded) { AudioStream *tempStream; result = childBuilder.openStream(&tempStream); if (result != Result::OK) { return result; } if (isCompatible(*tempStream)) { // The child stream would work as the requested stream so we can just use it directly. *streamPP = tempStream; return result; } else { AudioStreamBuilder parentBuilder = *this; // Build a stream that is as close as possible to the childStream. if (getFormat() == oboe::AudioFormat::Unspecified) { parentBuilder.setFormat(tempStream->getFormat()); } if (getChannelCount() == oboe::Unspecified) { parentBuilder.setChannelCount(tempStream->getChannelCount()); } if (getSampleRate() == oboe::Unspecified) { parentBuilder.setSampleRate(tempStream->getSampleRate()); } if (getFramesPerDataCallback() == oboe::Unspecified) { parentBuilder.setFramesPerCallback(tempStream->getFramesPerDataCallback()); } // Use childStream in a FilterAudioStream. LOGI("%s() create a FilterAudioStream for data conversion.", __func__); FilterAudioStream *filterStream = new FilterAudioStream(parentBuilder, tempStream); result = filterStream->configureFlowGraph(); if (result != Result::OK) { filterStream->close(); delete filterStream; // Just open streamP the old way. } else { streamP = static_cast(filterStream); } } } if (streamP == nullptr) { streamP = build(); if (streamP == nullptr) { return Result::ErrorNull; } } // If MMAP has a problem in this case then disable it temporarily. bool wasMMapOriginallyEnabled = AAudioExtensions::getInstance().isMMapEnabled(); bool wasMMapTemporarilyDisabled = false; if (wasMMapOriginallyEnabled) { bool isMMapSafe = QuirksManager::getInstance().isMMapSafe(childBuilder); if (!isMMapSafe) { AAudioExtensions::getInstance().setMMapEnabled(false); wasMMapTemporarilyDisabled = true; } } result = streamP->open(); if (wasMMapTemporarilyDisabled) { AAudioExtensions::getInstance().setMMapEnabled(wasMMapOriginallyEnabled); // restore original } if (result == Result::OK) { int32_t optimalBufferSize = -1; // Use a reasonable default buffer size. if (streamP->getDirection() == Direction::Input) { // For input, small size does not improve latency because the stream is usually // run close to empty. And a low size can result in XRuns so always use the maximum. optimalBufferSize = streamP->getBufferCapacityInFrames(); } else if (streamP->getPerformanceMode() == PerformanceMode::LowLatency && streamP->getDirection() == Direction::Output) { // Output check is redundant. optimalBufferSize = streamP->getFramesPerBurst() * kBufferSizeInBurstsForLowLatencyStreams; } if (optimalBufferSize >= 0) { auto setBufferResult = streamP->setBufferSizeInFrames(optimalBufferSize); if (!setBufferResult) { LOGW("Failed to setBufferSizeInFrames(%d). Error was %s", optimalBufferSize, convertToText(setBufferResult.error())); } } *streamPP = streamP; } else { delete streamP; } return result; } Result AudioStreamBuilder::openManagedStream(oboe::ManagedStream &stream) { stream.reset(); auto result = isValidConfig(); if (result != Result::OK) { return result; } AudioStream *streamptr; result = openStream(&streamptr); stream.reset(streamptr); return result; } Result AudioStreamBuilder::openStream(std::shared_ptr &sharedStream) { sharedStream.reset(); auto result = isValidConfig(); if (result != Result::OK) { return result; } AudioStream *streamptr; result = openStream(&streamptr); if (result == Result::OK) { sharedStream.reset(streamptr); // Save a weak_ptr in the stream for use with callbacks. streamptr->setWeakThis(sharedStream); } return result; } } // namespace oboe