• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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