1 /*
2 * Copyright (C) 2020 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 #include <log/log.h>
18 #include <fmq/EventFlag.h>
19 #include <fmq/MessageQueue.h>
20 #include <hidl/MQDescriptor.h>
21 #include <hidl/Status.h>
22 #include <utils/ThreadDefs.h>
23 #include <future>
24 #include <thread>
25 #include PATH(APM_XSD_ENUMS_H_FILENAME)
26 #include "stream_in.h"
27 #include "device_port_source.h"
28 #include "deleters.h"
29 #include "audio_ops.h"
30 #include "util.h"
31 #include "debug.h"
32
33 namespace xsd {
34 using namespace ::android::audio::policy::configuration::CPP_VERSION;
35 }
36
37 namespace android {
38 namespace hardware {
39 namespace audio {
40 namespace CPP_VERSION {
41 namespace implementation {
42
43 using ::android::hardware::Void;
44
45 namespace {
46
47 struct ReadThread : public IOThread {
48 typedef MessageQueue<IStreamIn::ReadParameters, kSynchronizedReadWrite> CommandMQ;
49 typedef MessageQueue<IStreamIn::ReadStatus, kSynchronizedReadWrite> StatusMQ;
50 typedef MessageQueue<uint8_t, kSynchronizedReadWrite> DataMQ;
51
ReadThreadandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread52 ReadThread(StreamIn *stream, const size_t bufferSize)
53 : mStream(stream)
54 , mCommandMQ(1)
55 , mStatusMQ(1)
56 , mDataMQ(bufferSize, true /* EventFlag */) {
57 if (!mCommandMQ.isValid()) {
58 ALOGE("ReadThread::%s:%d: mCommandMQ is invalid", __func__, __LINE__);
59 return;
60 }
61 if (!mDataMQ.isValid()) {
62 ALOGE("ReadThread::%s:%d: mDataMQ is invalid", __func__, __LINE__);
63 return;
64 }
65 if (!mStatusMQ.isValid()) {
66 ALOGE("ReadThread::%s:%d: mStatusMQ is invalid", __func__, __LINE__);
67 return;
68 }
69
70 status_t status;
71
72 EventFlag* rawEfGroup = nullptr;
73 status = EventFlag::createEventFlag(mDataMQ.getEventFlagWord(), &rawEfGroup);
74 if (status != OK || !rawEfGroup) {
75 ALOGE("ReadThread::%s:%d: rawEfGroup is invalid", __func__, __LINE__);
76 return;
77 } else {
78 mEfGroup.reset(rawEfGroup);
79 }
80
81 mThread = std::thread(&ReadThread::threadLoop, this);
82 }
83
~ReadThreadandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread84 ~ReadThread() {
85 if (mThread.joinable()) {
86 requestExit();
87 mThread.join();
88 }
89 }
90
getEventFlagandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread91 EventFlag *getEventFlag() override {
92 return mEfGroup.get();
93 }
94
isRunningandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread95 bool isRunning() const {
96 return mThread.joinable();
97 }
98
getTidandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread99 std::future<pthread_t> getTid() {
100 return mTid.get_future();
101 }
102
getCapturePositionandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread103 Result getCapturePosition(uint64_t &frames, uint64_t &ts) const {
104 std::lock_guard l(mExternalSourceReadLock);
105 if (mSource == nullptr) {
106 // this could return a slightly stale position under data race.
107 frames = 0;
108 ts = systemTime(SYSTEM_TIME_MONOTONIC);
109 return Result::OK;
110 } else {
111 return mSource->getCapturePosition(frames, ts);
112 }
113 }
114
getDescriptorsandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread115 auto getDescriptors() const {
116 return std::make_tuple(
117 mCommandMQ.getDesc(), mDataMQ.getDesc(), mStatusMQ.getDesc());
118 }
119
120 private:
threadLoopandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread121 void threadLoop() {
122 util::setThreadPriority(SP_AUDIO_SYS, PRIORITY_AUDIO);
123 mTid.set_value(pthread_self());
124
125 while (true) {
126 uint32_t efState = 0;
127 mEfGroup->wait(MessageQueueFlagBits::NOT_FULL | EXIT_REQUEST | STAND_BY_REQUEST,
128 &efState);
129 if (efState & EXIT_REQUEST) {
130 return;
131 }
132
133 if (efState & STAND_BY_REQUEST) {
134 ALOGD("%s: entering standby", __func__);
135 std::lock_guard l(mExternalSourceReadLock);
136 mSource.reset();
137 }
138
139 if (efState & (MessageQueueFlagBits::NOT_FULL | 0)) {
140 if (!mSource) {
141 auto source = DevicePortSource::create(mDataMQ.getQuantumCount(),
142 mStream->getDeviceAddress(),
143 mStream->getAudioConfig(),
144 mStream->getAudioOutputFlags(),
145 mStream->getFrameCounter());
146 LOG_ALWAYS_FATAL_IF(!source);
147 std::lock_guard l(mExternalSourceReadLock);
148 mSource = std::move(source);
149 }
150
151 processCommand();
152 }
153 }
154 }
155
processCommandandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread156 void processCommand() {
157 IStreamIn::ReadParameters rParameters;
158
159 if (!mCommandMQ.read(&rParameters)) {
160 return; // Nothing to do.
161 }
162
163 IStreamIn::ReadStatus rStatus;
164 switch (rParameters.command) {
165 case IStreamIn::ReadCommand::READ:
166 rStatus = doRead(rParameters);
167 break;
168
169 case IStreamIn::ReadCommand::GET_CAPTURE_POSITION:
170 rStatus = doGetCapturePosition();
171 break;
172
173 default:
174 ALOGE("ReadThread::%s:%d: Unknown read thread command code %d",
175 __func__, __LINE__, rParameters.command);
176 rStatus.retval = FAILURE(Result::NOT_SUPPORTED);
177 break;
178 }
179
180 rStatus.replyTo = rParameters.command;
181
182 if (!mStatusMQ.write(&rStatus)) {
183 ALOGE("ReadThread::%s:%d: status message queue write failed", __func__, __LINE__);
184 }
185
186 mEfGroup->wake(MessageQueueFlagBits::NOT_EMPTY | 0);
187 }
188
doReadandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread189 IStreamIn::ReadStatus doRead(const IStreamIn::ReadParameters &rParameters) {
190 struct MQWriter : public IWriter {
191 explicit MQWriter(DataMQ &mq) : dataMQ(mq) {}
192
193 size_t operator()(const void *dst, size_t sz) override {
194 if (dataMQ.write(static_cast<const uint8_t *>(dst), sz)) {
195 totalWritten += sz;
196 return sz;
197 } else {
198 ALOGE("WriteThread::%s:%d: DataMQ::write failed",
199 __func__, __LINE__);
200 return 0;
201 }
202 }
203
204 size_t totalWritten = 0;
205 DataMQ &dataMQ;
206 };
207
208 const size_t bytesToRead = std::min(mDataMQ.availableToWrite(),
209 static_cast<size_t>(rParameters.params.read));
210
211 MQWriter writer(mDataMQ);
212 const size_t framesLost =
213 mSource->read(mStream->getEffectiveVolume(), bytesToRead, writer);
214 if (framesLost > 0) {
215 mStream->addInputFramesLost(framesLost);
216 }
217
218 IStreamIn::ReadStatus status;
219 status.retval = Result::OK;
220 status.reply.read = writer.totalWritten;
221 return status;
222 }
223
doGetCapturePositionandroid::hardware::audio::CPP_VERSION::implementation::__anon24b2cd0a0111::ReadThread224 IStreamIn::ReadStatus doGetCapturePosition() {
225 IStreamIn::ReadStatus status;
226
227 status.retval = mSource->getCapturePosition(
228 status.reply.capturePosition.frames,
229 status.reply.capturePosition.time);
230
231 return status;
232 }
233
234 StreamIn *const mStream;
235 CommandMQ mCommandMQ;
236 StatusMQ mStatusMQ;
237 DataMQ mDataMQ;
238 std::unique_ptr<EventFlag, deleters::forEventFlag> mEfGroup;
239 std::thread mThread;
240 std::promise<pthread_t> mTid;
241 mutable std::mutex mExternalSourceReadLock; // used for external access to mSource.
242 std::unique_ptr<DevicePortSource> mSource;
243 };
244
245 } // namespace
246
StreamIn(sp<Device> dev,int32_t ioHandle,const DeviceAddress & device,const AudioConfig & config,hidl_vec<AudioInOutFlag> flags,const SinkMetadata & sinkMetadata)247 StreamIn::StreamIn(sp<Device> dev,
248 int32_t ioHandle,
249 const DeviceAddress& device,
250 const AudioConfig& config,
251 hidl_vec<AudioInOutFlag> flags,
252 const SinkMetadata& sinkMetadata)
253 : mDev(std::move(dev))
254 , mCommon(ioHandle, device, config, std::move(flags))
255 , mSinkMetadata(sinkMetadata) {
256 }
257
~StreamIn()258 StreamIn::~StreamIn() {
259 closeImpl(true);
260 }
261
getFrameSize()262 Return<uint64_t> StreamIn::getFrameSize() {
263 return mCommon.getFrameSize();
264 }
265
getFrameCount()266 Return<uint64_t> StreamIn::getFrameCount() {
267 return mCommon.getFrameCount();
268 }
269
getBufferSize()270 Return<uint64_t> StreamIn::getBufferSize() {
271 return mCommon.getBufferSize();
272 }
273
getSupportedProfiles(getSupportedProfiles_cb _hidl_cb)274 Return<void> StreamIn::getSupportedProfiles(getSupportedProfiles_cb _hidl_cb) {
275 mCommon.getSupportedProfiles(_hidl_cb);
276 return Void();
277 }
278
getAudioProperties(getAudioProperties_cb _hidl_cb)279 Return<void> StreamIn::getAudioProperties(getAudioProperties_cb _hidl_cb) {
280 mCommon.getAudioProperties(_hidl_cb);
281 return Void();
282 }
283
setAudioProperties(const AudioConfigBaseOptional & config)284 Return<Result> StreamIn::setAudioProperties(const AudioConfigBaseOptional& config) {
285 (void)config;
286 return FAILURE(Result::NOT_SUPPORTED);
287 }
288
addEffect(uint64_t effectId)289 Return<Result> StreamIn::addEffect(uint64_t effectId) {
290 (void)effectId;
291 return FAILURE(Result::INVALID_ARGUMENTS);
292 }
293
removeEffect(uint64_t effectId)294 Return<Result> StreamIn::removeEffect(uint64_t effectId) {
295 (void)effectId;
296 return FAILURE(Result::INVALID_ARGUMENTS);
297 }
298
standby()299 Return<Result> StreamIn::standby() {
300 if (mReadThread) {
301 LOG_ALWAYS_FATAL_IF(!mReadThread->standby());
302 }
303
304 return Result::OK;
305 }
306
getDevices(getDevices_cb _hidl_cb)307 Return<void> StreamIn::getDevices(getDevices_cb _hidl_cb) {
308 mCommon.getDevices(_hidl_cb);
309 return Void();
310 }
311
setDevices(const hidl_vec<DeviceAddress> & devices)312 Return<Result> StreamIn::setDevices(const hidl_vec<DeviceAddress>& devices) {
313 return mCommon.setDevices(devices);
314 }
315
getParameters(const hidl_vec<ParameterValue> & context,const hidl_vec<hidl_string> & keys,getParameters_cb _hidl_cb)316 Return<void> StreamIn::getParameters(const hidl_vec<ParameterValue>& context,
317 const hidl_vec<hidl_string>& keys,
318 getParameters_cb _hidl_cb) {
319 (void)context;
320 _hidl_cb((keys.size() > 0) ? Result::NOT_SUPPORTED : Result::OK, {});
321 return Void();
322 }
323
setParameters(const hidl_vec<ParameterValue> & context,const hidl_vec<ParameterValue> & parameters)324 Return<Result> StreamIn::setParameters(const hidl_vec<ParameterValue>& context,
325 const hidl_vec<ParameterValue>& parameters) {
326 (void)context;
327 (void)parameters;
328 return Result::OK;
329 }
330
setHwAvSync(uint32_t hwAvSync)331 Return<Result> StreamIn::setHwAvSync(uint32_t hwAvSync) {
332 (void)hwAvSync;
333 return FAILURE(Result::NOT_SUPPORTED);
334 }
335
start()336 Return<Result> StreamIn::start() {
337 return FAILURE(Result::NOT_SUPPORTED);
338 }
339
stop()340 Return<Result> StreamIn::stop() {
341 return FAILURE(Result::NOT_SUPPORTED);
342 }
343
createMmapBuffer(int32_t minSizeFrames,createMmapBuffer_cb _hidl_cb)344 Return<void> StreamIn::createMmapBuffer(int32_t minSizeFrames,
345 createMmapBuffer_cb _hidl_cb) {
346 (void)minSizeFrames;
347 _hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
348 return Void();
349 }
350
getMmapPosition(getMmapPosition_cb _hidl_cb)351 Return<void> StreamIn::getMmapPosition(getMmapPosition_cb _hidl_cb) {
352 _hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
353 return Void();
354 }
355
closeImpl(const bool fromDctor)356 Result StreamIn::closeImpl(const bool fromDctor) {
357 if (mDev) {
358 mReadThread.reset();
359 mDev->unrefDevice(this);
360 mDev = nullptr;
361 return Result::OK;
362 } else if (fromDctor) {
363 // closeImpl is always called from the dctor, it is ok if mDev is null,
364 // we don't want to log the error in this case.
365 return Result::OK;
366 } else {
367 return FAILURE(Result::INVALID_STATE);
368 }
369 }
370
close()371 Return<Result> StreamIn::close() {
372 return closeImpl(false);
373 }
374
getAudioSource(getAudioSource_cb _hidl_cb)375 Return<void> StreamIn::getAudioSource(getAudioSource_cb _hidl_cb) {
376 _hidl_cb(FAILURE(Result::NOT_SUPPORTED), {});
377 return Void();
378 }
379
setGain(float gain)380 Return<Result> StreamIn::setGain(float gain) {
381 (void)gain;
382 return FAILURE(Result::NOT_SUPPORTED);
383 }
384
updateSinkMetadata(const SinkMetadata & sinkMetadata)385 Return<Result> StreamIn::updateSinkMetadata(const SinkMetadata& sinkMetadata) {
386 (void)sinkMetadata;
387 return FAILURE(Result::NOT_SUPPORTED);
388 }
389
prepareForReading(uint32_t frameSize,uint32_t framesCount,prepareForReading_cb _hidl_cb)390 Return<void> StreamIn::prepareForReading(uint32_t frameSize,
391 uint32_t framesCount,
392 prepareForReading_cb _hidl_cb) {
393 if (!frameSize || !framesCount || frameSize > 256 || framesCount > (1u << 20)) {
394 _hidl_cb(FAILURE(Result::INVALID_ARGUMENTS), {}, {}, {}, -1);
395 return Void();
396 }
397
398 if (mReadThread) { // INVALID_STATE if the method was already called.
399 _hidl_cb(FAILURE(Result::INVALID_STATE), {}, {}, {}, -1);
400 return Void();
401 }
402
403 auto t = std::make_unique<ReadThread>(this, frameSize * framesCount);
404
405 if (t->isRunning()) {
406 const auto [commandDesc, dataDesc, statusDesc ] = t->getDescriptors();
407 _hidl_cb(Result::OK,
408 *commandDesc,
409 *dataDesc,
410 *statusDesc,
411 t->getTid().get());
412
413 mReadThread = std::move(t);
414 } else {
415 _hidl_cb(FAILURE(Result::INVALID_ARGUMENTS), {}, {}, {}, -1);
416 }
417
418 return Void();
419 }
420
getInputFramesLost()421 Return<uint32_t> StreamIn::getInputFramesLost() {
422 return 0;
423 }
424
getCapturePosition(getCapturePosition_cb _hidl_cb)425 Return<void> StreamIn::getCapturePosition(getCapturePosition_cb _hidl_cb) {
426 const auto rt = static_cast<ReadThread*>(mReadThread.get());
427 if (!rt) {
428 _hidl_cb(FAILURE(Result::INVALID_STATE), {}, {});
429 return Void();
430 }
431
432 uint64_t frames{};
433 uint64_t time{};
434 const Result r = rt->getCapturePosition(frames, time);
435 _hidl_cb(r, frames, time);
436 return Void();
437 }
438
getActiveMicrophones(getActiveMicrophones_cb _hidl_cb)439 Return<void> StreamIn::getActiveMicrophones(getActiveMicrophones_cb _hidl_cb) {
440 _hidl_cb(Result::OK, {util::getMicrophoneInfo()});
441 return Void();
442 }
443
setMicrophoneDirection(MicrophoneDirection direction)444 Return<Result> StreamIn::setMicrophoneDirection(MicrophoneDirection direction) {
445 (void)direction;
446 return Result::NOT_SUPPORTED;
447 }
448
setMicrophoneFieldDimension(float zoom)449 Return<Result> StreamIn::setMicrophoneFieldDimension(float zoom) {
450 (void)zoom;
451 return Result::NOT_SUPPORTED;
452 }
453
setMicMute(bool mute)454 void StreamIn::setMicMute(bool mute) {
455 mEffectiveVolume =
456 (mute && (xsd::stringToAudioDevice(getDeviceAddress().deviceType) ==
457 xsd::AudioDevice::AUDIO_DEVICE_IN_BUILTIN_MIC))
458 ? 0.0f : 1.0f;
459 }
460
validateDeviceAddress(const DeviceAddress & device)461 bool StreamIn::validateDeviceAddress(const DeviceAddress& device) {
462 return DevicePortSource::validateDeviceAddress(device);
463 }
464
validateFlags(const hidl_vec<AudioInOutFlag> & flags)465 bool StreamIn::validateFlags(const hidl_vec<AudioInOutFlag>& flags) {
466 return std::all_of(flags.begin(), flags.end(), [](const AudioInOutFlag& flag){
467 return xsd::stringToAudioInOutFlag(flag) != xsd::AudioInOutFlag::UNKNOWN;
468 });
469 }
470
validateSinkMetadata(const SinkMetadata & sinkMetadata)471 bool StreamIn::validateSinkMetadata(const SinkMetadata& sinkMetadata) {
472 for (const auto& track : sinkMetadata.tracks) {
473 if (xsd::isUnknownAudioSource(track.source)
474 || xsd::isUnknownAudioChannelMask(track.channelMask)) {
475 return false;
476 }
477 for (const auto& tag : track.tags) {
478 if (!xsd::isVendorExtension(tag)) {
479 return false;
480 }
481 }
482 }
483 return true;
484 }
485
486 } // namespace implementation
487 } // namespace CPP_VERSION
488 } // namespace audio
489 } // namespace hardware
490 } // namespace android
491