1204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk/*
2204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * Copyright 2016 The Android Open Source Project
3204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk *
4204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * Licensed under the Apache License, Version 2.0 (the "License");
5204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * you may not use this file except in compliance with the License.
6204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * You may obtain a copy of the License at
7204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk *
8204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk *      http://www.apache.org/licenses/LICENSE-2.0
9204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk *
10204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * Unless required by applicable law or agreed to in writing, software
11204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * distributed under the License is distributed on an "AS IS" BASIS,
12204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * See the License for the specific language governing permissions and
14204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * limitations under the License.
15204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk */
16204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
17c0c70e3c7dd10bc2c0caffcab1f3f5fb406b35fbPhil Burk#define LOG_TAG "AAudio"
18c0c70e3c7dd10bc2c0caffcab1f3f5fb406b35fbPhil Burk//#define LOG_NDEBUG 0
19c0c70e3c7dd10bc2c0caffcab1f3f5fb406b35fbPhil Burk#include <utils/Log.h>
20c0c70e3c7dd10bc2c0caffcab1f3f5fb406b35fbPhil Burk
21204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk#include <stdint.h>
22204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
23204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk#include <binder/Parcelable.h>
24204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
255ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk#include "binding/AAudioServiceDefinitions.h"
26204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk#include "binding/SharedRegionParcelable.h"
27204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk#include "binding/RingBufferParcelable.h"
28204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
295ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burkusing namespace aaudio;
30204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
31204a163c86f357a878873fe7d4c4164f3d55c9b6Phil BurkRingBufferParcelable::RingBufferParcelable() {}
32204a163c86f357a878873fe7d4c4164f3d55c9b6Phil BurkRingBufferParcelable::~RingBufferParcelable() {}
33204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
34204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk// TODO This assumes that all three use the same SharedMemoryParcelable
35204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkvoid RingBufferParcelable::setupMemory(int32_t sharedMemoryIndex,
36204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                 int32_t dataMemoryOffset,
37204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                 int32_t dataSizeInBytes,
38204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                 int32_t readCounterOffset,
39204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                 int32_t writeCounterOffset,
40204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                 int32_t counterSizeBytes) {
41204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mReadCounterParcelable.setup(sharedMemoryIndex, readCounterOffset, counterSizeBytes);
42204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mWriteCounterParcelable.setup(sharedMemoryIndex, writeCounterOffset, counterSizeBytes);
43204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mDataParcelable.setup(sharedMemoryIndex, dataMemoryOffset, dataSizeInBytes);
44204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
45204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
46204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkvoid RingBufferParcelable::setupMemory(int32_t sharedMemoryIndex,
47204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                 int32_t dataMemoryOffset,
48204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                 int32_t dataSizeInBytes) {
49204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mReadCounterParcelable.setup(sharedMemoryIndex, 0, 0);
50204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mWriteCounterParcelable.setup(sharedMemoryIndex, 0, 0);
51204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mDataParcelable.setup(sharedMemoryIndex, dataMemoryOffset, dataSizeInBytes);
52204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
53204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
54204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkint32_t RingBufferParcelable::getBytesPerFrame() {
55204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    return mBytesPerFrame;
56204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
57204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
58204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkvoid RingBufferParcelable::setBytesPerFrame(int32_t bytesPerFrame) {
59204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mBytesPerFrame = bytesPerFrame;
60204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
61204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
62204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkint32_t RingBufferParcelable::getFramesPerBurst() {
63204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    return mFramesPerBurst;
64204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
65204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
66204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkvoid RingBufferParcelable::setFramesPerBurst(int32_t framesPerBurst) {
67204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mFramesPerBurst = framesPerBurst;
68204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
69204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
70204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkint32_t RingBufferParcelable::getCapacityInFrames() {
71204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    return mCapacityInFrames;
72204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
73204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
74204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkvoid RingBufferParcelable::setCapacityInFrames(int32_t capacityInFrames) {
75204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    mCapacityInFrames = capacityInFrames;
76204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
77204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
78204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk/**
79204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk * The read and write must be symmetric.
80204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk */
81204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkstatus_t RingBufferParcelable::writeToParcel(Parcel* parcel) const {
825204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    status_t status = parcel->writeInt32(mCapacityInFrames);
835204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    if (status != NO_ERROR) goto error;
84204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    if (mCapacityInFrames > 0) {
855204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = parcel->writeInt32(mBytesPerFrame);
865204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
875204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = parcel->writeInt32(mFramesPerBurst);
885204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
895204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = parcel->writeInt32(mFlags);
905204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
915204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = mReadCounterParcelable.writeToParcel(parcel);
925204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
935204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = mWriteCounterParcelable.writeToParcel(parcel);
945204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
955204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = mDataParcelable.writeToParcel(parcel);
965204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
97204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
985204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    return NO_ERROR;
995204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burkerror:
1005204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    ALOGE("RingBufferParcelable::writeToParcel() error = %d", status);
1015204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    return status;
102204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
103204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
104204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkstatus_t RingBufferParcelable::readFromParcel(const Parcel* parcel) {
1055204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    status_t status = parcel->readInt32(&mCapacityInFrames);
1065204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    if (status != NO_ERROR) goto error;
107204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    if (mCapacityInFrames > 0) {
1085204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = parcel->readInt32(&mBytesPerFrame);
1095204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
1105204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = parcel->readInt32(&mFramesPerBurst);
1115204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
1125204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = parcel->readInt32((int32_t *)&mFlags);
1135204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
1145204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = mReadCounterParcelable.readFromParcel(parcel);
1155204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
1165204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = mWriteCounterParcelable.readFromParcel(parcel);
1175204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
1185204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        status = mDataParcelable.readFromParcel(parcel);
1195204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk        if (status != NO_ERROR) goto error;
120204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
1215204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    return NO_ERROR;
1225204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burkerror:
1235204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    ALOGE("RingBufferParcelable::readFromParcel() error = %d", status);
1245204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    return status;
125204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
126204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
1275ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burkaaudio_result_t RingBufferParcelable::resolve(SharedMemoryParcelable *memoryParcels, RingBufferDescriptor *descriptor) {
1285ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    aaudio_result_t result;
129204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
130204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    result = mReadCounterParcelable.resolve(memoryParcels,
131204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                                            (void **) &descriptor->readCounterAddress);
1325ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    if (result != AAUDIO_OK) {
133204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        return result;
134204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
135204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
136204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    result = mWriteCounterParcelable.resolve(memoryParcels,
137204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk                                             (void **) &descriptor->writeCounterAddress);
1385ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    if (result != AAUDIO_OK) {
139204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        return result;
140204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
141204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
142204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    result = mDataParcelable.resolve(memoryParcels, (void **) &descriptor->dataAddress);
1435ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    if (result != AAUDIO_OK) {
144204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        return result;
145204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
146204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
147204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    descriptor->bytesPerFrame = mBytesPerFrame;
148204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    descriptor->framesPerBurst = mFramesPerBurst;
149204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    descriptor->capacityInFrames = mCapacityInFrames;
150204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    descriptor->flags = mFlags;
1515ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    return AAUDIO_OK;
152204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
153204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
1545ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burkaaudio_result_t RingBufferParcelable::validate() {
1555ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    aaudio_result_t result;
156204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    if (mCapacityInFrames < 0 || mCapacityInFrames >= 32 * 1024) {
157204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGE("RingBufferParcelable invalid mCapacityInFrames = %d", mCapacityInFrames);
1585ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk        return AAUDIO_ERROR_INTERNAL;
159204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
160204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    if (mBytesPerFrame < 0 || mBytesPerFrame >= 256) {
161204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGE("RingBufferParcelable invalid mBytesPerFrame = %d", mBytesPerFrame);
1625ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk        return AAUDIO_ERROR_INTERNAL;
163204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
1645204d315c6c6f53188f8d1414dd1b55b6c90142bPhil Burk    if (mFramesPerBurst < 0 || mFramesPerBurst >= 16 * 1024) {
165204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGE("RingBufferParcelable invalid mFramesPerBurst = %d", mFramesPerBurst);
1665ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk        return AAUDIO_ERROR_INTERNAL;
167204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
1685ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    if ((result = mReadCounterParcelable.validate()) != AAUDIO_OK) {
169204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGE("RingBufferParcelable invalid mReadCounterParcelable = %d", result);
170204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        return result;
171204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
1725ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    if ((result = mWriteCounterParcelable.validate()) != AAUDIO_OK) {
173204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGE("RingBufferParcelable invalid mWriteCounterParcelable = %d", result);
174204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        return result;
175204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
1765ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    if ((result = mDataParcelable.validate()) != AAUDIO_OK) {
177204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGE("RingBufferParcelable invalid mDataParcelable = %d", result);
178204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        return result;
179204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
1805ed503c7a66c90f93759c90237a9b432dbd93f9fPhil Burk    return AAUDIO_OK;
181204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
182204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
183204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk
184204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burkvoid RingBufferParcelable::dump() {
185204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    ALOGD("RingBufferParcelable mCapacityInFrames = %d ---------", mCapacityInFrames);
186204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    if (mCapacityInFrames > 0) {
187204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGD("RingBufferParcelable mBytesPerFrame = %d", mBytesPerFrame);
188204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGD("RingBufferParcelable mFramesPerBurst = %d", mFramesPerBurst);
189204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        ALOGD("RingBufferParcelable mFlags = %u", mFlags);
190204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        mReadCounterParcelable.dump();
191204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        mWriteCounterParcelable.dump();
192204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk        mDataParcelable.dump();
193204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk    }
194204a163c86f357a878873fe7d4c4164f3d55c9b6Phil Burk}
195