| /* |
| * 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 "AAudio" |
| //#define LOG_NDEBUG 0 |
| #include <utils/Log.h> |
| |
| #include <stdint.h> |
| |
| #include <binder/Parcelable.h> |
| |
| #include "binding/AAudioServiceDefinitions.h" |
| #include "binding/SharedRegionParcelable.h" |
| #include "binding/RingBufferParcelable.h" |
| |
| using namespace aaudio; |
| |
| RingBufferParcelable::RingBufferParcelable() {} |
| RingBufferParcelable::~RingBufferParcelable() {} |
| |
| // TODO This assumes that all three use the same SharedMemoryParcelable |
| void RingBufferParcelable::setupMemory(int32_t sharedMemoryIndex, |
| int32_t dataMemoryOffset, |
| int32_t dataSizeInBytes, |
| int32_t readCounterOffset, |
| int32_t writeCounterOffset, |
| int32_t counterSizeBytes) { |
| mReadCounterParcelable.setup(sharedMemoryIndex, readCounterOffset, counterSizeBytes); |
| mWriteCounterParcelable.setup(sharedMemoryIndex, writeCounterOffset, counterSizeBytes); |
| mDataParcelable.setup(sharedMemoryIndex, dataMemoryOffset, dataSizeInBytes); |
| } |
| |
| void RingBufferParcelable::setupMemory(int32_t sharedMemoryIndex, |
| int32_t dataMemoryOffset, |
| int32_t dataSizeInBytes) { |
| mReadCounterParcelable.setup(sharedMemoryIndex, 0, 0); |
| mWriteCounterParcelable.setup(sharedMemoryIndex, 0, 0); |
| mDataParcelable.setup(sharedMemoryIndex, dataMemoryOffset, dataSizeInBytes); |
| } |
| |
| int32_t RingBufferParcelable::getBytesPerFrame() { |
| return mBytesPerFrame; |
| } |
| |
| void RingBufferParcelable::setBytesPerFrame(int32_t bytesPerFrame) { |
| mBytesPerFrame = bytesPerFrame; |
| } |
| |
| int32_t RingBufferParcelable::getFramesPerBurst() { |
| return mFramesPerBurst; |
| } |
| |
| void RingBufferParcelable::setFramesPerBurst(int32_t framesPerBurst) { |
| mFramesPerBurst = framesPerBurst; |
| } |
| |
| int32_t RingBufferParcelable::getCapacityInFrames() { |
| return mCapacityInFrames; |
| } |
| |
| void RingBufferParcelable::setCapacityInFrames(int32_t capacityInFrames) { |
| mCapacityInFrames = capacityInFrames; |
| } |
| |
| /** |
| * The read and write must be symmetric. |
| */ |
| status_t RingBufferParcelable::writeToParcel(Parcel* parcel) const { |
| status_t status = parcel->writeInt32(mCapacityInFrames); |
| if (status != NO_ERROR) goto error; |
| if (mCapacityInFrames > 0) { |
| status = parcel->writeInt32(mBytesPerFrame); |
| if (status != NO_ERROR) goto error; |
| status = parcel->writeInt32(mFramesPerBurst); |
| if (status != NO_ERROR) goto error; |
| status = parcel->writeInt32(mFlags); |
| if (status != NO_ERROR) goto error; |
| status = mReadCounterParcelable.writeToParcel(parcel); |
| if (status != NO_ERROR) goto error; |
| status = mWriteCounterParcelable.writeToParcel(parcel); |
| if (status != NO_ERROR) goto error; |
| status = mDataParcelable.writeToParcel(parcel); |
| if (status != NO_ERROR) goto error; |
| } |
| return NO_ERROR; |
| error: |
| ALOGE("RingBufferParcelable::writeToParcel() error = %d", status); |
| return status; |
| } |
| |
| status_t RingBufferParcelable::readFromParcel(const Parcel* parcel) { |
| status_t status = parcel->readInt32(&mCapacityInFrames); |
| if (status != NO_ERROR) goto error; |
| if (mCapacityInFrames > 0) { |
| status = parcel->readInt32(&mBytesPerFrame); |
| if (status != NO_ERROR) goto error; |
| status = parcel->readInt32(&mFramesPerBurst); |
| if (status != NO_ERROR) goto error; |
| status = parcel->readInt32((int32_t *)&mFlags); |
| if (status != NO_ERROR) goto error; |
| status = mReadCounterParcelable.readFromParcel(parcel); |
| if (status != NO_ERROR) goto error; |
| status = mWriteCounterParcelable.readFromParcel(parcel); |
| if (status != NO_ERROR) goto error; |
| status = mDataParcelable.readFromParcel(parcel); |
| if (status != NO_ERROR) goto error; |
| } |
| return NO_ERROR; |
| error: |
| ALOGE("RingBufferParcelable::readFromParcel() error = %d", status); |
| return status; |
| } |
| |
| aaudio_result_t RingBufferParcelable::resolve(SharedMemoryParcelable *memoryParcels, RingBufferDescriptor *descriptor) { |
| aaudio_result_t result; |
| |
| result = mReadCounterParcelable.resolve(memoryParcels, |
| (void **) &descriptor->readCounterAddress); |
| if (result != AAUDIO_OK) { |
| return result; |
| } |
| |
| result = mWriteCounterParcelable.resolve(memoryParcels, |
| (void **) &descriptor->writeCounterAddress); |
| if (result != AAUDIO_OK) { |
| return result; |
| } |
| |
| result = mDataParcelable.resolve(memoryParcels, (void **) &descriptor->dataAddress); |
| if (result != AAUDIO_OK) { |
| return result; |
| } |
| |
| descriptor->bytesPerFrame = mBytesPerFrame; |
| descriptor->framesPerBurst = mFramesPerBurst; |
| descriptor->capacityInFrames = mCapacityInFrames; |
| descriptor->flags = mFlags; |
| return AAUDIO_OK; |
| } |
| |
| aaudio_result_t RingBufferParcelable::validate() { |
| aaudio_result_t result; |
| if (mCapacityInFrames < 0 || mCapacityInFrames >= 32 * 1024) { |
| ALOGE("RingBufferParcelable invalid mCapacityInFrames = %d", mCapacityInFrames); |
| return AAUDIO_ERROR_INTERNAL; |
| } |
| if (mBytesPerFrame < 0 || mBytesPerFrame >= 256) { |
| ALOGE("RingBufferParcelable invalid mBytesPerFrame = %d", mBytesPerFrame); |
| return AAUDIO_ERROR_INTERNAL; |
| } |
| if (mFramesPerBurst < 0 || mFramesPerBurst >= 16 * 1024) { |
| ALOGE("RingBufferParcelable invalid mFramesPerBurst = %d", mFramesPerBurst); |
| return AAUDIO_ERROR_INTERNAL; |
| } |
| if ((result = mReadCounterParcelable.validate()) != AAUDIO_OK) { |
| ALOGE("RingBufferParcelable invalid mReadCounterParcelable = %d", result); |
| return result; |
| } |
| if ((result = mWriteCounterParcelable.validate()) != AAUDIO_OK) { |
| ALOGE("RingBufferParcelable invalid mWriteCounterParcelable = %d", result); |
| return result; |
| } |
| if ((result = mDataParcelable.validate()) != AAUDIO_OK) { |
| ALOGE("RingBufferParcelable invalid mDataParcelable = %d", result); |
| return result; |
| } |
| return AAUDIO_OK; |
| } |
| |
| |
| void RingBufferParcelable::dump() { |
| ALOGD("RingBufferParcelable mCapacityInFrames = %d ---------", mCapacityInFrames); |
| if (mCapacityInFrames > 0) { |
| ALOGD("RingBufferParcelable mBytesPerFrame = %d", mBytesPerFrame); |
| ALOGD("RingBufferParcelable mFramesPerBurst = %d", mFramesPerBurst); |
| ALOGD("RingBufferParcelable mFlags = %u", mFlags); |
| mReadCounterParcelable.dump(); |
| mWriteCounterParcelable.dump(); |
| mDataParcelable.dump(); |
| } |
| } |