blob: d15d2faf864586d5fc4833ffaff13fb48684f7e6 [file] [log] [blame]
/*
* 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.
*/
#define LOG_TAG "AudioEndpointParcelable"
//#define LOG_NDEBUG 0
#include <utils/Log.h>
#include <map>
#include <stdint.h>
#include <binder/Parcel.h>
#include <binder/Parcelable.h>
#include <utility/AAudioUtilities.h>
#include "binding/AAudioServiceDefinitions.h"
#include "binding/RingBufferParcelable.h"
#include "binding/AudioEndpointParcelable.h"
using android::base::unique_fd;
using android::status_t;
using namespace aaudio;
AudioEndpointParcelable::AudioEndpointParcelable(Endpoint&& parcelable)
: mUpMessageQueueParcelable(parcelable.upMessageQueueParcelable),
mDownMessageQueueParcelable(parcelable.downMessageQueueParcelable),
mUpDataQueueParcelable(parcelable.upDataQueueParcelable),
mDownDataQueueParcelable(parcelable.downDataQueueParcelable) {
for (size_t i = 0; i < parcelable.sharedMemories.size() && i < MAX_SHARED_MEMORIES; ++i) {
// Re-construct.
mSharedMemories[i].~SharedMemoryParcelable();
new(&mSharedMemories[i]) SharedMemoryParcelable(std::move(parcelable.sharedMemories[i]));
}
}
AudioEndpointParcelable& AudioEndpointParcelable::operator=(Endpoint&& parcelable) {
this->~AudioEndpointParcelable();
new(this) AudioEndpointParcelable(std::move(parcelable));
return *this;
}
namespace {
void updateSharedMemoryIndex(SharedRegion* sharedRegion, int oldIndex, int newIndex) {
if (sharedRegion->sharedMemoryIndex == oldIndex) {
sharedRegion->sharedMemoryIndex = newIndex;
}
}
void updateSharedMemoryIndex(RingBuffer* ringBuffer, int oldIndex, int newIndex) {
updateSharedMemoryIndex(&ringBuffer->readCounterParcelable, oldIndex, newIndex);
updateSharedMemoryIndex(&ringBuffer->writeCounterParcelable, oldIndex, newIndex);
updateSharedMemoryIndex(&ringBuffer->dataParcelable, oldIndex, newIndex);
}
void updateSharedMemoryIndex(Endpoint* endpoint, int oldIndex, int newIndex) {
updateSharedMemoryIndex(&endpoint->upMessageQueueParcelable, oldIndex, newIndex);
updateSharedMemoryIndex(&endpoint->downMessageQueueParcelable, oldIndex, newIndex);
updateSharedMemoryIndex(&endpoint->upDataQueueParcelable, oldIndex, newIndex);
updateSharedMemoryIndex(&endpoint->downDataQueueParcelable, oldIndex, newIndex);
}
} // namespace
Endpoint AudioEndpointParcelable::parcelable()&& {
Endpoint result;
result.upMessageQueueParcelable = mUpMessageQueueParcelable.parcelable();
result.downMessageQueueParcelable = mDownMessageQueueParcelable.parcelable();
result.upDataQueueParcelable = mUpDataQueueParcelable.parcelable();
result.downDataQueueParcelable = mDownDataQueueParcelable.parcelable();
// To transfer through binder, only valid/in-use shared memory is allowed. By design, the
// shared memories that are currently in-use may not be placed continuously from position 0.
// However, when marshalling the shared memories into Endpoint, the shared memories will be
// re-indexed from 0. In that case, when placing a shared memory, it is needed to update the
// corresponding cached indexes.
for (int i = 0; i < MAX_SHARED_MEMORIES; ++i) {
if (mSharedMemories[i].isInUse()) {
const int index = result.sharedMemories.size();
result.sharedMemories.emplace_back(std::move(mSharedMemories[i]).parcelable());
// Updating all the SharedRegion that is using `i` as shared memory index with the
// new shared memory index as `result.sharedMemories.size() - 1`.
updateSharedMemoryIndex(&result, i, index);
}
}
return result;
}
/**
* Add the file descriptor to the table.
* @return index in table or negative error
*/
int32_t AudioEndpointParcelable::addFileDescriptor(const unique_fd& fd,
int32_t sizeInBytes) {
const int32_t index = getNextAvailableSharedMemoryPosition();
if (index < 0) {
return AAUDIO_ERROR_OUT_OF_RANGE;
}
mSharedMemories[index].setup(fd, sizeInBytes);
return index;
}
void AudioEndpointParcelable::closeDataFileDescriptor() {
for (const int32_t memoryIndex : std::set{mDownDataQueueParcelable.getDataSharedMemoryIndex(),
mDownDataQueueParcelable.getReadCounterSharedMemoryIndex(),
mDownDataQueueParcelable.getWriteCounterSharedMemoryIndex()}) {
mSharedMemories[memoryIndex].closeAndReleaseFd();
}
}
aaudio_result_t AudioEndpointParcelable::updateDataFileDescriptor(
AudioEndpointParcelable* endpointParcelable) {
// Before updating data file descriptor, close the old shared memories.
closeDataFileDescriptor();
// The given endpoint parcelable and this one are two different objects, the indexes in
// `mSharedMemories` for `mDownDataQueueParcelable` can be different. In that case, an index
// map, which maps from the index in given endpoint parcelable to the index in this endpoint
// parcelable, is required when updating shared memory.
std::map<int32_t, int32_t> memoryIndexMap;
auto& downDataQueueParcelable = endpointParcelable->mDownDataQueueParcelable;
for (const int32_t memoryIndex : {downDataQueueParcelable.getDataSharedMemoryIndex(),
downDataQueueParcelable.getReadCounterSharedMemoryIndex(),
downDataQueueParcelable.getWriteCounterSharedMemoryIndex()}) {
if (memoryIndexMap.find(memoryIndex) != memoryIndexMap.end()) {
// This shared memory has been set up in this endpoint parcelable.
continue;
}
// Set up the memory in the next available shared memory position.
const int index = getNextAvailableSharedMemoryPosition();
if (index < 0) {
return AAUDIO_ERROR_OUT_OF_RANGE;
}
mSharedMemories[index].setup(endpointParcelable->mSharedMemories[memoryIndex]);
memoryIndexMap.emplace(memoryIndex, index);
}
mDownDataQueueParcelable.updateMemory(
endpointParcelable->mDownDataQueueParcelable, memoryIndexMap);
return AAUDIO_OK;
}
aaudio_result_t AudioEndpointParcelable::resolve(EndpointDescriptor *descriptor) {
aaudio_result_t result = mUpMessageQueueParcelable.resolve(mSharedMemories,
&descriptor->upMessageQueueDescriptor);
if (result != AAUDIO_OK) return result;
result = mDownMessageQueueParcelable.resolve(mSharedMemories,
&descriptor->downMessageQueueDescriptor);
if (result != AAUDIO_OK) return result;
result = mDownDataQueueParcelable.resolve(mSharedMemories,
&descriptor->dataQueueDescriptor);
return result;
}
aaudio_result_t AudioEndpointParcelable::resolveDataQueue(RingBufferDescriptor *descriptor) {
return mDownDataQueueParcelable.resolve(mSharedMemories, descriptor);
}
aaudio_result_t AudioEndpointParcelable::close() {
int err = 0;
for (auto& sharedMemory : mSharedMemories) {
const int lastErr = sharedMemory.close();
if (lastErr < 0) err = lastErr;
}
return AAudioConvert_androidToAAudioResult(err);
}
int32_t AudioEndpointParcelable::getNextAvailableSharedMemoryPosition() const {
for (int i = 0; i < MAX_SHARED_MEMORIES; ++i) {
if (!mSharedMemories[i].isInUse()) {
return i;
}
}
return -1;
}
void AudioEndpointParcelable::dump() {
ALOGD("======================================= BEGIN");
for (int i = 0; i < MAX_SHARED_MEMORIES; ++i) {
if (mSharedMemories[i].isInUse()) {
ALOGD("Shared memory index=%d", i);
mSharedMemories[i].dump();
}
}
ALOGD("mUpMessageQueueParcelable =========");
mUpMessageQueueParcelable.dump();
ALOGD("mDownMessageQueueParcelable =======");
mDownMessageQueueParcelable.dump();
ALOGD("mUpDataQueueParcelable ============");
mUpDataQueueParcelable.dump();
ALOGD("mDownDataQueueParcelable ==========");
mDownDataQueueParcelable.dump();
ALOGD("======================================= END");
}