xref: /aosp_15_r20/frameworks/av/services/oboeservice/AAudioServiceEndpointMMAP.cpp (revision ec779b8e0859a360c3d303172224686826e6e0e1)
1 /*
2  * Copyright (C) 2017 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 "AAudioServiceEndpointMMAP"
18 //#define LOG_NDEBUG 0
19 #include <utils/Log.h>
20 
21 #include <algorithm>
22 #include <assert.h>
23 #include <map>
24 #include <mutex>
25 #include <set>
26 #include <sstream>
27 #include <thread>
28 #include <utils/Singleton.h>
29 #include <vector>
30 
31 #include "AAudioEndpointManager.h"
32 #include "AAudioServiceEndpoint.h"
33 
34 #include "core/AudioStreamBuilder.h"
35 #include "AAudioServiceEndpoint.h"
36 #include "AAudioServiceStreamShared.h"
37 #include "AAudioServiceEndpointPlay.h"
38 #include "AAudioServiceEndpointMMAP.h"
39 
40 #include <com_android_media_aaudio.h>
41 
42 #define AAUDIO_BUFFER_CAPACITY_MIN    (4 * 512)
43 #define AAUDIO_SAMPLE_RATE_DEFAULT    48000
44 
45 // This is an estimate of the time difference between the HW and the MMAP time.
46 // TODO Get presentation timestamps from the HAL instead of using these estimates.
47 #define OUTPUT_ESTIMATED_HARDWARE_OFFSET_NANOS  (3 * AAUDIO_NANOS_PER_MILLISECOND)
48 #define INPUT_ESTIMATED_HARDWARE_OFFSET_NANOS   (-1 * AAUDIO_NANOS_PER_MILLISECOND)
49 
50 #define AAUDIO_MAX_OPEN_ATTEMPTS    10
51 
52 using namespace android;  // TODO just import names needed
53 using namespace aaudio;   // TODO just import names needed
54 
AAudioServiceEndpointMMAP(AAudioService & audioService)55 AAudioServiceEndpointMMAP::AAudioServiceEndpointMMAP(AAudioService &audioService)
56         : mMmapStream(nullptr)
57         , mAAudioService(audioService) {}
58 
dump() const59 std::string AAudioServiceEndpointMMAP::dump() const {
60     std::stringstream result;
61 
62     result << "  MMAP: framesTransferred = " << mFramesTransferred.get();
63     result << ", HW nanos = " << mHardwareTimeOffsetNanos;
64     result << ", port handle = " << mPortHandle;
65     result << ", audio data FD = " << mAudioDataWrapper->getDataFileDescriptor();
66     result << "\n";
67 
68     result << "    HW Offset Micros:     " <<
69                                       (getHardwareTimeOffsetNanos()
70                                        / AAUDIO_NANOS_PER_MICROSECOND) << "\n";
71 
72     result << AAudioServiceEndpoint::dump();
73     return result.str();
74 }
75 
76 namespace {
77 
78 const static std::map<audio_format_t, audio_format_t> NEXT_FORMAT_TO_TRY = {
79         {AUDIO_FORMAT_PCM_FLOAT,         AUDIO_FORMAT_PCM_32_BIT},
80         {AUDIO_FORMAT_PCM_32_BIT,        AUDIO_FORMAT_PCM_24_BIT_PACKED},
81         {AUDIO_FORMAT_PCM_24_BIT_PACKED, AUDIO_FORMAT_PCM_8_24_BIT},
82         {AUDIO_FORMAT_PCM_8_24_BIT,      AUDIO_FORMAT_PCM_16_BIT}
83 };
84 
getNextFormatToTry(audio_format_t curFormat)85 audio_format_t getNextFormatToTry(audio_format_t curFormat) {
86     const auto it = NEXT_FORMAT_TO_TRY.find(curFormat);
87     return it != NEXT_FORMAT_TO_TRY.end() ? it->second : curFormat;
88 }
89 
90 struct configComp {
operator ()__anon112ed0c40111::configComp91     bool operator() (const audio_config_base_t& lhs, const audio_config_base_t& rhs) const {
92         if (lhs.sample_rate != rhs.sample_rate) {
93             return lhs.sample_rate < rhs.sample_rate;
94         } else if (lhs.channel_mask != rhs.channel_mask) {
95             return lhs.channel_mask < rhs.channel_mask;
96         } else {
97             return lhs.format < rhs.format;
98         }
99     }
100 };
101 
102 } // namespace
103 
open(const aaudio::AAudioStreamRequest & request)104 aaudio_result_t AAudioServiceEndpointMMAP::open(const aaudio::AAudioStreamRequest &request) {
105     aaudio_result_t result = AAUDIO_OK;
106     mAudioDataWrapper = std::make_unique<SharedMemoryWrapper>();
107     copyFrom(request.getConstantConfiguration());
108     mRequestedDeviceId = android::getFirstDeviceId(getDeviceIds());
109 
110     mMmapClient.attributionSource = request.getAttributionSource();
111     // TODO b/182392769: use attribution source util
112     mMmapClient.attributionSource.uid = VALUE_OR_FATAL(
113         legacy2aidl_uid_t_int32_t(IPCThreadState::self()->getCallingUid()));
114     mMmapClient.attributionSource.pid = VALUE_OR_FATAL(
115         legacy2aidl_pid_t_int32_t(IPCThreadState::self()->getCallingPid()));
116 
117     audio_format_t audioFormat = getFormat();
118     int32_t sampleRate = getSampleRate();
119     if (sampleRate == AAUDIO_UNSPECIFIED) {
120         sampleRate = AAUDIO_SAMPLE_RATE_DEFAULT;
121     }
122 
123     const aaudio_direction_t direction = getDirection();
124     audio_config_base_t config;
125     config.format = audioFormat;
126     config.sample_rate = sampleRate;
127     config.channel_mask = AAudio_getChannelMaskForOpen(
128             getChannelMask(), getSamplesPerFrame(), direction == AAUDIO_DIRECTION_INPUT);
129 
130     std::set<audio_config_base_t, configComp> configsTried;
131     int32_t numberOfAttempts = 0;
132     while (numberOfAttempts < AAUDIO_MAX_OPEN_ATTEMPTS) {
133         if (configsTried.find(config) != configsTried.end()) {
134             // APM returning something that has already tried.
135             ALOGW("Have already tried to open with format=%#x and sr=%d, but failed before",
136                   config.format, config.sample_rate);
137             break;
138         }
139         configsTried.insert(config);
140 
141         audio_config_base_t previousConfig = config;
142         result = openWithConfig(&config);
143         if (result != AAUDIO_ERROR_UNAVAILABLE) {
144             // Return if it is successful or there is an error that is not
145             // AAUDIO_ERROR_UNAVAILABLE happens.
146             ALOGI("Opened format=%#x sr=%d, with result=%d", previousConfig.format,
147                     previousConfig.sample_rate, result);
148             break;
149         }
150 
151         // Try other formats if the config from APM is the same as our current config.
152         // Some HALs may report its format support incorrectly.
153         if ((previousConfig.format == config.format) &&
154                 (previousConfig.sample_rate == config.sample_rate)) {
155             config.format = getNextFormatToTry(config.format);
156         }
157 
158         ALOGD("%s() %#x %d failed, perhaps due to format or sample rate. Try again with %#x %d",
159                 __func__, previousConfig.format, previousConfig.sample_rate, config.format,
160                 config.sample_rate);
161         numberOfAttempts++;
162     }
163     return result;
164 }
165 
openWithConfig(audio_config_base_t * config)166 aaudio_result_t AAudioServiceEndpointMMAP::openWithConfig(
167         audio_config_base_t* config) {
168     aaudio_result_t result = AAUDIO_OK;
169     audio_config_base_t currentConfig = *config;
170     android::DeviceIdVector deviceIds;
171 
172     const audio_attributes_t attributes = getAudioAttributesFrom(this);
173 
174     if (mRequestedDeviceId != AAUDIO_UNSPECIFIED) {
175         deviceIds.push_back(mRequestedDeviceId);
176     }
177 
178     const aaudio_direction_t direction = getDirection();
179 
180     if (direction == AAUDIO_DIRECTION_OUTPUT) {
181         mHardwareTimeOffsetNanos = OUTPUT_ESTIMATED_HARDWARE_OFFSET_NANOS; // frames at DAC later
182 
183     } else if (direction == AAUDIO_DIRECTION_INPUT) {
184         mHardwareTimeOffsetNanos = INPUT_ESTIMATED_HARDWARE_OFFSET_NANOS; // frames at ADC earlier
185 
186     } else {
187         ALOGE("%s() invalid direction = %d", __func__, direction);
188         return AAUDIO_ERROR_ILLEGAL_ARGUMENT;
189     }
190 
191     const MmapStreamInterface::stream_direction_t streamDirection =
192             (direction == AAUDIO_DIRECTION_OUTPUT)
193             ? MmapStreamInterface::DIRECTION_OUTPUT
194             : MmapStreamInterface::DIRECTION_INPUT;
195 
196     const aaudio_session_id_t requestedSessionId = getSessionId();
197     audio_session_t sessionId = AAudioConvert_aaudioToAndroidSessionId(requestedSessionId);
198 
199     // Open HAL stream. Set mMmapStream
200     ALOGD("%s trying to open MMAP stream with format=%#x, "
201           "sample_rate=%u, channel_mask=%#x, device=%s",
202           __func__, config->format, config->sample_rate,
203           config->channel_mask, android::toString(deviceIds).c_str());
204 
205     const std::lock_guard<std::mutex> lock(mMmapStreamLock);
206     const status_t status = MmapStreamInterface::openMmapStream(streamDirection,
207                                                                 &attributes,
208                                                                 config,
209                                                                 mMmapClient,
210                                                                 &deviceIds,
211                                                                 &sessionId,
212                                                                 this, // callback
213                                                                 mMmapStream,
214                                                                 &mPortHandle);
215     ALOGD("%s() mMapClient.attributionSource = %s => portHandle = %d\n",
216           __func__, mMmapClient.attributionSource.toString().c_str(), mPortHandle);
217     if (status != OK) {
218         // This can happen if the resource is busy or the config does
219         // not match the hardware.
220         ALOGD("%s() - openMmapStream() returned status=%d, suggested format=%#x, sample_rate=%u, "
221               "channel_mask=%#x",
222               __func__, status, config->format, config->sample_rate, config->channel_mask);
223         // Keep the channel mask of the current config
224         config->channel_mask = currentConfig.channel_mask;
225         return AAUDIO_ERROR_UNAVAILABLE;
226     }
227 
228     if (deviceIds.empty()) {
229         ALOGW("%s() - openMmapStream() failed to set deviceIds", __func__);
230     }
231     setDeviceIds(deviceIds);
232 
233     if (sessionId == AUDIO_SESSION_ALLOCATE) {
234         ALOGW("%s() - openMmapStream() failed to set sessionId", __func__);
235     }
236 
237     const aaudio_session_id_t actualSessionId =
238             (requestedSessionId == AAUDIO_SESSION_ID_NONE)
239             ? AAUDIO_SESSION_ID_NONE
240             : (aaudio_session_id_t) sessionId;
241     setSessionId(actualSessionId);
242 
243     ALOGD("%s(format = 0x%X) deviceIds = %s, sessionId = %d",
244           __func__, config->format, toString(getDeviceIds()).c_str(), getSessionId());
245 
246     // Create MMAP/NOIRQ buffer.
247     result = createMmapBuffer_l();
248     if (result != AAUDIO_OK) {
249         goto error;
250     }
251 
252     // Get information about the stream and pass it back to the caller.
253     setChannelMask(AAudioConvert_androidToAAudioChannelMask(
254             config->channel_mask, getDirection() == AAUDIO_DIRECTION_INPUT,
255             AAudio_isChannelIndexMask(config->channel_mask)));
256 
257     setFormat(config->format);
258     setSampleRate(config->sample_rate);
259     setHardwareSampleRate(getSampleRate());
260     setHardwareFormat(getFormat());
261     setHardwareSamplesPerFrame(AAudioConvert_channelMaskToCount(getChannelMask()));
262 
263     // If the position is not updated while the timestamp is updated for more than a certain amount,
264     // the timestamp reported from the HAL may not be accurate. Here, a timestamp grace period is
265     // set as 5 burst size. We may want to update this value if there is any report from OEMs saying
266     // that is too short.
267     static constexpr int kTimestampGraceBurstCount = 5;
268     mTimestampGracePeriodMs = ((int64_t) kTimestampGraceBurstCount * mFramesPerBurst
269             * AAUDIO_MILLIS_PER_SECOND) / getSampleRate();
270 
271     mDataReportOffsetNanos = ((int64_t)mTimestampGracePeriodMs) * AAUDIO_NANOS_PER_MILLISECOND;
272 
273     ALOGD("%s() got rate = %d, channels = %d channelMask = %#x, deviceIds = %s, capacity = %d\n",
274           __func__, getSampleRate(), getSamplesPerFrame(), getChannelMask(),
275           android::toString(deviceIds).c_str(), getBufferCapacity());
276 
277     ALOGD("%s() got format = 0x%X = %s, frame size = %d, burst size = %d",
278           __func__, getFormat(), audio_format_to_string(getFormat()),
279           calculateBytesPerFrame(), mFramesPerBurst);
280 
281     return result;
282 
283 error:
284     close_l();
285     // restore original requests
286     android::DeviceIdVector requestedDeviceIds;
287     if (mRequestedDeviceId != AAUDIO_UNSPECIFIED) {
288         requestedDeviceIds.push_back(mRequestedDeviceId);
289     }
290     setDeviceIds(requestedDeviceIds);
291     setSessionId(requestedSessionId);
292     return result;
293 }
294 
close()295 void AAudioServiceEndpointMMAP::close() {
296     bool closedIt = false;
297     {
298         const std::lock_guard<std::mutex> lock(mMmapStreamLock);
299         closedIt = close_l();
300     }
301     if (closedIt) {
302         // TODO Why is this needed?
303         AudioClock::sleepForNanos(100 * AAUDIO_NANOS_PER_MILLISECOND);
304     }
305 }
306 
close_l()307 bool AAudioServiceEndpointMMAP::close_l() { // requires mMmapStreamLock
308     bool closedIt = false;
309     if (mMmapStream != nullptr) {
310         // Needs to be explicitly cleared or CTS will fail but it is not clear why.
311         ALOGD("%s() clear mMmapStream", __func__);
312         mMmapStream.clear();
313         closedIt = true;
314     }
315     return closedIt;
316 }
317 
startStream(sp<AAudioServiceStreamBase> stream,audio_port_handle_t * clientHandle __unused)318 aaudio_result_t AAudioServiceEndpointMMAP::startStream(sp<AAudioServiceStreamBase> stream,
319                                                    audio_port_handle_t *clientHandle __unused) {
320     // Start the client on behalf of the AAudio service.
321     // Use the port handle that was provided by openMmapStream().
322     audio_port_handle_t tempHandle = mPortHandle;
323     audio_attributes_t attr = {};
324     if (stream != nullptr) {
325         attr = getAudioAttributesFrom(stream.get());
326     }
327     const aaudio_result_t result = startClient(
328             mMmapClient, stream == nullptr ? nullptr : &attr, &tempHandle);
329     // When AudioFlinger is passed a valid port handle then it should not change it.
330     LOG_ALWAYS_FATAL_IF(tempHandle != mPortHandle,
331                         "%s() port handle not expected to change from %d to %d",
332                         __func__, mPortHandle, tempHandle);
333     ALOGV("%s() mPortHandle = %d", __func__, mPortHandle);
334     return result;
335 }
336 
stopStream(sp<AAudioServiceStreamBase>,audio_port_handle_t clientHandle)337 aaudio_result_t AAudioServiceEndpointMMAP::stopStream(sp<AAudioServiceStreamBase> /*stream*/,
338                                                       audio_port_handle_t clientHandle) {
339     mFramesTransferred.reset32();
340 
341     // Round 64-bit counter up to a multiple of the buffer capacity.
342     // This is required because the 64-bit counter is used as an index
343     // into a circular buffer and the actual HW position is reset to zero
344     // when the stream is stopped.
345     mFramesTransferred.roundUp64(getBufferCapacity());
346 
347     // Use the port handle that was provided by openMmapStream().
348     aaudio_result_t result = stopClient(mPortHandle);
349     ALOGD("%s(%d): called stopClient(%d=mPortHandle), returning %d", __func__,
350           (int)clientHandle, mPortHandle, result);
351     return result;
352 }
353 
startClient(const android::AudioClient & client,const audio_attributes_t * attr,audio_port_handle_t * portHandlePtr)354 aaudio_result_t AAudioServiceEndpointMMAP::startClient(const android::AudioClient& client,
355                                                        const audio_attributes_t *attr,
356                                                        audio_port_handle_t *portHandlePtr) {
357     const std::lock_guard<std::mutex> lock(mMmapStreamLock);
358     if (mMmapStream == nullptr) {
359         ALOGW("%s(): called after mMmapStream set to NULL", __func__);
360         return AAUDIO_ERROR_NULL;
361     } else if (!isConnected()) {
362         ALOGD("%s(): MMAP stream was disconnected", __func__);
363         return AAUDIO_ERROR_DISCONNECTED;
364     } else {
365         aaudio_result_t result = AAudioConvert_androidToAAudioResult(
366                 mMmapStream->start(client, attr, portHandlePtr));
367         if (!isConnected() && (portHandlePtr != nullptr)) {
368             ALOGD("%s(): MMAP stream DISCONNECTED after starting port %d, will stop it",
369                   __func__, *portHandlePtr);
370             mMmapStream->stop(*portHandlePtr);
371             *portHandlePtr = AUDIO_PORT_HANDLE_NONE;
372             result = AAUDIO_ERROR_DISCONNECTED;
373         }
374         ALOGD("%s(): returning port %d, result %d", __func__,
375               (portHandlePtr == nullptr) ? -1 : *portHandlePtr, result);
376         return result;
377     }
378 }
379 
stopClient(audio_port_handle_t portHandle)380 aaudio_result_t AAudioServiceEndpointMMAP::stopClient(audio_port_handle_t portHandle) {
381     const std::lock_guard<std::mutex> lock(mMmapStreamLock);
382     if (mMmapStream == nullptr) {
383         ALOGE("%s(%d): called after mMmapStream set to NULL", __func__, (int)portHandle);
384         return AAUDIO_ERROR_NULL;
385     } else {
386         aaudio_result_t result = AAudioConvert_androidToAAudioResult(
387                 mMmapStream->stop(portHandle));
388         ALOGD("%s(%d): returning %d", __func__, (int)portHandle, result);
389         return result;
390     }
391 }
392 
standby()393 aaudio_result_t AAudioServiceEndpointMMAP::standby() {
394     const std::lock_guard<std::mutex> lock(mMmapStreamLock);
395     if (mMmapStream == nullptr) {
396         ALOGW("%s(): called after mMmapStream set to NULL", __func__);
397         return AAUDIO_ERROR_NULL;
398     } else {
399         return AAudioConvert_androidToAAudioResult(mMmapStream->standby());
400     }
401 }
402 
exitStandby(AudioEndpointParcelable * parcelable)403 aaudio_result_t AAudioServiceEndpointMMAP::exitStandby(AudioEndpointParcelable* parcelable) {
404     const std::lock_guard<std::mutex> lock(mMmapStreamLock);
405     if (mMmapStream == nullptr) {
406         return AAUDIO_ERROR_NULL;
407     }
408     mAudioDataWrapper->reset();
409     const aaudio_result_t result = createMmapBuffer_l();
410     if (result == AAUDIO_OK) {
411         getDownDataDescription(parcelable);
412     }
413     return result;
414 }
415 
416 // Get free-running DSP or DMA hardware position from the HAL.
getFreeRunningPosition(int64_t * positionFrames,int64_t * timeNanos)417 aaudio_result_t AAudioServiceEndpointMMAP::getFreeRunningPosition(int64_t *positionFrames,
418                                                                 int64_t *timeNanos) {
419     const std::lock_guard<std::mutex> lock(mMmapStreamLock);
420     if (mMmapStream == nullptr) {
421         ALOGW("%s(): called after mMmapStream set to NULL", __func__);
422         return AAUDIO_ERROR_NULL;
423     }
424     struct audio_mmap_position position;
425     status_t status = mMmapStream->getMmapPosition(&position);
426     ALOGV("%s() status= %d, pos = %d, nanos = %lld\n",
427           __func__, status, position.position_frames, (long long) position.time_nanoseconds);
428     if (status == INVALID_OPERATION) {
429         // The HAL can return INVALID_OPERATION when the position is UNKNOWN.
430         // That can cause SHARED MMAP to break. So coerce it to NOT_ENOUGH_DATA.
431         // That will get converted to AAUDIO_ERROR_UNAVAILABLE.
432         ALOGW("%s(): change INVALID_OPERATION to NOT_ENOUGH_DATA", __func__);
433         status = NOT_ENOUGH_DATA; // see b/376467258
434     }
435 
436     const aaudio_result_t result = AAudioConvert_androidToAAudioResult(status);
437     if (result == AAUDIO_ERROR_UNAVAILABLE) {
438         ALOGW("%s(): getMmapPosition() has no position data available", __func__);
439     } else if (result != AAUDIO_OK) {
440         ALOGE("%s(): getMmapPosition() returned status %d", __func__, status);
441     } else {
442         // Convert 32-bit position to 64-bit position.
443         mFramesTransferred.update32(position.position_frames);
444         *positionFrames = mFramesTransferred.get();
445         *timeNanos = position.time_nanoseconds;
446     }
447     return result;
448 }
449 
getTimestamp(int64_t *,int64_t *)450 aaudio_result_t AAudioServiceEndpointMMAP::getTimestamp(int64_t* /*positionFrames*/,
451                                                         int64_t* /*timeNanos*/) {
452     return 0; // TODO
453 }
454 
455 // This is called by onTearDown() in a separate thread to avoid deadlocks.
handleTearDownAsync(audio_port_handle_t portHandle)456 void AAudioServiceEndpointMMAP::handleTearDownAsync(audio_port_handle_t portHandle) {
457     // Are we tearing down the EXCLUSIVE MMAP stream?
458     if (isStreamRegistered(portHandle)) {
459         ALOGD("%s(%d) tearing down this entire MMAP endpoint", __func__, portHandle);
460         disconnectRegisteredStreams();
461     } else {
462         // Must be a SHARED stream?
463         ALOGD("%s(%d) disconnect a specific stream", __func__, portHandle);
464         const aaudio_result_t result = mAAudioService.disconnectStreamByPortHandle(portHandle);
465         ALOGD("%s(%d) disconnectStreamByPortHandle returned %d", __func__, portHandle, result);
466     }
467 };
468 
469 // This is called by AudioFlinger when it wants to destroy a stream.
onTearDown(audio_port_handle_t portHandle)470 void AAudioServiceEndpointMMAP::onTearDown(audio_port_handle_t portHandle) {
471     ALOGD("%s(portHandle = %d) called", __func__, portHandle);
472     const android::sp<AAudioServiceEndpointMMAP> holdEndpoint(this);
473     std::thread asyncTask([holdEndpoint, portHandle]() {
474         holdEndpoint->handleTearDownAsync(portHandle);
475     });
476     asyncTask.detach();
477 }
478 
onVolumeChanged(float volume)479 void AAudioServiceEndpointMMAP::onVolumeChanged(float volume) {
480     ALOGD("%s() volume = %f", __func__, volume);
481     const std::lock_guard<std::mutex> lock(mLockStreams);
482     for(const auto& stream : mRegisteredStreams) {
483         stream->onVolumeChanged(volume);
484     }
485 };
486 
onRoutingChanged(const android::DeviceIdVector & deviceIds)487 void AAudioServiceEndpointMMAP::onRoutingChanged(const android::DeviceIdVector& deviceIds) {
488     ALOGD("%s() called with dev %s, old = %s", __func__, android::toString(deviceIds).c_str(),
489           android::toString(getDeviceIds()).c_str());
490     if (!android::areDeviceIdsEqual(getDeviceIds(), deviceIds)) {
491         if (!getDeviceIds().empty()) {
492             // When there is a routing changed, mmap stream should be disconnected. Set `mConnected`
493             // as false here so that there won't be a new stream connected to this endpoint.
494             mConnected.store(false);
495             const android::sp<AAudioServiceEndpointMMAP> holdEndpoint(this);
496             std::thread asyncTask([holdEndpoint, deviceIds]() {
497                 ALOGD("onRoutingChanged() asyncTask launched");
498                 // When routing changed, the stream is disconnected and cannot be used except for
499                 // closing. In that case, it should be safe to release all registered streams.
500                 // This can help release service side resource in case the client doesn't close
501                 // the stream after receiving disconnect event.
502                 holdEndpoint->releaseRegisteredStreams();
503                 holdEndpoint->setDeviceIds(deviceIds);
504             });
505             asyncTask.detach();
506         } else {
507             setDeviceIds(deviceIds);
508         }
509     }
510 };
511 
512 /**
513  * Get an immutable description of the data queue from the HAL.
514  */
getDownDataDescription(AudioEndpointParcelable * parcelable)515 aaudio_result_t AAudioServiceEndpointMMAP::getDownDataDescription(
516         AudioEndpointParcelable* parcelable)
517 {
518     if (mAudioDataWrapper->setupFifoBuffer(calculateBytesPerFrame(), getBufferCapacity())
519         != AAUDIO_OK) {
520         ALOGE("Failed to setup audio data wrapper, will not be able to "
521               "set data for sound dose computation");
522         // This will not affect the audio processing capability
523     }
524     // Gather information on the data queue based on HAL info.
525     mAudioDataWrapper->fillParcelable(parcelable, parcelable->mDownDataQueueParcelable,
526                                       calculateBytesPerFrame(), mFramesPerBurst,
527                                       getBufferCapacity(),
528                                       getDirection() == AAUDIO_DIRECTION_OUTPUT
529                                               ? SharedMemoryWrapper::WRITE
530                                               : SharedMemoryWrapper::NONE);
531     return AAUDIO_OK;
532 }
533 
getExternalPosition(uint64_t * positionFrames,int64_t * timeNanos)534 aaudio_result_t AAudioServiceEndpointMMAP::getExternalPosition(uint64_t *positionFrames,
535                                                                int64_t *timeNanos)
536 {
537     const std::lock_guard<std::mutex> lock(mMmapStreamLock);
538     if (mHalExternalPositionStatus != AAUDIO_OK) {
539         return mHalExternalPositionStatus;
540     }
541     if (mMmapStream == nullptr) {
542         ALOGW("%s(): called after mMmapStream set to NULL", __func__);
543         return AAUDIO_ERROR_NULL;
544     }
545     uint64_t tempPositionFrames;
546     int64_t tempTimeNanos;
547     const status_t status = mMmapStream->getExternalPosition(&tempPositionFrames, &tempTimeNanos);
548     if (status != OK) {
549         // getExternalPosition reports error. The HAL may not support the API. Cache the result
550         // so that the call will not go to the HAL next time.
551         mHalExternalPositionStatus = AAudioConvert_androidToAAudioResult(status);
552         return mHalExternalPositionStatus;
553     }
554 
555     // If the HAL keeps reporting the same position or timestamp, the HAL may be having some issues
556     // to report correct external position. In that case, we will not trust the values reported from
557     // the HAL. Ideally, we may want to stop querying external position if the HAL cannot report
558     // correct position within a period. But it may not be a good idea to get system time too often.
559     // In that case, a maximum number of frozen external position is defined so that if the
560     // count of the same timestamp or position is reported by the HAL continuously, the values from
561     // the HAL will no longer be trusted.
562     static constexpr int kMaxFrozenCount = 20;
563     // If the HAL version is less than 7.0, the getPresentationPosition is an optional API.
564     // If the HAL version is 7.0 or later, the getPresentationPosition is a mandatory API.
565     // In that case, even the returned status is NO_ERROR, it doesn't indicate the returned
566     // position is a valid one. Do a simple validation, which is checking if the position is
567     // forward within half a second or not, here so that this function can return error if
568     // the validation fails. Note that we don't only apply this validation logic to HAL API
569     // less than 7.0. The reason is that there is a chance the HAL is not reporting the
570     // timestamp and position correctly.
571     if (mLastPositionFrames > tempPositionFrames) {
572         // If the position is going backwards, there must be something wrong with the HAL.
573         // In that case, we do not trust the values reported by the HAL.
574         ALOGW("%s position is going backwards, last position(%jd) current position(%jd)",
575               __func__, mLastPositionFrames, tempPositionFrames);
576         mHalExternalPositionStatus = AAUDIO_ERROR_INTERNAL;
577         return mHalExternalPositionStatus;
578     } else if (mLastPositionFrames == tempPositionFrames) {
579         if (tempTimeNanos - mTimestampNanosForLastPosition >
580                 AAUDIO_NANOS_PER_MILLISECOND * mTimestampGracePeriodMs) {
581             ALOGW("%s, the reported position is not changed within %d msec. "
582                   "Set the external position as not supported", __func__, mTimestampGracePeriodMs);
583             mHalExternalPositionStatus = AAUDIO_ERROR_INTERNAL;
584             return mHalExternalPositionStatus;
585         }
586         mFrozenPositionCount++;
587     } else {
588         mFrozenPositionCount = 0;
589     }
590 
591     if (mTimestampNanosForLastPosition > tempTimeNanos) {
592         // If the timestamp is going backwards, there must be something wrong with the HAL.
593         // In that case, we do not trust the values reported by the HAL.
594         ALOGW("%s timestamp is going backwards, last timestamp(%jd), current timestamp(%jd)",
595               __func__, mTimestampNanosForLastPosition, tempTimeNanos);
596         mHalExternalPositionStatus = AAUDIO_ERROR_INTERNAL;
597         return mHalExternalPositionStatus;
598     } else if (mTimestampNanosForLastPosition == tempTimeNanos) {
599         mFrozenTimestampCount++;
600     } else {
601         mFrozenTimestampCount = 0;
602     }
603 
604     if (mFrozenTimestampCount + mFrozenPositionCount > kMaxFrozenCount) {
605         ALOGW("%s too many frozen external position from HAL.", __func__);
606         mHalExternalPositionStatus = AAUDIO_ERROR_INTERNAL;
607         return mHalExternalPositionStatus;
608     }
609 
610     mLastPositionFrames = tempPositionFrames;
611     mTimestampNanosForLastPosition = tempTimeNanos;
612 
613     // Only update the timestamp and position when they looks valid.
614     *positionFrames = tempPositionFrames;
615     *timeNanos = tempTimeNanos;
616     return mHalExternalPositionStatus;
617 }
618 
619 // mMmapStreamLock should be held when calling this function.
createMmapBuffer_l()620 aaudio_result_t AAudioServiceEndpointMMAP::createMmapBuffer_l()
621 {
622     memset(&mMmapBufferinfo, 0, sizeof(struct audio_mmap_buffer_info));
623     int32_t minSizeFrames = getBufferCapacity();
624     if (minSizeFrames <= 0) { // zero will get rejected
625         minSizeFrames = AAUDIO_BUFFER_CAPACITY_MIN;
626     }
627 
628     if (mMmapStream == nullptr) {
629         ALOGW("%s(): called after mMmapStream set to NULL", __func__);
630         return AAUDIO_ERROR_NULL;
631     }
632 
633     const status_t status = mMmapStream->createMmapBuffer(minSizeFrames, &mMmapBufferinfo);
634     const bool isBufferShareable = mMmapBufferinfo.flags & AUDIO_MMAP_APPLICATION_SHAREABLE;
635     if (status != OK) {
636         ALOGE("%s() - createMmapBuffer() failed with status %d %s",
637               __func__, status, strerror(-status));
638         return AAUDIO_ERROR_UNAVAILABLE;
639     } else {
640         ALOGD("%s() createMmapBuffer() buffer_size = %d fr, burst_size %d fr"
641                       ", Sharable FD: %s",
642               __func__,
643               mMmapBufferinfo.buffer_size_frames,
644               mMmapBufferinfo.burst_size_frames,
645               isBufferShareable ? "Yes" : "No");
646     }
647 
648     setBufferCapacity(mMmapBufferinfo.buffer_size_frames);
649     if (!isBufferShareable) {
650         // Exclusive mode can only be used by the service because the FD cannot be shared.
651         const int32_t audioServiceUid =
652             VALUE_OR_FATAL(legacy2aidl_uid_t_int32_t(getuid()));
653         if ((mMmapClient.attributionSource.uid != audioServiceUid) &&
654             getSharingMode() == AAUDIO_SHARING_MODE_EXCLUSIVE) {
655             ALOGW("%s() - exclusive FD cannot be used by client", __func__);
656             return AAUDIO_ERROR_UNAVAILABLE;
657         }
658     }
659 
660     // AAudio creates a copy of this FD and retains ownership of the copy.
661     // Assume that AudioFlinger will close the original shared_memory_fd.
662 
663     mAudioDataWrapper->getDataFileDescriptor().reset(dup(mMmapBufferinfo.shared_memory_fd));
664     if (mAudioDataWrapper->getDataFileDescriptor().get() == -1) {
665         ALOGE("%s() - could not dup shared_memory_fd", __func__);
666         return AAUDIO_ERROR_INTERNAL;
667     }
668 
669     // Call to HAL to make sure the transport FD was able to be closed by binder.
670     // This is a tricky workaround for a problem in Binder.
671     // TODO:[b/192048842] When that problem is fixed we may be able to remove or change this code.
672     ALOGD("%s() - call getMmapPosition() as a hack to clear FD stuck in Binder", __func__);
673     struct audio_mmap_position position;
674     mMmapStream->getMmapPosition(&position);
675 
676     mFramesPerBurst = mMmapBufferinfo.burst_size_frames;
677 
678     return AAUDIO_OK;
679 }
680 
nextDataReportTime()681 int64_t AAudioServiceEndpointMMAP::nextDataReportTime() {
682     return getDirection() == AAUDIO_DIRECTION_OUTPUT
683             ? AudioClock::getNanoseconds() + mDataReportOffsetNanos
684             : std::numeric_limits<int64_t>::max();
685 }
686 
reportData()687 void AAudioServiceEndpointMMAP::reportData() {
688     const std::lock_guard<std::mutex> lock(mMmapStreamLock);
689 
690     if (mMmapStream == nullptr) {
691         // This must not happen
692         ALOGE("%s() invalid state, mmap stream is not initialized", __func__);
693         return;
694     }
695 
696     auto fifo = mAudioDataWrapper->getFifoBuffer();
697     if (fifo == nullptr) {
698         ALOGE("%s() fifo buffer is not initialized, cannot report data", __func__);
699         return;
700     }
701 
702     WrappingBuffer wrappingBuffer;
703     fifo_frames_t framesAvailable = fifo->getFullDataAvailable(&wrappingBuffer);
704     for (size_t i = 0; i < WrappingBuffer::SIZE; ++i) {
705         if (wrappingBuffer.numFrames[i] > 0) {
706             mMmapStream->reportData(wrappingBuffer.data[i], wrappingBuffer.numFrames[i]);
707         }
708     }
709     fifo->advanceReadIndex(framesAvailable);
710 }
711