1 /*
2 * Copyright (C) 2012 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #define LOG_TAG "SourceAudioBufferProvider"
18 //#define LOG_NDEBUG 0
19
20 #include <utils/Log.h>
21 #include <media/nbaio/SourceAudioBufferProvider.h>
22
23 namespace android {
24
SourceAudioBufferProvider(const sp<NBAIO_Source> & source)25 SourceAudioBufferProvider::SourceAudioBufferProvider(const sp<NBAIO_Source>& source) :
26 mSource(source),
27 // mFrameSize below
28 mAllocated(NULL), mSize(0), mOffset(0), mRemaining(0), mGetCount(0), mFramesReleased(0)
29 {
30 ALOG_ASSERT(source != 0);
31
32 // negotiate with source
33 NBAIO_Format counterOffers[1];
34 size_t numCounterOffers = 1;
35 [[maybe_unused]] ssize_t index = source->negotiate(NULL, 0, counterOffers, numCounterOffers);
36 ALOG_ASSERT(index == (ssize_t) NEGOTIATE && numCounterOffers > 0);
37 numCounterOffers = 0;
38 index = source->negotiate(counterOffers, 1, NULL, numCounterOffers);
39 ALOG_ASSERT(index == 0);
40 mFrameSize = Format_frameSize(source->format());
41 }
42
~SourceAudioBufferProvider()43 SourceAudioBufferProvider::~SourceAudioBufferProvider()
44 {
45 free(mAllocated);
46 }
47
getNextBuffer(Buffer * buffer)48 status_t SourceAudioBufferProvider::getNextBuffer(Buffer *buffer)
49 {
50 ALOG_ASSERT(buffer != NULL && buffer->frameCount > 0 && mGetCount == 0);
51 // any leftover data available?
52 if (mRemaining > 0) {
53 ALOG_ASSERT(mOffset + mRemaining <= mSize);
54 if (mRemaining < buffer->frameCount) {
55 buffer->frameCount = mRemaining;
56 }
57 buffer->raw = (char *) mAllocated + (mOffset * mFrameSize);
58 mGetCount = buffer->frameCount;
59 return OK;
60 }
61 // do we need to reallocate?
62 if (buffer->frameCount > mSize) {
63 free(mAllocated);
64 // Android convention is to _not_ check the return value of malloc and friends.
65 // But in this case the calloc() can also fail due to integer overflow,
66 // so we check and recover.
67 mAllocated = calloc(buffer->frameCount, mFrameSize);
68 if (mAllocated == NULL) {
69 mSize = 0;
70 goto fail;
71 }
72 mSize = buffer->frameCount;
73 }
74 {
75 // read from source
76 ssize_t actual = mSource->read(mAllocated, buffer->frameCount);
77 if (actual > 0) {
78 ALOG_ASSERT((size_t) actual <= buffer->frameCount);
79 mOffset = 0;
80 mRemaining = actual;
81 buffer->raw = mAllocated;
82 buffer->frameCount = actual;
83 mGetCount = actual;
84 return OK;
85 }
86 }
87 fail:
88 buffer->raw = NULL;
89 buffer->frameCount = 0;
90 mGetCount = 0;
91 return NOT_ENOUGH_DATA;
92 }
93
releaseBuffer(Buffer * buffer)94 void SourceAudioBufferProvider::releaseBuffer(Buffer *buffer)
95 {
96 ALOG_ASSERT((buffer != NULL) &&
97 (buffer->raw == (char *) mAllocated + (mOffset * mFrameSize)) &&
98 (buffer->frameCount <= mGetCount) &&
99 (mGetCount <= mRemaining) &&
100 (mOffset + mRemaining <= mSize));
101 mOffset += buffer->frameCount;
102 mRemaining -= buffer->frameCount;
103 mFramesReleased += buffer->frameCount;
104 buffer->raw = NULL;
105 buffer->frameCount = 0;
106 mGetCount = 0;
107 }
108
framesReady() const109 size_t SourceAudioBufferProvider::framesReady() const
110 {
111 ssize_t avail = mSource->availableToRead();
112 return avail < 0 ? 0 : (size_t) avail;
113 }
114
framesReleased() const115 int64_t SourceAudioBufferProvider::framesReleased() const
116 {
117 return mFramesReleased;
118 }
119
onTimestamp(const ExtendedTimestamp & timestamp)120 void SourceAudioBufferProvider::onTimestamp(const ExtendedTimestamp ×tamp)
121 {
122 mSource->onTimestamp(timestamp);
123 }
124
125 } // namespace android
126