/*
* Copyright (C) 2012 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 "NBAIO"
//#define LOG_NDEBUG 0
#include <utils/Log.h>
#include <media/nbaio/NBAIO.h>
namespace android {
size_t Format_frameSize(const NBAIO_Format& format)
{
return format.mFrameSize;
}
const NBAIO_Format Format_Invalid = { 0, 0, AUDIO_FORMAT_INVALID, 0 };
unsigned Format_sampleRate(const NBAIO_Format& format)
{
if (!Format_isValid(format)) {
return 0;
}
return format.mSampleRate;
}
unsigned Format_channelCount(const NBAIO_Format& format)
{
if (!Format_isValid(format)) {
return 0;
}
return format.mChannelCount;
}
NBAIO_Format Format_from_SR_C(unsigned sampleRate, unsigned channelCount,
audio_format_t format)
{
if (sampleRate == 0 || channelCount == 0 || !audio_is_valid_format(format)) {
return Format_Invalid;
}
NBAIO_Format ret;
ret.mSampleRate = sampleRate;
ret.mChannelCount = channelCount;
ret.mFormat = format;
ret.mFrameSize = audio_is_linear_pcm(format) ?
channelCount * audio_bytes_per_sample(format) : sizeof(uint8_t);
return ret;
}
// This is a default implementation; it is expected that subclasses will optimize this.
ssize_t NBAIO_Sink::writeVia(writeVia_t via, size_t total, void *user, size_t block)
{
if (!mNegotiated) {
return (ssize_t) NEGOTIATE;
}
static const size_t maxBlock = 32;
size_t frameSize = Format_frameSize(mFormat);
ALOG_ASSERT(frameSize > 0 && frameSize <= 8);
// double guarantees alignment for stack similar to what malloc() gives for heap
if (block == 0 || block > maxBlock) {
block = maxBlock;
}
double buffer[((frameSize * block) + sizeof(double) - 1) / sizeof(double)];
size_t accumulator = 0;
while (accumulator < total) {
size_t count = total - accumulator;
if (count > block) {
count = block;
}
ssize_t ret = via(user, buffer, count);
if (ret > 0) {
ALOG_ASSERT((size_t) ret <= count);
size_t maxRet = ret;
ret = write(buffer, maxRet);
if (ret > 0) {
ALOG_ASSERT((size_t) ret <= maxRet);
accumulator += ret;
continue;
}
}
return accumulator > 0 ? accumulator : ret;
}
return accumulator;
}
// This is a default implementation; it is expected that subclasses will optimize this.
ssize_t NBAIO_Source::readVia(readVia_t via, size_t total, void *user,
int64_t readPTS, size_t block)
{
if (!mNegotiated) {
return (ssize_t) NEGOTIATE;
}
static const size_t maxBlock = 32;
size_t frameSize = Format_frameSize(mFormat);
ALOG_ASSERT(frameSize > 0 && frameSize <= 8);
// double guarantees alignment for stack similar to what malloc() gives for heap
if (block == 0 || block > maxBlock) {
block = maxBlock;
}
double buffer[((frameSize * block) + sizeof(double) - 1) / sizeof(double)];
size_t accumulator = 0;
while (accumulator < total) {
size_t count = total - accumulator;
if (count > block) {
count = block;
}
ssize_t ret = read(buffer, count, readPTS);
if (ret > 0) {
ALOG_ASSERT((size_t) ret <= count);
size_t maxRet = ret;
ret = via(user, buffer, maxRet, readPTS);
if (ret > 0) {
ALOG_ASSERT((size_t) ret <= maxRet);
accumulator += ret;
continue;
}
}
return accumulator > 0 ? accumulator : ret;
}
return accumulator;
}
// Default implementation that only accepts my mFormat
ssize_t NBAIO_Port::negotiate(const NBAIO_Format offers[], size_t numOffers,
NBAIO_Format counterOffers[], size_t& numCounterOffers)
{
ALOGV("negotiate offers=%p numOffers=%zu countersOffers=%p numCounterOffers=%zu",
offers, numOffers, counterOffers, numCounterOffers);
if (Format_isValid(mFormat)) {
for (size_t i = 0; i < numOffers; ++i) {
if (Format_isEqual(offers[i], mFormat)) {
mNegotiated = true;
return i;
}
}
if (numCounterOffers > 0) {
counterOffers[0] = mFormat;
}
numCounterOffers = 1;
} else {
numCounterOffers = 0;
}
return (ssize_t) NEGOTIATE;
}
bool Format_isValid(const NBAIO_Format& format)
{
return format.mSampleRate != 0 && format.mChannelCount != 0 &&
format.mFormat != AUDIO_FORMAT_INVALID && format.mFrameSize != 0;
}
bool Format_isEqual(const NBAIO_Format& format1, const NBAIO_Format& format2)
{
return format1.mSampleRate == format2.mSampleRate &&
format1.mChannelCount == format2.mChannelCount && format1.mFormat == format2.mFormat &&
format1.mFrameSize == format2.mFrameSize;
}
} // namespace android