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.
543 lines
16 KiB
543 lines
16 KiB
/*
|
|
* Copyright (C) 2020 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.
|
|
*/
|
|
|
|
#include <log/log.h>
|
|
#include <fmq/EventFlag.h>
|
|
#include <fmq/MessageQueue.h>
|
|
#include <hidl/MQDescriptor.h>
|
|
#include <hidl/Status.h>
|
|
#include <utils/ThreadDefs.h>
|
|
#include <future>
|
|
#include <thread>
|
|
#include "stream_out.h"
|
|
#include "device_port_sink.h"
|
|
#include "deleters.h"
|
|
#include "audio_ops.h"
|
|
#include "util.h"
|
|
#include "debug.h"
|
|
|
|
namespace android {
|
|
namespace hardware {
|
|
namespace audio {
|
|
namespace V6_0 {
|
|
namespace implementation {
|
|
|
|
using ::android::hardware::Void;
|
|
using namespace ::android::hardware::audio::common::V6_0;
|
|
using namespace ::android::hardware::audio::V6_0;
|
|
|
|
namespace {
|
|
|
|
struct WriteThread : public IOThread {
|
|
typedef MessageQueue<IStreamOut::WriteCommand, kSynchronizedReadWrite> CommandMQ;
|
|
typedef MessageQueue<IStreamOut::WriteStatus, kSynchronizedReadWrite> StatusMQ;
|
|
typedef MessageQueue<uint8_t, kSynchronizedReadWrite> DataMQ;
|
|
|
|
WriteThread(StreamOut *stream, const size_t mqBufferSize)
|
|
: mStream(stream)
|
|
, mCommandMQ(1)
|
|
, mStatusMQ(1)
|
|
, mDataMQ(mqBufferSize, true /* EventFlag */) {
|
|
if (!mCommandMQ.isValid()) {
|
|
ALOGE("WriteThread::%s:%d: mCommandMQ is invalid", __func__, __LINE__);
|
|
return;
|
|
}
|
|
if (!mDataMQ.isValid()) {
|
|
ALOGE("WriteThread::%s:%d: mDataMQ is invalid", __func__, __LINE__);
|
|
return;
|
|
}
|
|
if (!mStatusMQ.isValid()) {
|
|
ALOGE("WriteThread::%s:%d: mStatusMQ is invalid", __func__, __LINE__);
|
|
return;
|
|
}
|
|
|
|
status_t status;
|
|
|
|
EventFlag* rawEfGroup = nullptr;
|
|
status = EventFlag::createEventFlag(mDataMQ.getEventFlagWord(), &rawEfGroup);
|
|
if (status != OK || !rawEfGroup) {
|
|
ALOGE("WriteThread::%s:%d: rawEfGroup is invalid", __func__, __LINE__);
|
|
return;
|
|
} else {
|
|
mEfGroup.reset(rawEfGroup);
|
|
}
|
|
|
|
mThread = std::thread(&WriteThread::threadLoop, this);
|
|
}
|
|
|
|
~WriteThread() {
|
|
if (mThread.joinable()) {
|
|
requestExit();
|
|
mThread.join();
|
|
}
|
|
}
|
|
|
|
EventFlag *getEventFlag() override {
|
|
return mEfGroup.get();
|
|
}
|
|
|
|
bool isRunning() const {
|
|
return mThread.joinable();
|
|
}
|
|
|
|
std::future<pthread_t> getTid() {
|
|
return mTid.get_future();
|
|
}
|
|
|
|
void threadLoop() {
|
|
util::setThreadPriority(PRIORITY_URGENT_AUDIO);
|
|
mTid.set_value(pthread_self());
|
|
|
|
while (true) {
|
|
uint32_t efState = 0;
|
|
mEfGroup->wait(MessageQueueFlagBits::NOT_EMPTY | STAND_BY_REQUEST | EXIT_REQUEST,
|
|
&efState);
|
|
if (efState & EXIT_REQUEST) {
|
|
return;
|
|
}
|
|
|
|
if (efState & STAND_BY_REQUEST) {
|
|
mSink.reset();
|
|
}
|
|
|
|
if (efState & (MessageQueueFlagBits::NOT_EMPTY | 0)) {
|
|
if (!mSink) {
|
|
mSink = DevicePortSink::create(mDataMQ.getQuantumCount(),
|
|
mStream->getDeviceAddress(),
|
|
mStream->getAudioConfig(),
|
|
mStream->getAudioOutputFlags(),
|
|
mStream->getFrameCounter());
|
|
LOG_ALWAYS_FATAL_IF(!mSink);
|
|
}
|
|
|
|
processCommand();
|
|
}
|
|
}
|
|
}
|
|
|
|
void processCommand() {
|
|
IStreamOut::WriteCommand wCommand;
|
|
|
|
if (!mCommandMQ.read(&wCommand)) {
|
|
return; // Nothing to do.
|
|
}
|
|
|
|
IStreamOut::WriteStatus wStatus;
|
|
switch (wCommand) {
|
|
case IStreamOut::WriteCommand::WRITE:
|
|
wStatus = doWrite();
|
|
break;
|
|
|
|
case IStreamOut::WriteCommand::GET_PRESENTATION_POSITION:
|
|
wStatus = doGetPresentationPosition();
|
|
break;
|
|
|
|
case IStreamOut::WriteCommand::GET_LATENCY:
|
|
wStatus = doGetLatency();
|
|
break;
|
|
|
|
default:
|
|
ALOGE("WriteThread::%s:%d: Unknown write thread command code %d",
|
|
__func__, __LINE__, wCommand);
|
|
wStatus.retval = FAILURE(Result::NOT_SUPPORTED);
|
|
break;
|
|
}
|
|
|
|
wStatus.replyTo = wCommand;
|
|
|
|
if (!mStatusMQ.write(&wStatus)) {
|
|
ALOGE("status message queue write failed");
|
|
}
|
|
|
|
mEfGroup->wake(MessageQueueFlagBits::NOT_FULL | 0);
|
|
}
|
|
|
|
IStreamOut::WriteStatus doWrite() {
|
|
struct MQReader : public IReader {
|
|
explicit MQReader(DataMQ &mq) : dataMQ(mq) {}
|
|
|
|
size_t operator()(void *dst, size_t sz) override {
|
|
if (dataMQ.read(static_cast<uint8_t *>(dst), sz)) {
|
|
totalRead += sz;
|
|
return sz;
|
|
} else {
|
|
ALOGE("WriteThread::%s:%d: DataMQ::read failed",
|
|
__func__, __LINE__);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
size_t totalRead = 0;
|
|
DataMQ &dataMQ;
|
|
};
|
|
|
|
MQReader reader(mDataMQ);
|
|
mSink->write(mStream->getEffectiveVolume(), mDataMQ.availableToRead(), reader);
|
|
|
|
IStreamOut::WriteStatus status;
|
|
status.retval = Result::OK;
|
|
status.reply.written = reader.totalRead;
|
|
return status;
|
|
}
|
|
|
|
IStreamOut::WriteStatus doGetPresentationPosition() {
|
|
IStreamOut::WriteStatus status;
|
|
|
|
status.retval = mSink->getPresentationPosition(
|
|
status.reply.presentationPosition.frames,
|
|
status.reply.presentationPosition.timeStamp);
|
|
|
|
return status;
|
|
}
|
|
|
|
IStreamOut::WriteStatus doGetLatency() {
|
|
IStreamOut::WriteStatus status;
|
|
|
|
status.retval = Result::OK;
|
|
status.reply.latencyMs = mStream->getLatency();
|
|
|
|
return status;
|
|
}
|
|
|
|
StreamOut *const mStream;
|
|
CommandMQ mCommandMQ;
|
|
StatusMQ mStatusMQ;
|
|
DataMQ mDataMQ;
|
|
std::unique_ptr<EventFlag, deleters::forEventFlag> mEfGroup;
|
|
std::unique_ptr<DevicePortSink> mSink;
|
|
std::thread mThread;
|
|
std::promise<pthread_t> mTid;
|
|
};
|
|
|
|
} // namespace
|
|
|
|
StreamOut::StreamOut(sp<PrimaryDevice> dev,
|
|
int32_t ioHandle,
|
|
const DeviceAddress& device,
|
|
const AudioConfig& config,
|
|
hidl_bitfield<AudioOutputFlag> flags,
|
|
const SourceMetadata& sourceMetadata)
|
|
: mDev(std::move(dev))
|
|
, mCommon(ioHandle, device, config, flags)
|
|
, mSourceMetadata(sourceMetadata) {}
|
|
|
|
StreamOut::~StreamOut() {
|
|
closeImpl(true);
|
|
}
|
|
|
|
Return<uint64_t> StreamOut::getFrameSize() {
|
|
return mCommon.getFrameSize();
|
|
}
|
|
|
|
Return<uint64_t> StreamOut::getFrameCount() {
|
|
return mCommon.getFrameCount();
|
|
}
|
|
|
|
Return<uint64_t> StreamOut::getBufferSize() {
|
|
return mCommon.getBufferSize();
|
|
}
|
|
|
|
Return<uint32_t> StreamOut::getSampleRate() {
|
|
return mCommon.getSampleRate();
|
|
}
|
|
|
|
Return<void> StreamOut::getSupportedSampleRates(AudioFormat format,
|
|
getSupportedSampleRates_cb _hidl_cb) {
|
|
mCommon.getSupportedSampleRates(format, _hidl_cb);
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setSampleRate(uint32_t sampleRateHz) {
|
|
return mCommon.setSampleRate(sampleRateHz);
|
|
}
|
|
|
|
Return<hidl_bitfield<AudioChannelMask>> StreamOut::getChannelMask() {
|
|
return mCommon.getChannelMask();
|
|
}
|
|
|
|
Return<void> StreamOut::getSupportedChannelMasks(AudioFormat format,
|
|
IStream::getSupportedChannelMasks_cb _hidl_cb) {
|
|
mCommon.getSupportedChannelMasks(format, _hidl_cb);
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setChannelMask(hidl_bitfield<AudioChannelMask> mask) {
|
|
return mCommon.setChannelMask(mask);
|
|
}
|
|
|
|
Return<AudioFormat> StreamOut::getFormat() {
|
|
return mCommon.getFormat();
|
|
}
|
|
|
|
Return<void> StreamOut::getSupportedFormats(getSupportedFormats_cb _hidl_cb) {
|
|
mCommon.getSupportedFormats(_hidl_cb);
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setFormat(AudioFormat format) {
|
|
return mCommon.setFormat(format);
|
|
}
|
|
|
|
Return<void> StreamOut::getAudioProperties(getAudioProperties_cb _hidl_cb) {
|
|
mCommon.getAudioProperties(_hidl_cb);
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::addEffect(uint64_t effectId) {
|
|
(void)effectId;
|
|
return FAILURE(Result::INVALID_ARGUMENTS);
|
|
}
|
|
|
|
Return<Result> StreamOut::removeEffect(uint64_t effectId) {
|
|
(void)effectId;
|
|
return FAILURE(Result::INVALID_ARGUMENTS);
|
|
}
|
|
|
|
Return<Result> StreamOut::standby() {
|
|
if (mWriteThread) {
|
|
LOG_ALWAYS_FATAL_IF(!mWriteThread->standby());
|
|
}
|
|
|
|
return Result::OK;
|
|
}
|
|
|
|
Return<void> StreamOut::getDevices(getDevices_cb _hidl_cb) {
|
|
mCommon.getDevices(_hidl_cb);
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setDevices(const hidl_vec<DeviceAddress>& devices) {
|
|
return mCommon.setDevices(devices);
|
|
}
|
|
|
|
Return<void> StreamOut::getParameters(const hidl_vec<ParameterValue>& context,
|
|
const hidl_vec<hidl_string>& keys,
|
|
getParameters_cb _hidl_cb) {
|
|
(void)context;
|
|
_hidl_cb((keys.size() > 0) ? FAILURE(Result::NOT_SUPPORTED) : Result::OK, {});
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setParameters(const hidl_vec<ParameterValue>& context,
|
|
const hidl_vec<ParameterValue>& parameters) {
|
|
(void)context;
|
|
(void)parameters;
|
|
return Result::OK;
|
|
}
|
|
|
|
Return<Result> StreamOut::setHwAvSync(uint32_t hwAvSync) {
|
|
(void)hwAvSync;
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Result StreamOut::closeImpl(const bool fromDctor) {
|
|
if (mDev) {
|
|
mWriteThread.reset();
|
|
mDev->unrefDevice(this);
|
|
mDev = nullptr;
|
|
return Result::OK;
|
|
} else if (fromDctor) {
|
|
// closeImpl is always called from the dctor, it is ok if mDev is null,
|
|
// we don't want to log the error in this case.
|
|
return Result::OK;
|
|
} else {
|
|
return FAILURE(Result::INVALID_STATE);
|
|
}
|
|
}
|
|
|
|
Return<Result> StreamOut::close() {
|
|
return closeImpl(false);
|
|
}
|
|
|
|
Return<Result> StreamOut::start() {
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<Result> StreamOut::stop() {
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<void> StreamOut::createMmapBuffer(int32_t minSizeFrames,
|
|
createMmapBuffer_cb _hidl_cb) {
|
|
(void)minSizeFrames;
|
|
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
|
|
return Void();
|
|
}
|
|
|
|
Return<void> StreamOut::getMmapPosition(getMmapPosition_cb _hidl_cb) {
|
|
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
|
|
return Void();
|
|
}
|
|
|
|
Return<uint32_t> StreamOut::getLatency() {
|
|
return mCommon.getFrameCount() * 1000 / mCommon.getSampleRate();
|
|
}
|
|
|
|
Return<Result> StreamOut::setVolume(float left, float right) {
|
|
if (isnan(left) || left < 0.0f || left > 1.0f
|
|
|| right < 0.0f || right > 1.0f || isnan(right)) {
|
|
return FAILURE(Result::INVALID_ARGUMENTS);
|
|
}
|
|
|
|
std::lock_guard<std::mutex> guard(mMutex);
|
|
mStreamVolume = (left + right) / 2.0f;
|
|
updateEffectiveVolumeLocked();
|
|
return Result::OK;
|
|
}
|
|
|
|
Return<void> StreamOut::updateSourceMetadata(const SourceMetadata& sourceMetadata) {
|
|
(void)sourceMetadata;
|
|
return Void();
|
|
}
|
|
|
|
Return<void> StreamOut::prepareForWriting(uint32_t frameSize,
|
|
uint32_t framesCount,
|
|
prepareForWriting_cb _hidl_cb) {
|
|
if (!frameSize || !framesCount || frameSize > 256 || framesCount > (1u << 20)) {
|
|
_hidl_cb(FAILURE(Result::INVALID_ARGUMENTS), {}, {}, {}, {});
|
|
return Void();
|
|
}
|
|
|
|
if (mWriteThread) { // INVALID_STATE if the method was already called.
|
|
_hidl_cb(FAILURE(Result::INVALID_STATE), {}, {}, {}, {});
|
|
return Void();
|
|
}
|
|
|
|
auto t = std::make_unique<WriteThread>(this, frameSize * framesCount);
|
|
|
|
if (t->isRunning()) {
|
|
_hidl_cb(Result::OK,
|
|
*(t->mCommandMQ.getDesc()),
|
|
*(t->mDataMQ.getDesc()),
|
|
*(t->mStatusMQ.getDesc()),
|
|
{.pid = getpid(), .tid = t->getTid().get()});
|
|
|
|
mWriteThread = std::move(t);
|
|
} else {
|
|
_hidl_cb(FAILURE(Result::INVALID_ARGUMENTS), {}, {}, {}, {});
|
|
}
|
|
|
|
return Void();
|
|
}
|
|
|
|
Return<void> StreamOut::getRenderPosition(getRenderPosition_cb _hidl_cb) {
|
|
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), 0);
|
|
return Void();
|
|
}
|
|
|
|
Return<void> StreamOut::getNextWriteTimestamp(getNextWriteTimestamp_cb _hidl_cb) {
|
|
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), 0);
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setCallback(const sp<IStreamOutCallback>& callback) {
|
|
(void)callback;
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<Result> StreamOut::clearCallback() {
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<Result> StreamOut::setEventCallback(const sp<IStreamOutEventCallback>& callback) {
|
|
(void)callback;
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<void> StreamOut::supportsPauseAndResume(supportsPauseAndResume_cb _hidl_cb) {
|
|
_hidl_cb(false, false);
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::pause() {
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<Result> StreamOut::resume() {
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<bool> StreamOut::supportsDrain() {
|
|
return false;
|
|
}
|
|
|
|
Return<Result> StreamOut::drain(AudioDrain type) {
|
|
(void)type;
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<Result> StreamOut::flush() {
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<void> StreamOut::getPresentationPosition(getPresentationPosition_cb _hidl_cb) {
|
|
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), {}, {}); // see WriteThread::doGetPresentationPosition
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::selectPresentation(int32_t presentationId,
|
|
int32_t programId) {
|
|
(void)presentationId;
|
|
(void)programId;
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<void> StreamOut::getDualMonoMode(getDualMonoMode_cb _hidl_cb) {
|
|
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setDualMonoMode(DualMonoMode mode) {
|
|
(void)mode;
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<void> StreamOut::getAudioDescriptionMixLevel(getAudioDescriptionMixLevel_cb _hidl_cb) {
|
|
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), 0);
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setAudioDescriptionMixLevel(float leveldB) {
|
|
(void)leveldB;
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
Return<void> StreamOut::getPlaybackRateParameters(getPlaybackRateParameters_cb _hidl_cb) {
|
|
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
|
|
return Void();
|
|
}
|
|
|
|
Return<Result> StreamOut::setPlaybackRateParameters(const PlaybackRate &playbackRate) {
|
|
(void)playbackRate;
|
|
return FAILURE(Result::NOT_SUPPORTED);
|
|
}
|
|
|
|
void StreamOut::setMasterVolume(float masterVolume) {
|
|
std::lock_guard<std::mutex> guard(mMutex);
|
|
mMasterVolume = masterVolume;
|
|
updateEffectiveVolumeLocked();
|
|
}
|
|
|
|
void StreamOut::updateEffectiveVolumeLocked() {
|
|
mEffectiveVolume = mMasterVolume * mStreamVolume;
|
|
}
|
|
|
|
} // namespace implementation
|
|
} // namespace V6_0
|
|
} // namespace audio
|
|
} // namespace hardware
|
|
} // namespace android
|