| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 1 | /* | 
|  | 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 |  | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 17 | #define LOG_TAG "AAudioServiceStreamBase" | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 18 | //#define LOG_NDEBUG 0 | 
|  | 19 | #include <utils/Log.h> | 
|  | 20 |  | 
| Phil Burk | a5222e2 | 2017-07-28 13:31:14 -0700 | [diff] [blame] | 21 | #include <iomanip> | 
|  | 22 | #include <iostream> | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 23 | #include <mutex> | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 24 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 25 | #include <media/MediaMetricsItem.h> | 
|  | 26 | #include <media/TypeConverter.h> | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 27 | #include <mediautils/SchedulingPolicyService.h> | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 28 |  | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 29 | #include "binding/AAudioServiceMessage.h" | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 30 | #include "core/AudioGlobal.h" | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 31 | #include "utility/AudioClock.h" | 
|  | 32 |  | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 33 | #include "AAudioEndpointManager.h" | 
|  | 34 | #include "AAudioService.h" | 
|  | 35 | #include "AAudioServiceEndpoint.h" | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 36 | #include "AAudioServiceStreamBase.h" | 
|  | 37 | #include "TimestampScheduler.h" | 
|  | 38 |  | 
|  | 39 | using namespace android;  // TODO just import names needed | 
|  | 40 | using namespace aaudio;   // TODO just import names needed | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 41 |  | 
| Philip P. Moltmann | bda4575 | 2020-07-17 16:41:18 -0700 | [diff] [blame] | 42 | using media::permission::Identity; | 
|  | 43 |  | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 44 | /** | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 45 | * Base class for streams in the service. | 
|  | 46 | * @return | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 47 | */ | 
|  | 48 |  | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 49 | AAudioServiceStreamBase::AAudioServiceStreamBase(AAudioService &audioService) | 
| Phil Burk | 8f4fe50 | 2020-07-15 23:54:50 +0000 | [diff] [blame] | 50 | : mTimestampThread("AATime") | 
| Phil Burk | a53ffa6 | 2018-10-10 16:21:37 -0700 | [diff] [blame] | 51 | , mAtomicStreamTimestamp() | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 52 | , mAudioService(audioService) { | 
| Philip P. Moltmann | bda4575 | 2020-07-17 16:41:18 -0700 | [diff] [blame] | 53 | mMmapClient.identity = Identity(); | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 54 | } | 
|  | 55 |  | 
| Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 56 | AAudioServiceStreamBase::~AAudioServiceStreamBase() { | 
| Phil Burk | 8f4fe50 | 2020-07-15 23:54:50 +0000 | [diff] [blame] | 57 | ALOGD("%s() called", __func__); | 
|  | 58 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 59 | // May not be set if open failed. | 
|  | 60 | if (mMetricsId.size() > 0) { | 
|  | 61 | mediametrics::LogItem(mMetricsId) | 
|  | 62 | .set(AMEDIAMETRICS_PROP_EVENT, AMEDIAMETRICS_PROP_EVENT_VALUE_DTOR) | 
|  | 63 | .set(AMEDIAMETRICS_PROP_STATE, AudioGlobal_convertStreamStateToText(getState())) | 
|  | 64 | .record(); | 
|  | 65 | } | 
|  | 66 |  | 
| Phil Burk | 5a26e66 | 2017-07-07 12:44:48 -0700 | [diff] [blame] | 67 | // If the stream is deleted when OPEN or in use then audio resources will leak. | 
|  | 68 | // This would indicate an internal error. So we want to find this ASAP. | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 69 | LOG_ALWAYS_FATAL_IF(!(getState() == AAUDIO_STREAM_STATE_CLOSED | 
| Phil Burk | db46614 | 2021-04-16 16:50:00 +0000 | [diff] [blame^] | 70 | || getState() == AAUDIO_STREAM_STATE_UNINITIALIZED), | 
| Phil Burk | 8b4e05e | 2019-12-17 12:12:09 -0800 | [diff] [blame] | 71 | "service stream %p still open, state = %d", | 
|  | 72 | this, getState()); | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 73 | } | 
|  | 74 |  | 
| Phil Burk | a5222e2 | 2017-07-28 13:31:14 -0700 | [diff] [blame] | 75 | std::string AAudioServiceStreamBase::dumpHeader() { | 
| Phil Burk | bbd5286 | 2018-04-13 11:37:42 -0700 | [diff] [blame] | 76 | return std::string("    T   Handle   UId   Port Run State Format Burst Chan Capacity"); | 
| Phil Burk | a5222e2 | 2017-07-28 13:31:14 -0700 | [diff] [blame] | 77 | } | 
|  | 78 |  | 
| Phil Burk | 4501b35 | 2017-06-29 18:12:36 -0700 | [diff] [blame] | 79 | std::string AAudioServiceStreamBase::dump() const { | 
|  | 80 | std::stringstream result; | 
|  | 81 |  | 
| Phil Burk | a5222e2 | 2017-07-28 13:31:14 -0700 | [diff] [blame] | 82 | result << "    0x" << std::setfill('0') << std::setw(8) << std::hex << mHandle | 
|  | 83 | << std::dec << std::setfill(' ') ; | 
| Philip P. Moltmann | bda4575 | 2020-07-17 16:41:18 -0700 | [diff] [blame] | 84 | result << std::setw(6) << mMmapClient.identity.uid; | 
| Phil Burk | bbd5286 | 2018-04-13 11:37:42 -0700 | [diff] [blame] | 85 | result << std::setw(7) << mClientHandle; | 
| Phil Burk | a5222e2 | 2017-07-28 13:31:14 -0700 | [diff] [blame] | 86 | result << std::setw(4) << (isRunning() ? "yes" : " no"); | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 87 | result << std::setw(6) << getState(); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 88 | result << std::setw(7) << getFormat(); | 
| Phil Burk | a5222e2 | 2017-07-28 13:31:14 -0700 | [diff] [blame] | 89 | result << std::setw(6) << mFramesPerBurst; | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 90 | result << std::setw(5) << getSamplesPerFrame(); | 
|  | 91 | result << std::setw(9) << getBufferCapacity(); | 
| Phil Burk | 4501b35 | 2017-06-29 18:12:36 -0700 | [diff] [blame] | 92 |  | 
|  | 93 | return result.str(); | 
|  | 94 | } | 
|  | 95 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 96 | void AAudioServiceStreamBase::logOpen(aaudio_handle_t streamHandle) { | 
|  | 97 | // This is the first log sent from the AAudio Service for a stream. | 
|  | 98 | mMetricsId = std::string(AMEDIAMETRICS_KEY_PREFIX_AUDIO_STREAM) | 
|  | 99 | + std::to_string(streamHandle); | 
|  | 100 |  | 
|  | 101 | audio_attributes_t attributes = AAudioServiceEndpoint::getAudioAttributesFrom(this); | 
|  | 102 |  | 
|  | 103 | // Once this item is logged by the server, the client with the same PID, UID | 
|  | 104 | // can also log properties. | 
|  | 105 | mediametrics::LogItem(mMetricsId) | 
|  | 106 | .setPid(getOwnerProcessId()) | 
|  | 107 | .setUid(getOwnerUserId()) | 
| Andy Hung | d203eb6 | 2020-04-27 09:12:46 -0700 | [diff] [blame] | 108 | .set(AMEDIAMETRICS_PROP_ALLOWUID, (int32_t)getOwnerUserId()) | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 109 | .set(AMEDIAMETRICS_PROP_EVENT, AMEDIAMETRICS_PROP_EVENT_VALUE_OPEN) | 
|  | 110 | // the following are immutable | 
|  | 111 | .set(AMEDIAMETRICS_PROP_BUFFERCAPACITYFRAMES, (int32_t)getBufferCapacity()) | 
|  | 112 | .set(AMEDIAMETRICS_PROP_BURSTFRAMES, (int32_t)getFramesPerBurst()) | 
|  | 113 | .set(AMEDIAMETRICS_PROP_CHANNELCOUNT, (int32_t)getSamplesPerFrame()) | 
|  | 114 | .set(AMEDIAMETRICS_PROP_CONTENTTYPE, toString(attributes.content_type).c_str()) | 
|  | 115 | .set(AMEDIAMETRICS_PROP_DIRECTION, | 
|  | 116 | AudioGlobal_convertDirectionToText(getDirection())) | 
|  | 117 | .set(AMEDIAMETRICS_PROP_ENCODING, toString(getFormat()).c_str()) | 
|  | 118 | .set(AMEDIAMETRICS_PROP_ROUTEDDEVICEID, (int32_t)getDeviceId()) | 
|  | 119 | .set(AMEDIAMETRICS_PROP_SAMPLERATE, (int32_t)getSampleRate()) | 
|  | 120 | .set(AMEDIAMETRICS_PROP_SESSIONID, (int32_t)getSessionId()) | 
|  | 121 | .set(AMEDIAMETRICS_PROP_SOURCE, toString(attributes.source).c_str()) | 
|  | 122 | .set(AMEDIAMETRICS_PROP_USAGE, toString(attributes.usage).c_str()) | 
|  | 123 | .record(); | 
|  | 124 | } | 
|  | 125 |  | 
| Phil Burk | 15f97c9 | 2018-09-04 14:06:27 -0700 | [diff] [blame] | 126 | aaudio_result_t AAudioServiceStreamBase::open(const aaudio::AAudioStreamRequest &request) { | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 127 | AAudioEndpointManager &mEndpointManager = AAudioEndpointManager::getInstance(); | 
|  | 128 | aaudio_result_t result = AAUDIO_OK; | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 129 |  | 
| Philip P. Moltmann | bda4575 | 2020-07-17 16:41:18 -0700 | [diff] [blame] | 130 | mMmapClient.identity = request.getIdentity(); | 
|  | 131 | // TODO b/182392769: use identity util | 
|  | 132 | mMmapClient.identity.uid = VALUE_OR_FATAL( | 
|  | 133 | legacy2aidl_uid_t_int32_t(IPCThreadState::self()->getCallingUid())); | 
|  | 134 | mMmapClient.identity.pid = VALUE_OR_FATAL( | 
|  | 135 | legacy2aidl_pid_t_int32_t(IPCThreadState::self()->getCallingPid())); | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 136 |  | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 137 | // Limit scope of lock to avoid recursive lock in close(). | 
|  | 138 | { | 
|  | 139 | std::lock_guard<std::mutex> lock(mUpMessageQueueLock); | 
|  | 140 | if (mUpMessageQueue != nullptr) { | 
| Phil Burk | 19e990e | 2018-03-22 13:59:34 -0700 | [diff] [blame] | 141 | ALOGE("%s() called twice", __func__); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 142 | return AAUDIO_ERROR_INVALID_STATE; | 
|  | 143 | } | 
|  | 144 |  | 
| Phil Burk | 8f4fe50 | 2020-07-15 23:54:50 +0000 | [diff] [blame] | 145 | mUpMessageQueue = std::make_shared<SharedRingBuffer>(); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 146 | result = mUpMessageQueue->allocate(sizeof(AAudioServiceMessage), | 
|  | 147 | QUEUE_UP_CAPACITY_COMMANDS); | 
|  | 148 | if (result != AAUDIO_OK) { | 
|  | 149 | goto error; | 
|  | 150 | } | 
|  | 151 |  | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 152 | // This is not protected by a lock because the stream cannot be | 
|  | 153 | // referenced until the service returns a handle to the client. | 
|  | 154 | // So only one thread can open a stream. | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 155 | mServiceEndpoint = mEndpointManager.openEndpoint(mAudioService, | 
| Phil Burk | 15f97c9 | 2018-09-04 14:06:27 -0700 | [diff] [blame] | 156 | request); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 157 | if (mServiceEndpoint == nullptr) { | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 158 | result = AAUDIO_ERROR_UNAVAILABLE; | 
|  | 159 | goto error; | 
|  | 160 | } | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 161 | // Save a weak pointer that we will use to access the endpoint. | 
|  | 162 | mServiceEndpointWeak = mServiceEndpoint; | 
|  | 163 |  | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 164 | mFramesPerBurst = mServiceEndpoint->getFramesPerBurst(); | 
|  | 165 | copyFrom(*mServiceEndpoint); | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 166 | } | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 167 | return result; | 
|  | 168 |  | 
|  | 169 | error: | 
|  | 170 | close(); | 
|  | 171 | return result; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 172 | } | 
| Phil Burk | dec33ab | 2017-01-17 14:48:16 -0800 | [diff] [blame] | 173 |  | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 174 | aaudio_result_t AAudioServiceStreamBase::close() { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 175 | std::lock_guard<std::mutex> lock(mLock); | 
|  | 176 | return close_l(); | 
|  | 177 | } | 
|  | 178 |  | 
|  | 179 | aaudio_result_t AAudioServiceStreamBase::close_l() { | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 180 | if (getState() == AAUDIO_STREAM_STATE_CLOSED) { | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 181 | return AAUDIO_OK; | 
|  | 182 | } | 
|  | 183 |  | 
| Phil Burk | 8f4fe50 | 2020-07-15 23:54:50 +0000 | [diff] [blame] | 184 | // This will call stopTimestampThread() and also stop the stream, | 
|  | 185 | // just in case it was not already stopped. | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 186 | stop_l(); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 187 |  | 
| Phil Burk | 8b4e05e | 2019-12-17 12:12:09 -0800 | [diff] [blame] | 188 | aaudio_result_t result = AAUDIO_OK; | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 189 | sp<AAudioServiceEndpoint> endpoint = mServiceEndpointWeak.promote(); | 
|  | 190 | if (endpoint == nullptr) { | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 191 | result = AAUDIO_ERROR_INVALID_STATE; | 
|  | 192 | } else { | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 193 | endpoint->unregisterStream(this); | 
|  | 194 | AAudioEndpointManager &endpointManager = AAudioEndpointManager::getInstance(); | 
|  | 195 | endpointManager.closeEndpoint(endpoint); | 
|  | 196 |  | 
|  | 197 | // AAudioService::closeStream() prevents two threads from closing at the same time. | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 198 | mServiceEndpoint.clear(); // endpoint will hold the pointer after this method returns. | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 199 | } | 
|  | 200 |  | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 201 | setState(AAUDIO_STREAM_STATE_CLOSED); | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 202 |  | 
|  | 203 | mediametrics::LogItem(mMetricsId) | 
|  | 204 | .set(AMEDIAMETRICS_PROP_EVENT, AMEDIAMETRICS_PROP_EVENT_VALUE_CLOSE) | 
|  | 205 | .record(); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 206 | return result; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 207 | } | 
|  | 208 |  | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 209 | aaudio_result_t AAudioServiceStreamBase::startDevice() { | 
|  | 210 | mClientHandle = AUDIO_PORT_HANDLE_NONE; | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 211 | sp<AAudioServiceEndpoint> endpoint = mServiceEndpointWeak.promote(); | 
|  | 212 | if (endpoint == nullptr) { | 
|  | 213 | ALOGE("%s() has no endpoint", __func__); | 
|  | 214 | return AAUDIO_ERROR_INVALID_STATE; | 
|  | 215 | } | 
|  | 216 | return endpoint->startStream(this, &mClientHandle); | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 217 | } | 
|  | 218 |  | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 219 | /** | 
|  | 220 | * Start the flow of audio data. | 
|  | 221 | * | 
|  | 222 | * An AAUDIO_SERVICE_EVENT_STARTED will be sent to the client when complete. | 
|  | 223 | */ | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 224 | aaudio_result_t AAudioServiceStreamBase::start() { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 225 | std::lock_guard<std::mutex> lock(mLock); | 
|  | 226 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 227 | const int64_t beginNs = AudioClock::getNanoseconds(); | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 228 | aaudio_result_t result = AAUDIO_OK; | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 229 |  | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 230 | if (auto state = getState(); | 
| Phil Burk | db46614 | 2021-04-16 16:50:00 +0000 | [diff] [blame^] | 231 | state == AAUDIO_STREAM_STATE_CLOSED || isDisconnected_l()) { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 232 | ALOGW("%s() already CLOSED, returns INVALID_STATE, handle = %d", | 
|  | 233 | __func__, getHandle()); | 
|  | 234 | return AAUDIO_ERROR_INVALID_STATE; | 
|  | 235 | } | 
|  | 236 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 237 | mediametrics::Defer defer([&] { | 
|  | 238 | mediametrics::LogItem(mMetricsId) | 
|  | 239 | .set(AMEDIAMETRICS_PROP_EVENT, AMEDIAMETRICS_PROP_EVENT_VALUE_START) | 
| Andy Hung | ea84038 | 2020-05-05 21:50:17 -0700 | [diff] [blame] | 240 | .set(AMEDIAMETRICS_PROP_EXECUTIONTIMENS, (int64_t)(AudioClock::getNanoseconds() - beginNs)) | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 241 | .set(AMEDIAMETRICS_PROP_STATE, AudioGlobal_convertStreamStateToText(getState())) | 
|  | 242 | .set(AMEDIAMETRICS_PROP_STATUS, (int32_t)result) | 
|  | 243 | .record(); }); | 
|  | 244 |  | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 245 | if (isRunning()) { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 246 | return result; | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 247 | } | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 248 |  | 
| Phil Burk | 2329638 | 2017-11-20 15:45:11 -0800 | [diff] [blame] | 249 | setFlowing(false); | 
| Phil Burk | 762365c | 2018-12-10 16:02:16 -0800 | [diff] [blame] | 250 | setSuspended(false); | 
| Phil Burk | 2329638 | 2017-11-20 15:45:11 -0800 | [diff] [blame] | 251 |  | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 252 | // Start with fresh presentation timestamps. | 
| Phil Burk | a53ffa6 | 2018-10-10 16:21:37 -0700 | [diff] [blame] | 253 | mAtomicStreamTimestamp.clear(); | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 254 |  | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 255 | mClientHandle = AUDIO_PORT_HANDLE_NONE; | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 256 | result = startDevice(); | 
|  | 257 | if (result != AAUDIO_OK) goto error; | 
|  | 258 |  | 
|  | 259 | // This should happen at the end of the start. | 
|  | 260 | sendServiceEvent(AAUDIO_SERVICE_EVENT_STARTED); | 
|  | 261 | setState(AAUDIO_STREAM_STATE_STARTED); | 
|  | 262 | mThreadEnabled.store(true); | 
|  | 263 | result = mTimestampThread.start(this); | 
|  | 264 | if (result != AAUDIO_OK) goto error; | 
|  | 265 |  | 
|  | 266 | return result; | 
|  | 267 |  | 
|  | 268 | error: | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 269 | disconnect_l(); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 270 | return result; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 271 | } | 
|  | 272 |  | 
|  | 273 | aaudio_result_t AAudioServiceStreamBase::pause() { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 274 | std::lock_guard<std::mutex> lock(mLock); | 
|  | 275 | return pause_l(); | 
|  | 276 | } | 
|  | 277 |  | 
|  | 278 | aaudio_result_t AAudioServiceStreamBase::pause_l() { | 
| Phil Burk | 11e8d33 | 2017-05-24 09:59:02 -0700 | [diff] [blame] | 279 | aaudio_result_t result = AAUDIO_OK; | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 280 | if (!isRunning()) { | 
|  | 281 | return result; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 282 | } | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 283 | const int64_t beginNs = AudioClock::getNanoseconds(); | 
| Phil Burk | 73af62a | 2017-10-26 12:11:47 -0700 | [diff] [blame] | 284 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 285 | mediametrics::Defer defer([&] { | 
|  | 286 | mediametrics::LogItem(mMetricsId) | 
|  | 287 | .set(AMEDIAMETRICS_PROP_EVENT, AMEDIAMETRICS_PROP_EVENT_VALUE_PAUSE) | 
| Andy Hung | ea84038 | 2020-05-05 21:50:17 -0700 | [diff] [blame] | 288 | .set(AMEDIAMETRICS_PROP_EXECUTIONTIMENS, (int64_t)(AudioClock::getNanoseconds() - beginNs)) | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 289 | .set(AMEDIAMETRICS_PROP_STATE, AudioGlobal_convertStreamStateToText(getState())) | 
|  | 290 | .set(AMEDIAMETRICS_PROP_STATUS, (int32_t)result) | 
|  | 291 | .record(); }); | 
|  | 292 |  | 
| Phil Burk | 73af62a | 2017-10-26 12:11:47 -0700 | [diff] [blame] | 293 | result = stopTimestampThread(); | 
|  | 294 | if (result != AAUDIO_OK) { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 295 | disconnect_l(); | 
| Phil Burk | 73af62a | 2017-10-26 12:11:47 -0700 | [diff] [blame] | 296 | return result; | 
|  | 297 | } | 
|  | 298 |  | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 299 | sp<AAudioServiceEndpoint> endpoint = mServiceEndpointWeak.promote(); | 
|  | 300 | if (endpoint == nullptr) { | 
|  | 301 | ALOGE("%s() has no endpoint", __func__); | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 302 | result =  AAUDIO_ERROR_INVALID_STATE; // for MediaMetric tracking | 
|  | 303 | return result; | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 304 | } | 
|  | 305 | result = endpoint->stopStream(this, mClientHandle); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 306 | if (result != AAUDIO_OK) { | 
| Phil Burk | 19e990e | 2018-03-22 13:59:34 -0700 | [diff] [blame] | 307 | ALOGE("%s() mServiceEndpoint returned %d, %s", __func__, result, getTypeText()); | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 308 | disconnect_l(); // TODO should we return or pause Base first? | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 309 | } | 
|  | 310 |  | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 311 | sendServiceEvent(AAUDIO_SERVICE_EVENT_PAUSED); | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 312 | setState(AAUDIO_STREAM_STATE_PAUSED); | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 313 | return result; | 
|  | 314 | } | 
|  | 315 |  | 
| Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 316 | aaudio_result_t AAudioServiceStreamBase::stop() { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 317 | std::lock_guard<std::mutex> lock(mLock); | 
|  | 318 | return stop_l(); | 
|  | 319 | } | 
|  | 320 |  | 
|  | 321 | aaudio_result_t AAudioServiceStreamBase::stop_l() { | 
| Phil Burk | 11e8d33 | 2017-05-24 09:59:02 -0700 | [diff] [blame] | 322 | aaudio_result_t result = AAUDIO_OK; | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 323 | if (!isRunning()) { | 
|  | 324 | return result; | 
| Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 325 | } | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 326 | const int64_t beginNs = AudioClock::getNanoseconds(); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 327 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 328 | mediametrics::Defer defer([&] { | 
|  | 329 | mediametrics::LogItem(mMetricsId) | 
|  | 330 | .set(AMEDIAMETRICS_PROP_EVENT, AMEDIAMETRICS_PROP_EVENT_VALUE_STOP) | 
| Andy Hung | ea84038 | 2020-05-05 21:50:17 -0700 | [diff] [blame] | 331 | .set(AMEDIAMETRICS_PROP_EXECUTIONTIMENS, (int64_t)(AudioClock::getNanoseconds() - beginNs)) | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 332 | .set(AMEDIAMETRICS_PROP_STATE, AudioGlobal_convertStreamStateToText(getState())) | 
|  | 333 | .set(AMEDIAMETRICS_PROP_STATUS, (int32_t)result) | 
|  | 334 | .record(); }); | 
|  | 335 |  | 
| Phil Burk | 83fb844 | 2017-10-05 16:55:17 -0700 | [diff] [blame] | 336 | setState(AAUDIO_STREAM_STATE_STOPPING); | 
|  | 337 |  | 
| Phil Burk | d50ae62 | 2021-04-15 23:40:25 +0000 | [diff] [blame] | 338 | // Temporarily unlock because we are joining the timestamp thread and it may try | 
|  | 339 | // to acquire mLock. | 
|  | 340 | mLock.unlock(); | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 341 | result = stopTimestampThread(); | 
| Phil Burk | d50ae62 | 2021-04-15 23:40:25 +0000 | [diff] [blame] | 342 | mLock.lock(); | 
|  | 343 |  | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 344 | if (result != AAUDIO_OK) { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 345 | disconnect_l(); | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 346 | return result; | 
|  | 347 | } | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 348 |  | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 349 | sp<AAudioServiceEndpoint> endpoint = mServiceEndpointWeak.promote(); | 
|  | 350 | if (endpoint == nullptr) { | 
|  | 351 | ALOGE("%s() has no endpoint", __func__); | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 352 | result =  AAUDIO_ERROR_INVALID_STATE; // for MediaMetric tracking | 
|  | 353 | return result; | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 354 | } | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 355 | // TODO wait for data to be played out | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 356 | result = endpoint->stopStream(this, mClientHandle); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 357 | if (result != AAUDIO_OK) { | 
| Phil Burk | 6e2770e | 2018-05-01 13:03:52 -0700 | [diff] [blame] | 358 | ALOGE("%s() stopStream returned %d, %s", __func__, result, getTypeText()); | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 359 | disconnect_l(); | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 360 | // TODO what to do with result here? | 
|  | 361 | } | 
|  | 362 |  | 
| Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 363 | sendServiceEvent(AAUDIO_SERVICE_EVENT_STOPPED); | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 364 | setState(AAUDIO_STREAM_STATE_STOPPED); | 
| Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 365 | return result; | 
|  | 366 | } | 
|  | 367 |  | 
| Phil Burk | 98d6d92 | 2017-07-06 11:52:45 -0700 | [diff] [blame] | 368 | aaudio_result_t AAudioServiceStreamBase::stopTimestampThread() { | 
|  | 369 | aaudio_result_t result = AAUDIO_OK; | 
|  | 370 | // clear flag that tells thread to loop | 
|  | 371 | if (mThreadEnabled.exchange(false)) { | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 372 | result = mTimestampThread.stop(); | 
| Phil Burk | 98d6d92 | 2017-07-06 11:52:45 -0700 | [diff] [blame] | 373 | } | 
|  | 374 | return result; | 
|  | 375 | } | 
|  | 376 |  | 
| Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 377 | aaudio_result_t AAudioServiceStreamBase::flush() { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 378 | std::lock_guard<std::mutex> lock(mLock); | 
| Phil Burk | 5cc83c3 | 2017-11-28 15:43:18 -0800 | [diff] [blame] | 379 | aaudio_result_t result = AAudio_isFlushAllowed(getState()); | 
|  | 380 | if (result != AAUDIO_OK) { | 
|  | 381 | return result; | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 382 | } | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 383 | const int64_t beginNs = AudioClock::getNanoseconds(); | 
| Phil Burk | 5cc83c3 | 2017-11-28 15:43:18 -0800 | [diff] [blame] | 384 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 385 | mediametrics::Defer defer([&] { | 
|  | 386 | mediametrics::LogItem(mMetricsId) | 
|  | 387 | .set(AMEDIAMETRICS_PROP_EVENT, AMEDIAMETRICS_PROP_EVENT_VALUE_FLUSH) | 
| Andy Hung | ea84038 | 2020-05-05 21:50:17 -0700 | [diff] [blame] | 388 | .set(AMEDIAMETRICS_PROP_EXECUTIONTIMENS, (int64_t)(AudioClock::getNanoseconds() - beginNs)) | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 389 | .set(AMEDIAMETRICS_PROP_STATE, AudioGlobal_convertStreamStateToText(getState())) | 
|  | 390 | .set(AMEDIAMETRICS_PROP_STATUS, (int32_t)result) | 
|  | 391 | .record(); }); | 
|  | 392 |  | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 393 | // Data will get flushed when the client receives the FLUSHED event. | 
| Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 394 | sendServiceEvent(AAUDIO_SERVICE_EVENT_FLUSHED); | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 395 | setState(AAUDIO_STREAM_STATE_FLUSHED); | 
| Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 396 | return AAUDIO_OK; | 
|  | 397 | } | 
|  | 398 |  | 
| Phil Burk | cf5f6d2 | 2017-05-26 12:35:07 -0700 | [diff] [blame] | 399 | // implement Runnable, periodically send timestamps to client | 
| Phil Burk | a53ffa6 | 2018-10-10 16:21:37 -0700 | [diff] [blame] | 400 | __attribute__((no_sanitize("integer"))) | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 401 | void AAudioServiceStreamBase::run() { | 
| Phil Burk | 19e990e | 2018-03-22 13:59:34 -0700 | [diff] [blame] | 402 | ALOGD("%s() %s entering >>>>>>>>>>>>>> TIMESTAMPS", __func__, getTypeText()); | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 403 | TimestampScheduler timestampScheduler; | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 404 | timestampScheduler.setBurstPeriod(mFramesPerBurst, getSampleRate()); | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 405 | timestampScheduler.start(AudioClock::getNanoseconds()); | 
|  | 406 | int64_t nextTime = timestampScheduler.nextAbsoluteTime(); | 
| Phil Burk | a53ffa6 | 2018-10-10 16:21:37 -0700 | [diff] [blame] | 407 | int32_t loopCount = 0; | 
| Phil Burk | 45da1b7 | 2021-04-07 06:30:50 +0000 | [diff] [blame] | 408 | aaudio_result_t result = AAUDIO_OK; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 409 | while(mThreadEnabled.load()) { | 
| Phil Burk | a53ffa6 | 2018-10-10 16:21:37 -0700 | [diff] [blame] | 410 | loopCount++; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 411 | if (AudioClock::getNanoseconds() >= nextTime) { | 
| Phil Burk | 45da1b7 | 2021-04-07 06:30:50 +0000 | [diff] [blame] | 412 | result = sendCurrentTimestamp(); | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 413 | if (result != AAUDIO_OK) { | 
| Phil Burk | a53ffa6 | 2018-10-10 16:21:37 -0700 | [diff] [blame] | 414 | ALOGE("%s() timestamp thread got result = %d", __func__, result); | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 415 | break; | 
|  | 416 | } | 
|  | 417 | nextTime = timestampScheduler.nextAbsoluteTime(); | 
|  | 418 | } else  { | 
|  | 419 | // Sleep until it is time to send the next timestamp. | 
| Phil Burk | 98d6d92 | 2017-07-06 11:52:45 -0700 | [diff] [blame] | 420 | // TODO Wait for a signal with a timeout so that we can stop more quickly. | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 421 | AudioClock::sleepUntilNanoTime(nextTime); | 
|  | 422 | } | 
|  | 423 | } | 
| Phil Burk | 45da1b7 | 2021-04-07 06:30:50 +0000 | [diff] [blame] | 424 | // This was moved from the calls in stop_l() and pause_l(), which could cause a deadlock | 
|  | 425 | // if it resulted in a call to disconnect. | 
|  | 426 | if (result == AAUDIO_OK) { | 
|  | 427 | (void) sendCurrentTimestamp(); | 
|  | 428 | } | 
| Phil Burk | a53ffa6 | 2018-10-10 16:21:37 -0700 | [diff] [blame] | 429 | ALOGD("%s() %s exiting after %d loops <<<<<<<<<<<<<< TIMESTAMPS", | 
|  | 430 | __func__, getTypeText(), loopCount); | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 431 | } | 
|  | 432 |  | 
| Phil Burk | 5ef003b | 2017-06-30 11:43:37 -0700 | [diff] [blame] | 433 | void AAudioServiceStreamBase::disconnect() { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 434 | std::lock_guard<std::mutex> lock(mLock); | 
|  | 435 | disconnect_l(); | 
|  | 436 | } | 
|  | 437 |  | 
|  | 438 | void AAudioServiceStreamBase::disconnect_l() { | 
| Phil Burk | db46614 | 2021-04-16 16:50:00 +0000 | [diff] [blame^] | 439 | if (!isDisconnected_l() && getState() != AAUDIO_STREAM_STATE_CLOSED) { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 440 |  | 
| Phil Burk | a987670 | 2020-04-20 18:16:15 -0700 | [diff] [blame] | 441 | mediametrics::LogItem(mMetricsId) | 
|  | 442 | .set(AMEDIAMETRICS_PROP_EVENT, AMEDIAMETRICS_PROP_EVENT_VALUE_DISCONNECT) | 
|  | 443 | .set(AMEDIAMETRICS_PROP_STATE, AudioGlobal_convertStreamStateToText(getState())) | 
|  | 444 | .record(); | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 445 |  | 
| Phil Burk | 5ef003b | 2017-06-30 11:43:37 -0700 | [diff] [blame] | 446 | sendServiceEvent(AAUDIO_SERVICE_EVENT_DISCONNECTED); | 
| Phil Burk | db46614 | 2021-04-16 16:50:00 +0000 | [diff] [blame^] | 447 | setDisconnected_l(true); | 
| Phil Burk | 5ef003b | 2017-06-30 11:43:37 -0700 | [diff] [blame] | 448 | } | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 449 | } | 
|  | 450 |  | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 451 | aaudio_result_t AAudioServiceStreamBase::registerAudioThread(pid_t clientThreadId, | 
|  | 452 | int priority) { | 
|  | 453 | std::lock_guard<std::mutex> lock(mLock); | 
|  | 454 | aaudio_result_t result = AAUDIO_OK; | 
|  | 455 | if (getRegisteredThread() != AAudioServiceStreamBase::ILLEGAL_THREAD_ID) { | 
|  | 456 | ALOGE("AAudioService::registerAudioThread(), thread already registered"); | 
|  | 457 | result = AAUDIO_ERROR_INVALID_STATE; | 
|  | 458 | } else { | 
|  | 459 | const pid_t ownerPid = IPCThreadState::self()->getCallingPid(); // TODO review | 
|  | 460 | setRegisteredThread(clientThreadId); | 
|  | 461 | int err = android::requestPriority(ownerPid, clientThreadId, | 
|  | 462 | priority, true /* isForApp */); | 
|  | 463 | if (err != 0) { | 
|  | 464 | ALOGE("AAudioService::registerAudioThread(%d) failed, errno = %d, priority = %d", | 
|  | 465 | clientThreadId, errno, priority); | 
|  | 466 | result = AAUDIO_ERROR_INTERNAL; | 
|  | 467 | } | 
|  | 468 | } | 
|  | 469 | return result; | 
|  | 470 | } | 
|  | 471 |  | 
|  | 472 | aaudio_result_t AAudioServiceStreamBase::unregisterAudioThread(pid_t clientThreadId) { | 
|  | 473 | std::lock_guard<std::mutex> lock(mLock); | 
|  | 474 | aaudio_result_t result = AAUDIO_OK; | 
|  | 475 | if (getRegisteredThread() != clientThreadId) { | 
|  | 476 | ALOGE("%s(), wrong thread", __func__); | 
|  | 477 | result = AAUDIO_ERROR_ILLEGAL_ARGUMENT; | 
|  | 478 | } else { | 
|  | 479 | setRegisteredThread(0); | 
|  | 480 | } | 
|  | 481 | return result; | 
|  | 482 | } | 
|  | 483 |  | 
|  | 484 | void AAudioServiceStreamBase::setState(aaudio_stream_state_t state) { | 
|  | 485 | // CLOSED is a final state. | 
|  | 486 | if (mState != AAUDIO_STREAM_STATE_CLOSED) { | 
|  | 487 | mState = state; | 
|  | 488 | } else { | 
|  | 489 | ALOGW_IF(mState != state, "%s(%d) when already CLOSED", __func__, state); | 
|  | 490 | } | 
|  | 491 | } | 
|  | 492 |  | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 493 | aaudio_result_t AAudioServiceStreamBase::sendServiceEvent(aaudio_service_event_t event, | 
| Phil Burk | 2329638 | 2017-11-20 15:45:11 -0800 | [diff] [blame] | 494 | double  dataDouble) { | 
| Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 495 | AAudioServiceMessage command; | 
|  | 496 | command.what = AAudioServiceMessage::code::EVENT; | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 497 | command.event.event = event; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 498 | command.event.dataDouble = dataDouble; | 
| Phil Burk | 2329638 | 2017-11-20 15:45:11 -0800 | [diff] [blame] | 499 | return writeUpMessageQueue(&command); | 
|  | 500 | } | 
|  | 501 |  | 
|  | 502 | aaudio_result_t AAudioServiceStreamBase::sendServiceEvent(aaudio_service_event_t event, | 
|  | 503 | int64_t dataLong) { | 
|  | 504 | AAudioServiceMessage command; | 
|  | 505 | command.what = AAudioServiceMessage::code::EVENT; | 
|  | 506 | command.event.event = event; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 507 | command.event.dataLong = dataLong; | 
|  | 508 | return writeUpMessageQueue(&command); | 
|  | 509 | } | 
|  | 510 |  | 
| Phil Burk | f878a8d | 2019-03-29 17:23:00 -0700 | [diff] [blame] | 511 | bool AAudioServiceStreamBase::isUpMessageQueueBusy() { | 
|  | 512 | std::lock_guard<std::mutex> lock(mUpMessageQueueLock); | 
|  | 513 | if (mUpMessageQueue == nullptr) { | 
|  | 514 | ALOGE("%s(): mUpMessageQueue null! - stream not open", __func__); | 
|  | 515 | return true; | 
|  | 516 | } | 
| Phil Burk | f878a8d | 2019-03-29 17:23:00 -0700 | [diff] [blame] | 517 | // Is it half full or more | 
| Phil Burk | 8f4fe50 | 2020-07-15 23:54:50 +0000 | [diff] [blame] | 518 | return mUpMessageQueue->getFractionalFullness() >= 0.5; | 
| Phil Burk | f878a8d | 2019-03-29 17:23:00 -0700 | [diff] [blame] | 519 | } | 
|  | 520 |  | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 521 | aaudio_result_t AAudioServiceStreamBase::writeUpMessageQueue(AAudioServiceMessage *command) { | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 522 | std::lock_guard<std::mutex> lock(mUpMessageQueueLock); | 
| Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 523 | if (mUpMessageQueue == nullptr) { | 
| Phil Burk | 19e990e | 2018-03-22 13:59:34 -0700 | [diff] [blame] | 524 | ALOGE("%s(): mUpMessageQueue null! - stream not open", __func__); | 
| Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 525 | return AAUDIO_ERROR_NULL; | 
|  | 526 | } | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 527 | int32_t count = mUpMessageQueue->getFifoBuffer()->write(command, 1); | 
|  | 528 | if (count != 1) { | 
| Phil Burk | 762365c | 2018-12-10 16:02:16 -0800 | [diff] [blame] | 529 | ALOGW("%s(): Queue full. Did client stop? Suspending stream. what = %u, %s", | 
|  | 530 | __func__, command->what, getTypeText()); | 
|  | 531 | setSuspended(true); | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 532 | return AAUDIO_ERROR_WOULD_BLOCK; | 
|  | 533 | } else { | 
|  | 534 | return AAUDIO_OK; | 
|  | 535 | } | 
|  | 536 | } | 
|  | 537 |  | 
| Phil Burk | 2329638 | 2017-11-20 15:45:11 -0800 | [diff] [blame] | 538 | aaudio_result_t AAudioServiceStreamBase::sendXRunCount(int32_t xRunCount) { | 
|  | 539 | return sendServiceEvent(AAUDIO_SERVICE_EVENT_XRUN, (int64_t) xRunCount); | 
|  | 540 | } | 
|  | 541 |  | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 542 | aaudio_result_t AAudioServiceStreamBase::sendCurrentTimestamp() { | 
|  | 543 | AAudioServiceMessage command; | 
| Phil Burk | f878a8d | 2019-03-29 17:23:00 -0700 | [diff] [blame] | 544 | // It is not worth filling up the queue with timestamps. | 
|  | 545 | // That can cause the stream to get suspended. | 
|  | 546 | // So just drop the timestamp if the queue is getting full. | 
|  | 547 | if (isUpMessageQueueBusy()) { | 
|  | 548 | return AAUDIO_OK; | 
|  | 549 | } | 
|  | 550 |  | 
| Phil Burk | 97350f9 | 2017-07-21 15:59:44 -0700 | [diff] [blame] | 551 | // Send a timestamp for the clock model. | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 552 | aaudio_result_t result = getFreeRunningPosition(&command.timestamp.position, | 
|  | 553 | &command.timestamp.timestamp); | 
|  | 554 | if (result == AAUDIO_OK) { | 
| Phil Burk | 19e990e | 2018-03-22 13:59:34 -0700 | [diff] [blame] | 555 | ALOGV("%s() SERVICE  %8lld at %lld", __func__, | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 556 | (long long) command.timestamp.position, | 
|  | 557 | (long long) command.timestamp.timestamp); | 
| Phil Burk | 97350f9 | 2017-07-21 15:59:44 -0700 | [diff] [blame] | 558 | command.what = AAudioServiceMessage::code::TIMESTAMP_SERVICE; | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 559 | result = writeUpMessageQueue(&command); | 
| Phil Burk | 97350f9 | 2017-07-21 15:59:44 -0700 | [diff] [blame] | 560 |  | 
|  | 561 | if (result == AAUDIO_OK) { | 
|  | 562 | // Send a hardware timestamp for presentation time. | 
|  | 563 | result = getHardwareTimestamp(&command.timestamp.position, | 
|  | 564 | &command.timestamp.timestamp); | 
|  | 565 | if (result == AAUDIO_OK) { | 
| Phil Burk | 19e990e | 2018-03-22 13:59:34 -0700 | [diff] [blame] | 566 | ALOGV("%s() HARDWARE %8lld at %lld", __func__, | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 567 | (long long) command.timestamp.position, | 
|  | 568 | (long long) command.timestamp.timestamp); | 
| Phil Burk | 97350f9 | 2017-07-21 15:59:44 -0700 | [diff] [blame] | 569 | command.what = AAudioServiceMessage::code::TIMESTAMP_HARDWARE; | 
|  | 570 | result = writeUpMessageQueue(&command); | 
|  | 571 | } | 
|  | 572 | } | 
|  | 573 | } | 
|  | 574 |  | 
| Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 575 | if (result == AAUDIO_ERROR_UNAVAILABLE) { // TODO review best error code | 
| Phil Burk | 940083c | 2017-07-17 17:00:02 -0700 | [diff] [blame] | 576 | result = AAUDIO_OK; // just not available yet, try again later | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 577 | } | 
|  | 578 | return result; | 
| Phil Burk | 2355edb | 2016-12-26 13:54:02 -0800 | [diff] [blame] | 579 | } | 
|  | 580 |  | 
| Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 581 | /** | 
|  | 582 | * Get an immutable description of the in-memory queues | 
|  | 583 | * used to communicate with the underlying HAL or Service. | 
|  | 584 | */ | 
|  | 585 | aaudio_result_t AAudioServiceStreamBase::getDescription(AudioEndpointParcelable &parcelable) { | 
| Phil Burk | 7ebbc11 | 2020-05-13 15:55:17 -0700 | [diff] [blame] | 586 | std::lock_guard<std::mutex> lock(mLock); | 
| Phil Burk | 523b304 | 2017-09-13 13:03:08 -0700 | [diff] [blame] | 587 | { | 
|  | 588 | std::lock_guard<std::mutex> lock(mUpMessageQueueLock); | 
|  | 589 | if (mUpMessageQueue == nullptr) { | 
| Phil Burk | 19e990e | 2018-03-22 13:59:34 -0700 | [diff] [blame] | 590 | ALOGE("%s(): mUpMessageQueue null! - stream not open", __func__); | 
| Phil Burk | 523b304 | 2017-09-13 13:03:08 -0700 | [diff] [blame] | 591 | return AAUDIO_ERROR_NULL; | 
|  | 592 | } | 
|  | 593 | // Gather information on the message queue. | 
|  | 594 | mUpMessageQueue->fillParcelable(parcelable, | 
|  | 595 | parcelable.mUpMessageQueueParcelable); | 
|  | 596 | } | 
|  | 597 | return getAudioDataDescription(parcelable); | 
| Phil Burk | 11e8d33 | 2017-05-24 09:59:02 -0700 | [diff] [blame] | 598 | } | 
| Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 599 |  | 
|  | 600 | void AAudioServiceStreamBase::onVolumeChanged(float volume) { | 
|  | 601 | sendServiceEvent(AAUDIO_SERVICE_EVENT_VOLUME, volume); | 
|  | 602 | } |