/* * 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 "AAudioServiceStreamShared" //#define LOG_NDEBUG 0 #include <utils/Log.h> #include <iomanip> #include <iostream> #include <mutex> #include <aaudio/AAudio.h> #include "binding/IAAudioService.h" #include "binding/AAudioServiceMessage.h" #include "AAudioServiceStreamBase.h" #include "AAudioServiceStreamShared.h" #include "AAudioEndpointManager.h" #include "AAudioService.h" #include "AAudioServiceEndpoint.h" using namespace android; using namespace aaudio; #define MIN_BURSTS_PER_BUFFER 2 #define DEFAULT_BURSTS_PER_BUFFER 16 // This is an arbitrary range. TODO review. #define MAX_FRAMES_PER_BUFFER (32 * 1024) AAudioServiceStreamShared::AAudioServiceStreamShared(AAudioService &audioService) : AAudioServiceStreamBase(audioService) , mTimestampPositionOffset(0) , mXRunCount(0) { } std::string AAudioServiceStreamShared::dumpHeader() { std::stringstream result; result << AAudioServiceStreamBase::dumpHeader(); result << " Write# Read# Avail XRuns"; return result.str(); } std::string AAudioServiceStreamShared::dump() const { std::stringstream result; result << AAudioServiceStreamBase::dump(); auto fifo = mAudioDataQueue->getFifoBuffer(); int32_t readCounter = fifo->getReadCounter(); int32_t writeCounter = fifo->getWriteCounter(); result << std::setw(10) << writeCounter; result << std::setw(10) << readCounter; result << std::setw(8) << (writeCounter - readCounter); result << std::setw(8) << getXRunCount(); return result.str(); } int32_t AAudioServiceStreamShared::calculateBufferCapacity(int32_t requestedCapacityFrames, int32_t framesPerBurst) { if (requestedCapacityFrames > MAX_FRAMES_PER_BUFFER) { ALOGE("calculateBufferCapacity() requested capacity %d > max %d", requestedCapacityFrames, MAX_FRAMES_PER_BUFFER); return AAUDIO_ERROR_OUT_OF_RANGE; } // Determine how many bursts will fit in the buffer. int32_t numBursts; if (requestedCapacityFrames == AAUDIO_UNSPECIFIED) { // Use fewer bursts if default is too many. if ((DEFAULT_BURSTS_PER_BUFFER * framesPerBurst) > MAX_FRAMES_PER_BUFFER) { numBursts = MAX_FRAMES_PER_BUFFER / framesPerBurst; } else { numBursts = DEFAULT_BURSTS_PER_BUFFER; } } else { // round up to nearest burst boundary numBursts = (requestedCapacityFrames + framesPerBurst - 1) / framesPerBurst; } // Clip to bare minimum. if (numBursts < MIN_BURSTS_PER_BUFFER) { numBursts = MIN_BURSTS_PER_BUFFER; } // Check for numeric overflow. if (numBursts > 0x8000 || framesPerBurst > 0x8000) { ALOGE("calculateBufferCapacity() overflow, capacity = %d * %d", numBursts, framesPerBurst); return AAUDIO_ERROR_OUT_OF_RANGE; } int32_t capacityInFrames = numBursts * framesPerBurst; // Final sanity check. if (capacityInFrames > MAX_FRAMES_PER_BUFFER) { ALOGE("calculateBufferCapacity() calc capacity %d > max %d", capacityInFrames, MAX_FRAMES_PER_BUFFER); return AAUDIO_ERROR_OUT_OF_RANGE; } ALOGD("calculateBufferCapacity() requested %d frames, actual = %d", requestedCapacityFrames, capacityInFrames); return capacityInFrames; } aaudio_result_t AAudioServiceStreamShared::open(const aaudio::AAudioStreamRequest &request) { sp<AAudioServiceStreamShared> keep(this); aaudio_result_t result = AAudioServiceStreamBase::open(request, AAUDIO_SHARING_MODE_SHARED); if (result != AAUDIO_OK) { ALOGE("%s() returned %d", __func__, result); return result; } const AAudioStreamConfiguration &configurationInput = request.getConstantConfiguration(); sp<AAudioServiceEndpoint> endpoint = mServiceEndpointWeak.promote(); if (endpoint == nullptr) { result = AAUDIO_ERROR_INVALID_STATE; goto error; } // Is the request compatible with the shared endpoint? setFormat(configurationInput.getFormat()); if (getFormat() == AAUDIO_FORMAT_UNSPECIFIED) { setFormat(AAUDIO_FORMAT_PCM_FLOAT); } else if (getFormat() != AAUDIO_FORMAT_PCM_FLOAT) { ALOGD("%s() mAudioFormat = %d, need FLOAT", __func__, getFormat()); result = AAUDIO_ERROR_INVALID_FORMAT; goto error; } setSampleRate(configurationInput.getSampleRate()); if (getSampleRate() == AAUDIO_UNSPECIFIED) { setSampleRate(endpoint->getSampleRate()); } else if (getSampleRate() != endpoint->getSampleRate()) { ALOGD("%s() mSampleRate = %d, need %d", __func__, getSampleRate(), endpoint->getSampleRate()); result = AAUDIO_ERROR_INVALID_RATE; goto error; } setSamplesPerFrame(configurationInput.getSamplesPerFrame()); if (getSamplesPerFrame() == AAUDIO_UNSPECIFIED) { setSamplesPerFrame(endpoint->getSamplesPerFrame()); } else if (getSamplesPerFrame() != endpoint->getSamplesPerFrame()) { ALOGD("%s() mSamplesPerFrame = %d, need %d", __func__, getSamplesPerFrame(), endpoint->getSamplesPerFrame()); result = AAUDIO_ERROR_OUT_OF_RANGE; goto error; } setBufferCapacity(calculateBufferCapacity(configurationInput.getBufferCapacity(), mFramesPerBurst)); if (getBufferCapacity() < 0) { result = getBufferCapacity(); // negative error code setBufferCapacity(0); goto error; } { std::lock_guard<std::mutex> lock(mAudioDataQueueLock); // Create audio data shared memory buffer for client. mAudioDataQueue = new SharedRingBuffer(); result = mAudioDataQueue->allocate(calculateBytesPerFrame(), getBufferCapacity()); if (result != AAUDIO_OK) { ALOGE("%s() could not allocate FIFO with %d frames", __func__, getBufferCapacity()); result = AAUDIO_ERROR_NO_MEMORY; goto error; } } ALOGD("AAudioServiceStreamShared::open() actual rate = %d, channels = %d, deviceId = %d", getSampleRate(), getSamplesPerFrame(), endpoint->getDeviceId()); result = endpoint->registerStream(keep); if (result != AAUDIO_OK) { goto error; } setState(AAUDIO_STREAM_STATE_OPEN); return AAUDIO_OK; error: close(); return result; } aaudio_result_t AAudioServiceStreamShared::close() { aaudio_result_t result = AAudioServiceStreamBase::close(); { std::lock_guard<std::mutex> lock(mAudioDataQueueLock); delete mAudioDataQueue; mAudioDataQueue = nullptr; } return result; } /** * Get an immutable description of the data queue created by this service. */ aaudio_result_t AAudioServiceStreamShared::getAudioDataDescription( AudioEndpointParcelable &parcelable) { std::lock_guard<std::mutex> lock(mAudioDataQueueLock); if (mAudioDataQueue == nullptr) { ALOGE("%s(): mUpMessageQueue null! - stream not open", __func__); return AAUDIO_ERROR_NULL; } // Gather information on the data queue. mAudioDataQueue->fillParcelable(parcelable, parcelable.mDownDataQueueParcelable); parcelable.mDownDataQueueParcelable.setFramesPerBurst(getFramesPerBurst()); return AAUDIO_OK; } void AAudioServiceStreamShared::markTransferTime(Timestamp ×tamp) { mAtomicTimestamp.write(timestamp); } // Get timestamp that was written by mixer or distributor. aaudio_result_t AAudioServiceStreamShared::getFreeRunningPosition(int64_t *positionFrames, int64_t *timeNanos) { // TODO Get presentation timestamp from the HAL if (mAtomicTimestamp.isValid()) { Timestamp timestamp = mAtomicTimestamp.read(); *positionFrames = timestamp.getPosition(); *timeNanos = timestamp.getNanoseconds(); return AAUDIO_OK; } else { return AAUDIO_ERROR_UNAVAILABLE; } } // Get timestamp from lower level service. aaudio_result_t AAudioServiceStreamShared::getHardwareTimestamp(int64_t *positionFrames, int64_t *timeNanos) { int64_t position = 0; sp<AAudioServiceEndpoint> endpoint = mServiceEndpointWeak.promote(); if (endpoint == nullptr) { ALOGE("%s() has no endpoint", __func__); return AAUDIO_ERROR_INVALID_STATE; } aaudio_result_t result = endpoint->getTimestamp(&position, timeNanos); if (result == AAUDIO_OK) { int64_t offset = mTimestampPositionOffset.load(); // TODO, do not go below starting value position -= offset; // Offset from shared MMAP stream ALOGV("%s() %8lld = %8lld - %8lld", __func__, (long long) position, (long long) (position + offset), (long long) offset); } *positionFrames = position; return result; }