You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
214 lines
6.4 KiB
214 lines
6.4 KiB
/*
|
|
**
|
|
** Copyright 2015, 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 "AudioFlinger"
|
|
//#define LOG_NDEBUG 0
|
|
|
|
#include <media/audiohal/DeviceHalInterface.h>
|
|
#include <media/audiohal/StreamHalInterface.h>
|
|
#include <system/audio.h>
|
|
#include <utils/Log.h>
|
|
|
|
#include "AudioHwDevice.h"
|
|
#include "AudioStreamOut.h"
|
|
|
|
namespace android {
|
|
|
|
// ----------------------------------------------------------------------------
|
|
AudioStreamOut::AudioStreamOut(AudioHwDevice *dev, audio_output_flags_t flags)
|
|
: audioHwDev(dev)
|
|
, stream(NULL)
|
|
, flags(flags)
|
|
, mFramesWritten(0)
|
|
, mFramesWrittenAtStandby(0)
|
|
, mRenderPosition(0)
|
|
, mRateMultiplier(1)
|
|
, mHalFormatHasProportionalFrames(false)
|
|
, mHalFrameSize(0)
|
|
{
|
|
}
|
|
|
|
AudioStreamOut::~AudioStreamOut()
|
|
{
|
|
}
|
|
|
|
sp<DeviceHalInterface> AudioStreamOut::hwDev() const
|
|
{
|
|
return audioHwDev->hwDevice();
|
|
}
|
|
|
|
status_t AudioStreamOut::getRenderPosition(uint64_t *frames)
|
|
{
|
|
if (stream == 0) {
|
|
return NO_INIT;
|
|
}
|
|
|
|
uint32_t halPosition = 0;
|
|
status_t status = stream->getRenderPosition(&halPosition);
|
|
if (status != NO_ERROR) {
|
|
return status;
|
|
}
|
|
|
|
// Maintain a 64-bit render position using the 32-bit result from the HAL.
|
|
// This delta calculation relies on the arithmetic overflow behavior
|
|
// of integers. For example (100 - 0xFFFFFFF0) = 116.
|
|
const uint32_t truncatedPosition = (uint32_t)mRenderPosition;
|
|
int32_t deltaHalPosition; // initialization not needed, overwitten by __builtin_sub_overflow()
|
|
(void) __builtin_sub_overflow(halPosition, truncatedPosition, &deltaHalPosition);
|
|
if (deltaHalPosition > 0) {
|
|
mRenderPosition += deltaHalPosition;
|
|
}
|
|
// Scale from HAL sample rate to application rate.
|
|
*frames = mRenderPosition / mRateMultiplier;
|
|
|
|
return status;
|
|
}
|
|
|
|
// return bottom 32-bits of the render position
|
|
status_t AudioStreamOut::getRenderPosition(uint32_t *frames)
|
|
{
|
|
uint64_t position64 = 0;
|
|
status_t status = getRenderPosition(&position64);
|
|
if (status == NO_ERROR) {
|
|
*frames = (uint32_t)position64;
|
|
}
|
|
return status;
|
|
}
|
|
|
|
status_t AudioStreamOut::getPresentationPosition(uint64_t *frames, struct timespec *timestamp)
|
|
{
|
|
if (stream == 0) {
|
|
return NO_INIT;
|
|
}
|
|
|
|
uint64_t halPosition = 0;
|
|
status_t status = stream->getPresentationPosition(&halPosition, timestamp);
|
|
if (status != NO_ERROR) {
|
|
return status;
|
|
}
|
|
|
|
// Adjust for standby using HAL rate frames.
|
|
// Only apply this correction if the HAL is getting PCM frames.
|
|
if (mHalFormatHasProportionalFrames) {
|
|
uint64_t adjustedPosition = (halPosition <= mFramesWrittenAtStandby) ?
|
|
0 : (halPosition - mFramesWrittenAtStandby);
|
|
// Scale from HAL sample rate to application rate.
|
|
*frames = adjustedPosition / mRateMultiplier;
|
|
} else {
|
|
// For offloaded MP3 and other compressed formats.
|
|
*frames = halPosition;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
status_t AudioStreamOut::open(
|
|
audio_io_handle_t handle,
|
|
audio_devices_t deviceType,
|
|
struct audio_config *config,
|
|
const char *address)
|
|
{
|
|
sp<StreamOutHalInterface> outStream;
|
|
|
|
audio_output_flags_t customFlags = (config->format == AUDIO_FORMAT_IEC61937)
|
|
? (audio_output_flags_t)(flags | AUDIO_OUTPUT_FLAG_IEC958_NONAUDIO)
|
|
: flags;
|
|
|
|
int status = hwDev()->openOutputStream(
|
|
handle,
|
|
deviceType,
|
|
customFlags,
|
|
config,
|
|
address,
|
|
&outStream);
|
|
ALOGV("AudioStreamOut::open(), HAL returned "
|
|
" stream %p, sampleRate %d, Format %#x, "
|
|
"channelMask %#x, status %d",
|
|
outStream.get(),
|
|
config->sample_rate,
|
|
config->format,
|
|
config->channel_mask,
|
|
status);
|
|
|
|
// Some HALs may not recognize AUDIO_FORMAT_IEC61937. But if we declare
|
|
// it as PCM then it will probably work.
|
|
if (status != NO_ERROR && config->format == AUDIO_FORMAT_IEC61937) {
|
|
struct audio_config customConfig = *config;
|
|
customConfig.format = AUDIO_FORMAT_PCM_16_BIT;
|
|
|
|
status = hwDev()->openOutputStream(
|
|
handle,
|
|
deviceType,
|
|
customFlags,
|
|
&customConfig,
|
|
address,
|
|
&outStream);
|
|
ALOGV("AudioStreamOut::open(), treat IEC61937 as PCM, status = %d", status);
|
|
}
|
|
|
|
if (status == NO_ERROR) {
|
|
stream = outStream;
|
|
mHalFormatHasProportionalFrames = audio_has_proportional_frames(config->format);
|
|
status = stream->getFrameSize(&mHalFrameSize);
|
|
LOG_ALWAYS_FATAL_IF(status != OK, "Error retrieving frame size from HAL: %d", status);
|
|
LOG_ALWAYS_FATAL_IF(mHalFrameSize <= 0, "Error frame size was %zu but must be greater than"
|
|
" zero", mHalFrameSize);
|
|
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
audio_config_base_t AudioStreamOut::getAudioProperties() const
|
|
{
|
|
audio_config_base_t result = AUDIO_CONFIG_BASE_INITIALIZER;
|
|
if (stream->getAudioProperties(&result) != OK) {
|
|
result.sample_rate = 0;
|
|
result.channel_mask = AUDIO_CHANNEL_INVALID;
|
|
result.format = AUDIO_FORMAT_INVALID;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
int AudioStreamOut::flush()
|
|
{
|
|
mRenderPosition = 0;
|
|
mFramesWritten = 0;
|
|
mFramesWrittenAtStandby = 0;
|
|
status_t result = stream->flush();
|
|
return result != INVALID_OPERATION ? result : NO_ERROR;
|
|
}
|
|
|
|
int AudioStreamOut::standby()
|
|
{
|
|
mRenderPosition = 0;
|
|
mFramesWrittenAtStandby = mFramesWritten;
|
|
return stream->standby();
|
|
}
|
|
|
|
ssize_t AudioStreamOut::write(const void *buffer, size_t numBytes)
|
|
{
|
|
size_t bytesWritten;
|
|
status_t result = stream->write(buffer, numBytes, &bytesWritten);
|
|
if (result == OK && bytesWritten > 0 && mHalFrameSize > 0) {
|
|
mFramesWritten += bytesWritten / mHalFrameSize;
|
|
}
|
|
return result == OK ? bytesWritten : result;
|
|
}
|
|
|
|
} // namespace android
|