blob: b444423738ad08d0d5280702ff0e6af9255a7962 [file] [log] [blame]
Mikhail Naganov10548292016-10-31 10:39:47 -07001/*
2 * Copyright (C) 2016 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#define LOG_TAG "StreamInHAL"
Mikhail Naganovb29438e2016-12-22 09:21:34 -080018//#define LOG_NDEBUG 0
Mikhail Naganovb0abafb2017-01-31 17:24:48 -080019#define ATRACE_TAG ATRACE_TAG_AUDIO
Mikhail Naganov10548292016-10-31 10:39:47 -070020
Yifan Hongf9d30342016-11-30 13:45:34 -080021#include <android/log.h>
Mikhail Naganovb29438e2016-12-22 09:21:34 -080022#include <hardware/audio.h>
23#include <mediautils/SchedulingPolicyService.h>
Mikhail Naganovb0abafb2017-01-31 17:24:48 -080024#include <utils/Trace.h>
Mikhail Naganov10548292016-10-31 10:39:47 -070025
26#include "StreamIn.h"
27
Mikhail Naganovb29438e2016-12-22 09:21:34 -080028using ::android::hardware::audio::V2_0::MessageQueueFlagBits;
29
Mikhail Naganov10548292016-10-31 10:39:47 -070030namespace android {
31namespace hardware {
32namespace audio {
33namespace V2_0 {
34namespace implementation {
35
Mikhail Naganovb29438e2016-12-22 09:21:34 -080036namespace {
37
38class ReadThread : public Thread {
39 public:
40 // ReadThread's lifespan never exceeds StreamIn's lifespan.
41 ReadThread(std::atomic<bool>* stop,
42 audio_stream_in_t* stream,
43 StreamIn::DataMQ* dataMQ,
44 StreamIn::StatusMQ* statusMQ,
45 EventFlag* efGroup,
46 ThreadPriority threadPriority)
47 : Thread(false /*canCallJava*/),
48 mStop(stop),
49 mStream(stream),
50 mDataMQ(dataMQ),
51 mStatusMQ(statusMQ),
52 mEfGroup(efGroup),
53 mThreadPriority(threadPriority),
54 mBuffer(new uint8_t[dataMQ->getQuantumCount()]) {
55 }
56 virtual ~ReadThread() {}
57
58 status_t readyToRun() override;
59
60 private:
61 std::atomic<bool>* mStop;
62 audio_stream_in_t* mStream;
63 StreamIn::DataMQ* mDataMQ;
64 StreamIn::StatusMQ* mStatusMQ;
65 EventFlag* mEfGroup;
66 ThreadPriority mThreadPriority;
67 std::unique_ptr<uint8_t[]> mBuffer;
68
69 bool threadLoop() override;
70};
71
72status_t ReadThread::readyToRun() {
73 if (mThreadPriority != ThreadPriority::NORMAL) {
74 int err = requestPriority(
75 getpid(), getTid(), static_cast<int>(mThreadPriority), true /*asynchronous*/);
76 ALOGW_IF(err, "failed to set priority %d for pid %d tid %d; error %d",
77 static_cast<int>(mThreadPriority), getpid(), getTid(), err);
78 }
79 return OK;
80}
81
82bool ReadThread::threadLoop() {
83 // This implementation doesn't return control back to the Thread until it decides to stop,
84 // as the Thread uses mutexes, and this can lead to priority inversion.
85 while(!std::atomic_load_explicit(mStop, std::memory_order_acquire)) {
86 // TODO: Remove manual event flag handling once blocking MQ is implemented. b/33815422
87 uint32_t efState = 0;
88 mEfGroup->wait(static_cast<uint32_t>(MessageQueueFlagBits::NOT_FULL), &efState, NS_PER_SEC);
89 if (!(efState & static_cast<uint32_t>(MessageQueueFlagBits::NOT_FULL))) {
90 continue; // Nothing to do.
91 }
92
93 const size_t availToWrite = mDataMQ->availableToWrite();
94 ssize_t readResult = mStream->read(mStream, &mBuffer[0], availToWrite);
95 Result retval = Result::OK;
96 uint64_t read = 0;
97 if (readResult >= 0) {
98 read = readResult;
99 if (!mDataMQ->write(&mBuffer[0], readResult)) {
100 ALOGW("data message queue write failed");
101 }
102 } else {
103 retval = Stream::analyzeStatus("read", readResult);
104 }
105 IStreamIn::ReadStatus status = { retval, read };
106 if (!mStatusMQ->write(&status)) {
107 ALOGW("status message queue write failed");
108 }
109 mEfGroup->wake(static_cast<uint32_t>(MessageQueueFlagBits::NOT_EMPTY));
110 }
111
112 return false;
113}
114
115} // namespace
116
Mikhail Naganov10548292016-10-31 10:39:47 -0700117StreamIn::StreamIn(audio_hw_device_t* device, audio_stream_in_t* stream)
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800118 : mIsClosed(false), mDevice(device), mStream(stream),
Eric Laurent7deb7da2016-12-15 19:15:45 -0800119 mStreamCommon(new Stream(&stream->common)),
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800120 mStreamMmap(new StreamMmap<audio_stream_in_t>(stream)),
121 mEfGroup(nullptr), mStopReadThread(false) {
Mikhail Naganov10548292016-10-31 10:39:47 -0700122}
123
124StreamIn::~StreamIn() {
Mikhail Naganovb0abafb2017-01-31 17:24:48 -0800125 ATRACE_CALL();
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800126 close();
Mikhail Naganovb0abafb2017-01-31 17:24:48 -0800127 if (mReadThread.get()) {
128 ATRACE_NAME("mReadThread->join");
129 status_t status = mReadThread->join();
130 ALOGE_IF(status, "read thread exit error: %s", strerror(-status));
131 }
132 if (mEfGroup) {
133 status_t status = EventFlag::deleteEventFlag(&mEfGroup);
134 ALOGE_IF(status, "read MQ event flag deletion error: %s", strerror(-status));
135 }
136 mDevice->close_input_stream(mDevice, mStream);
Mikhail Naganov10548292016-10-31 10:39:47 -0700137 mStream = nullptr;
138 mDevice = nullptr;
139}
140
141// Methods from ::android::hardware::audio::V2_0::IStream follow.
142Return<uint64_t> StreamIn::getFrameSize() {
143 return audio_stream_in_frame_size(mStream);
144}
145
146Return<uint64_t> StreamIn::getFrameCount() {
147 return mStreamCommon->getFrameCount();
148}
149
150Return<uint64_t> StreamIn::getBufferSize() {
151 return mStreamCommon->getBufferSize();
152}
153
154Return<uint32_t> StreamIn::getSampleRate() {
155 return mStreamCommon->getSampleRate();
156}
157
158Return<void> StreamIn::getSupportedSampleRates(getSupportedSampleRates_cb _hidl_cb) {
159 return mStreamCommon->getSupportedSampleRates(_hidl_cb);
160}
161
162Return<Result> StreamIn::setSampleRate(uint32_t sampleRateHz) {
163 return mStreamCommon->setSampleRate(sampleRateHz);
164}
165
166Return<AudioChannelMask> StreamIn::getChannelMask() {
167 return mStreamCommon->getChannelMask();
168}
169
170Return<void> StreamIn::getSupportedChannelMasks(getSupportedChannelMasks_cb _hidl_cb) {
171 return mStreamCommon->getSupportedChannelMasks(_hidl_cb);
172}
173
174Return<Result> StreamIn::setChannelMask(AudioChannelMask mask) {
175 return mStreamCommon->setChannelMask(mask);
176}
177
178Return<AudioFormat> StreamIn::getFormat() {
179 return mStreamCommon->getFormat();
180}
181
182Return<void> StreamIn::getSupportedFormats(getSupportedFormats_cb _hidl_cb) {
183 return mStreamCommon->getSupportedFormats(_hidl_cb);
184}
185
186Return<Result> StreamIn::setFormat(AudioFormat format) {
187 return mStreamCommon->setFormat(format);
188}
189
190Return<void> StreamIn::getAudioProperties(getAudioProperties_cb _hidl_cb) {
191 return mStreamCommon->getAudioProperties(_hidl_cb);
192}
193
194Return<Result> StreamIn::addEffect(uint64_t effectId) {
195 return mStreamCommon->addEffect(effectId);
196}
197
198Return<Result> StreamIn::removeEffect(uint64_t effectId) {
199 return mStreamCommon->removeEffect(effectId);
200}
201
202Return<Result> StreamIn::standby() {
203 return mStreamCommon->standby();
204}
205
206Return<AudioDevice> StreamIn::getDevice() {
207 return mStreamCommon->getDevice();
208}
209
210Return<Result> StreamIn::setDevice(const DeviceAddress& address) {
211 return mStreamCommon->setDevice(address);
212}
213
214Return<Result> StreamIn::setConnectedState(const DeviceAddress& address, bool connected) {
215 return mStreamCommon->setConnectedState(address, connected);
216}
217
218Return<Result> StreamIn::setHwAvSync(uint32_t hwAvSync) {
219 return mStreamCommon->setHwAvSync(hwAvSync);
220}
221
222Return<void> StreamIn::getParameters(const hidl_vec<hidl_string>& keys, getParameters_cb _hidl_cb) {
223 return mStreamCommon->getParameters(keys, _hidl_cb);
224}
225
226Return<Result> StreamIn::setParameters(const hidl_vec<ParameterValue>& parameters) {
227 return mStreamCommon->setParameters(parameters);
228}
229
Martijn Coenen70b9a152016-11-18 15:29:32 +0100230Return<void> StreamIn::debugDump(const hidl_handle& fd) {
Mikhail Naganov10548292016-10-31 10:39:47 -0700231 return mStreamCommon->debugDump(fd);
232}
233
Eric Laurent7deb7da2016-12-15 19:15:45 -0800234Return<Result> StreamIn::start() {
235 return mStreamMmap->start();
236}
237
238Return<Result> StreamIn::stop() {
239 return mStreamMmap->stop();
240}
241
242Return<void> StreamIn::createMmapBuffer(int32_t minSizeFrames, createMmapBuffer_cb _hidl_cb) {
243 return mStreamMmap->createMmapBuffer(
244 minSizeFrames, audio_stream_in_frame_size(mStream), _hidl_cb);
245}
246
247Return<void> StreamIn::getMmapPosition(getMmapPosition_cb _hidl_cb) {
248 return mStreamMmap->getMmapPosition(_hidl_cb);
249}
Mikhail Naganov10548292016-10-31 10:39:47 -0700250
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800251Return<Result> StreamIn::close() {
252 if (mIsClosed) return Result::INVALID_STATE;
253 mIsClosed = true;
254 if (mReadThread.get()) {
255 mStopReadThread.store(true, std::memory_order_release);
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800256 }
257 if (mEfGroup) {
Mikhail Naganovb0abafb2017-01-31 17:24:48 -0800258 mEfGroup->wake(static_cast<uint32_t>(MessageQueueFlagBits::NOT_FULL));
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800259 }
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800260 return Result::OK;
261}
262
Mikhail Naganov10548292016-10-31 10:39:47 -0700263// Methods from ::android::hardware::audio::V2_0::IStreamIn follow.
264Return<void> StreamIn::getAudioSource(getAudioSource_cb _hidl_cb) {
265 int halSource;
266 Result retval = mStreamCommon->getParam(AudioParameter::keyInputSource, &halSource);
267 AudioSource source(AudioSource::DEFAULT);
268 if (retval == Result::OK) {
269 source = AudioSource(halSource);
270 }
271 _hidl_cb(retval, source);
272 return Void();
273}
274
275Return<Result> StreamIn::setGain(float gain) {
Eric Laurent7deb7da2016-12-15 19:15:45 -0800276 return Stream::analyzeStatus("set_gain", mStream->set_gain(mStream, gain));
Mikhail Naganov10548292016-10-31 10:39:47 -0700277}
278
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800279Return<void> StreamIn::prepareForReading(
280 uint32_t frameSize, uint32_t framesCount, ThreadPriority threadPriority,
281 prepareForReading_cb _hidl_cb) {
282 status_t status;
283 // Create message queues.
284 if (mDataMQ) {
285 ALOGE("the client attempts to call prepareForReading twice");
286 _hidl_cb(Result::INVALID_STATE,
Hridya Valsaraju790db102017-01-10 08:58:23 -0800287 DataMQ::Descriptor(), StatusMQ::Descriptor());
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800288 return Void();
Mikhail Naganov10548292016-10-31 10:39:47 -0700289 }
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800290 std::unique_ptr<DataMQ> tempDataMQ(
291 new DataMQ(frameSize * framesCount, true /* EventFlag */));
292 std::unique_ptr<StatusMQ> tempStatusMQ(new StatusMQ(1));
293 if (!tempDataMQ->isValid() || !tempStatusMQ->isValid()) {
294 ALOGE_IF(!tempDataMQ->isValid(), "data MQ is invalid");
295 ALOGE_IF(!tempStatusMQ->isValid(), "status MQ is invalid");
296 _hidl_cb(Result::INVALID_ARGUMENTS,
Hridya Valsaraju790db102017-01-10 08:58:23 -0800297 DataMQ::Descriptor(), StatusMQ::Descriptor());
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800298 return Void();
299 }
300 // TODO: Remove event flag management once blocking MQ is implemented. b/33815422
301 status = EventFlag::createEventFlag(tempDataMQ->getEventFlagWord(), &mEfGroup);
302 if (status != OK || !mEfGroup) {
303 ALOGE("failed creating event flag for data MQ: %s", strerror(-status));
304 _hidl_cb(Result::INVALID_ARGUMENTS,
Hridya Valsaraju790db102017-01-10 08:58:23 -0800305 DataMQ::Descriptor(), StatusMQ::Descriptor());
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800306 return Void();
307 }
308
309 // Create and launch the thread.
310 mReadThread = new ReadThread(
311 &mStopReadThread,
312 mStream,
313 tempDataMQ.get(),
314 tempStatusMQ.get(),
315 mEfGroup,
316 threadPriority);
317 status = mReadThread->run("reader", PRIORITY_URGENT_AUDIO);
318 if (status != OK) {
319 ALOGW("failed to start reader thread: %s", strerror(-status));
320 _hidl_cb(Result::INVALID_ARGUMENTS,
Hridya Valsaraju790db102017-01-10 08:58:23 -0800321 DataMQ::Descriptor(), StatusMQ::Descriptor());
Mikhail Naganovb29438e2016-12-22 09:21:34 -0800322 return Void();
323 }
324
325 mDataMQ = std::move(tempDataMQ);
326 mStatusMQ = std::move(tempStatusMQ);
327 _hidl_cb(Result::OK, *mDataMQ->getDesc(), *mStatusMQ->getDesc());
Mikhail Naganov10548292016-10-31 10:39:47 -0700328 return Void();
329}
330
331Return<uint32_t> StreamIn::getInputFramesLost() {
332 return mStream->get_input_frames_lost(mStream);
333}
334
335Return<void> StreamIn::getCapturePosition(getCapturePosition_cb _hidl_cb) {
336 Result retval(Result::NOT_SUPPORTED);
337 uint64_t frames = 0, time = 0;
338 if (mStream->get_capture_position != NULL) {
339 int64_t halFrames, halTime;
Eric Laurent7deb7da2016-12-15 19:15:45 -0800340 retval = Stream::analyzeStatus(
Mikhail Naganov10548292016-10-31 10:39:47 -0700341 "get_capture_position",
Mikhail Naganovee901e32017-01-12 09:28:52 -0800342 mStream->get_capture_position(mStream, &halFrames, &halTime),
343 // HAL may have a stub function, always returning ENOSYS, don't
344 // spam the log in this case.
345 ENOSYS);
Mikhail Naganov10548292016-10-31 10:39:47 -0700346 if (retval == Result::OK) {
347 frames = halFrames;
348 time = halTime;
349 }
350 }
351 _hidl_cb(retval, frames, time);
352 return Void();
353}
354
355} // namespace implementation
356} // namespace V2_0
357} // namespace audio
358} // namespace hardware
359} // namespace android