blob: 50c89e822da51be45254e89eb55ef0acda3a4c65 [file] [log] [blame]
/*
* 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 PATH(APM_XSD_ENUMS_H_FILENAME)
#include "stream_in.h"
#include "device_port_source.h"
#include "deleters.h"
#include "audio_ops.h"
#include "util.h"
#include "debug.h"
namespace xsd {
using namespace ::android::audio::policy::configuration::CPP_VERSION;
}
namespace android {
namespace hardware {
namespace audio {
namespace CPP_VERSION {
namespace implementation {
using ::android::hardware::Void;
namespace {
struct ReadThread : public IOThread {
typedef MessageQueue<IStreamIn::ReadParameters, kSynchronizedReadWrite> CommandMQ;
typedef MessageQueue<IStreamIn::ReadStatus, kSynchronizedReadWrite> StatusMQ;
typedef MessageQueue<uint8_t, kSynchronizedReadWrite> DataMQ;
ReadThread(StreamIn *stream, const size_t bufferSize)
: mStream(stream)
, mCommandMQ(1)
, mStatusMQ(1)
, mDataMQ(bufferSize, true /* EventFlag */) {
if (!mCommandMQ.isValid()) {
ALOGE("ReadThread::%s:%d: mCommandMQ is invalid", __func__, __LINE__);
return;
}
if (!mDataMQ.isValid()) {
ALOGE("ReadThread::%s:%d: mDataMQ is invalid", __func__, __LINE__);
return;
}
if (!mStatusMQ.isValid()) {
ALOGE("ReadThread::%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("ReadThread::%s:%d: rawEfGroup is invalid", __func__, __LINE__);
return;
} else {
mEfGroup.reset(rawEfGroup);
}
mThread = std::thread(&ReadThread::threadLoop, this);
}
~ReadThread() {
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_FULL | EXIT_REQUEST | STAND_BY_REQUEST,
&efState);
if (efState & EXIT_REQUEST) {
return;
}
if (efState & STAND_BY_REQUEST) {
mSource.reset();
}
if (efState & (MessageQueueFlagBits::NOT_FULL | 0)) {
if (!mSource) {
mSource = DevicePortSource::create(mDataMQ.getQuantumCount(),
mStream->getDeviceAddress(),
mStream->getAudioConfig(),
mStream->getAudioOutputFlags(),
mStream->getFrameCounter());
LOG_ALWAYS_FATAL_IF(!mSource);
}
processCommand();
}
}
}
void processCommand() {
IStreamIn::ReadParameters rParameters;
if (!mCommandMQ.read(&rParameters)) {
return; // Nothing to do.
}
IStreamIn::ReadStatus rStatus;
switch (rParameters.command) {
case IStreamIn::ReadCommand::READ:
rStatus = doRead(rParameters);
break;
case IStreamIn::ReadCommand::GET_CAPTURE_POSITION:
rStatus = doGetCapturePosition();
break;
default:
ALOGE("ReadThread::%s:%d: Unknown read thread command code %d",
__func__, __LINE__, rParameters.command);
rStatus.retval = FAILURE(Result::NOT_SUPPORTED);
break;
}
rStatus.replyTo = rParameters.command;
if (!mStatusMQ.write(&rStatus)) {
ALOGE("ReadThread::%s:%d: status message queue write failed", __func__, __LINE__);
}
mEfGroup->wake(MessageQueueFlagBits::NOT_EMPTY | 0);
}
IStreamIn::ReadStatus doRead(const IStreamIn::ReadParameters &rParameters) {
struct MQWriter : public IWriter {
explicit MQWriter(DataMQ &mq) : dataMQ(mq) {}
size_t operator()(const void *dst, size_t sz) override {
if (dataMQ.write(static_cast<const uint8_t *>(dst), sz)) {
totalWritten += sz;
return sz;
} else {
ALOGE("WriteThread::%s:%d: DataMQ::write failed",
__func__, __LINE__);
return 0;
}
}
size_t totalWritten = 0;
DataMQ &dataMQ;
};
const size_t bytesToRead = std::min(mDataMQ.availableToWrite(),
static_cast<size_t>(rParameters.params.read));
MQWriter writer(mDataMQ);
const size_t framesLost =
mSource->read(mStream->getEffectiveVolume(), bytesToRead, writer);
if (framesLost > 0) {
mStream->addInputFramesLost(framesLost);
}
IStreamIn::ReadStatus status;
status.retval = Result::OK;
status.reply.read = writer.totalWritten;
return status;
}
IStreamIn::ReadStatus doGetCapturePosition() {
IStreamIn::ReadStatus status;
status.retval = mSource->getCapturePosition(
status.reply.capturePosition.frames,
status.reply.capturePosition.time);
return status;
}
StreamIn *const mStream;
CommandMQ mCommandMQ;
StatusMQ mStatusMQ;
DataMQ mDataMQ;
std::unique_ptr<EventFlag, deleters::forEventFlag> mEfGroup;
std::unique_ptr<DevicePortSource> mSource;
std::thread mThread;
std::promise<pthread_t> mTid;
};
} // namespace
StreamIn::StreamIn(sp<Device> dev,
int32_t ioHandle,
const DeviceAddress& device,
const AudioConfig& config,
hidl_vec<AudioInOutFlag> flags,
const SinkMetadata& sinkMetadata)
: mDev(std::move(dev))
, mCommon(ioHandle, device, config, std::move(flags))
, mSinkMetadata(sinkMetadata) {
}
StreamIn::~StreamIn() {
closeImpl(true);
}
Return<uint64_t> StreamIn::getFrameSize() {
return mCommon.getFrameSize();
}
Return<uint64_t> StreamIn::getFrameCount() {
return mCommon.getFrameCount();
}
Return<uint64_t> StreamIn::getBufferSize() {
return mCommon.getBufferSize();
}
Return<void> StreamIn::getSupportedProfiles(getSupportedProfiles_cb _hidl_cb) {
mCommon.getSupportedProfiles(_hidl_cb);
return Void();
}
Return<void> StreamIn::getAudioProperties(getAudioProperties_cb _hidl_cb) {
mCommon.getAudioProperties(_hidl_cb);
return Void();
}
Return<Result> StreamIn::setAudioProperties(const AudioConfigBaseOptional& config) {
(void)config;
return FAILURE(Result::NOT_SUPPORTED);
}
Return<Result> StreamIn::addEffect(uint64_t effectId) {
(void)effectId;
return FAILURE(Result::INVALID_ARGUMENTS);
}
Return<Result> StreamIn::removeEffect(uint64_t effectId) {
(void)effectId;
return FAILURE(Result::INVALID_ARGUMENTS);
}
Return<Result> StreamIn::standby() {
if (mReadThread) {
LOG_ALWAYS_FATAL_IF(!mReadThread->standby());
}
return Result::OK;
}
Return<void> StreamIn::getDevices(getDevices_cb _hidl_cb) {
mCommon.getDevices(_hidl_cb);
return Void();
}
Return<Result> StreamIn::setDevices(const hidl_vec<DeviceAddress>& devices) {
return mCommon.setDevices(devices);
}
Return<void> StreamIn::getParameters(const hidl_vec<ParameterValue>& context,
const hidl_vec<hidl_string>& keys,
getParameters_cb _hidl_cb) {
(void)context;
_hidl_cb((keys.size() > 0) ? Result::NOT_SUPPORTED : Result::OK, {});
return Void();
}
Return<Result> StreamIn::setParameters(const hidl_vec<ParameterValue>& context,
const hidl_vec<ParameterValue>& parameters) {
(void)context;
(void)parameters;
return Result::OK;
}
Return<Result> StreamIn::setHwAvSync(uint32_t hwAvSync) {
(void)hwAvSync;
return FAILURE(Result::NOT_SUPPORTED);
}
Return<Result> StreamIn::start() {
return FAILURE(Result::NOT_SUPPORTED);
}
Return<Result> StreamIn::stop() {
return FAILURE(Result::NOT_SUPPORTED);
}
Return<void> StreamIn::createMmapBuffer(int32_t minSizeFrames,
createMmapBuffer_cb _hidl_cb) {
(void)minSizeFrames;
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
return Void();
}
Return<void> StreamIn::getMmapPosition(getMmapPosition_cb _hidl_cb) {
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
return Void();
}
Result StreamIn::closeImpl(const bool fromDctor) {
if (mDev) {
mReadThread.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> StreamIn::close() {
return closeImpl(false);
}
Return<void> StreamIn::getAudioSource(getAudioSource_cb _hidl_cb) {
_hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
return Void();
}
Return<Result> StreamIn::setGain(float gain) {
(void)gain;
return FAILURE(Result::NOT_SUPPORTED);
}
Return<Result> StreamIn::updateSinkMetadata(const SinkMetadata& sinkMetadata) {
(void)sinkMetadata;
return FAILURE(Result::NOT_SUPPORTED);
}
Return<void> StreamIn::prepareForReading(uint32_t frameSize,
uint32_t framesCount,
prepareForReading_cb _hidl_cb) {
if (!frameSize || !framesCount || frameSize > 256 || framesCount > (1u << 20)) {
_hidl_cb(FAILURE(Result::INVALID_ARGUMENTS), {}, {}, {}, -1);
return Void();
}
if (mReadThread) { // INVALID_STATE if the method was already called.
_hidl_cb(FAILURE(Result::INVALID_STATE), {}, {}, {}, -1);
return Void();
}
auto t = std::make_unique<ReadThread>(this, frameSize * framesCount);
if (t->isRunning()) {
_hidl_cb(Result::OK,
*(t->mCommandMQ.getDesc()),
*(t->mDataMQ.getDesc()),
*(t->mStatusMQ.getDesc()),
t->getTid().get());
mReadThread = std::move(t);
} else {
_hidl_cb(FAILURE(Result::INVALID_ARGUMENTS), {}, {}, {}, -1);
}
return Void();
}
Return<uint32_t> StreamIn::getInputFramesLost() {
return 0;
}
Return<void> StreamIn::getCapturePosition(getCapturePosition_cb _hidl_cb) {
const auto r = static_cast<ReadThread*>(mReadThread.get());
if (!r) {
_hidl_cb(FAILURE(Result::INVALID_STATE), {}, {});
return Void();
}
const auto s = r->mSource.get();
if (!s) {
_hidl_cb(Result::OK, mFrames, systemTime(SYSTEM_TIME_MONOTONIC));
} else {
uint64_t frames;
uint64_t time;
const Result r = s->getCapturePosition(frames, time);
_hidl_cb(r, frames, time);
}
return Void();
}
Return<void> StreamIn::getActiveMicrophones(getActiveMicrophones_cb _hidl_cb) {
_hidl_cb(Result::OK, {util::getMicrophoneInfo()});
return Void();
}
Return<Result> StreamIn::setMicrophoneDirection(MicrophoneDirection direction) {
(void)direction;
return FAILURE(Result::NOT_SUPPORTED);
}
Return<Result> StreamIn::setMicrophoneFieldDimension(float zoom) {
(void)zoom;
return FAILURE(Result::NOT_SUPPORTED);
}
void StreamIn::setMicMute(bool mute) {
mEffectiveVolume =
(mute && (xsd::stringToAudioDevice(getDeviceAddress().deviceType) ==
xsd::AudioDevice::AUDIO_DEVICE_IN_BUILTIN_MIC))
? 0.0f : 1.0f;
}
bool StreamIn::validateDeviceAddress(const DeviceAddress& device) {
return DevicePortSource::validateDeviceAddress(device);
}
bool StreamIn::validateFlags(const hidl_vec<AudioInOutFlag>& flags) {
return std::all_of(flags.begin(), flags.end(), [](const AudioInOutFlag& flag){
return xsd::stringToAudioInOutFlag(flag) != xsd::AudioInOutFlag::UNKNOWN;
});
}
bool StreamIn::validateSinkMetadata(const SinkMetadata& sinkMetadata) {
for (const auto& track : sinkMetadata.tracks) {
if (xsd::isUnknownAudioSource(track.source)
|| xsd::isUnknownAudioChannelMask(track.channelMask)) {
return false;
}
for (const auto& tag : track.tags) {
if (!xsd::isVendorExtension(tag)) {
return false;
}
}
}
return true;
}
} // namespace implementation
} // namespace CPP_VERSION
} // namespace audio
} // namespace hardware
} // namespace android