blob: aaab567f31ba979bfb9250524aa2affdab2e3362 [file] [log] [blame]
Phil Burk2355edb2016-12-26 13:54:02 -08001/*
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
Phil Burk5ed503c2017-02-01 09:38:15 -080017#ifndef AAUDIO_AAUDIO_SERVICE_STREAM_BASE_H
18#define AAUDIO_AAUDIO_SERVICE_STREAM_BASE_H
Phil Burk2355edb2016-12-26 13:54:02 -080019
Phil Burk71f35bb2017-04-13 16:05:07 -070020#include <assert.h>
Phil Burkc0c70e32017-02-09 13:18:38 -080021#include <mutex>
Phil Burkdec33ab2017-01-17 14:48:16 -080022
Phil Burk97350f92017-07-21 15:59:44 -070023#include <media/AudioClient.h>
Phil Burk11e8d332017-05-24 09:59:02 -070024#include <utils/RefBase.h>
25
Phil Burk2355edb2016-12-26 13:54:02 -080026#include "fifo/FifoBuffer.h"
Phil Burkc0c70e32017-02-09 13:18:38 -080027#include "binding/IAAudioService.h"
28#include "binding/AudioEndpointParcelable.h"
29#include "binding/AAudioServiceMessage.h"
30#include "utility/AAudioUtilities.h"
Phil Burk97350f92017-07-21 15:59:44 -070031#include "utility/AudioClock.h"
Phil Burkc0c70e32017-02-09 13:18:38 -080032
Phil Burk2355edb2016-12-26 13:54:02 -080033#include "SharedRingBuffer.h"
Phil Burk5ed503c2017-02-01 09:38:15 -080034#include "AAudioThread.h"
Phil Burk523b3042017-09-13 13:03:08 -070035
36namespace android {
37 class AAudioService;
38}
Phil Burk2355edb2016-12-26 13:54:02 -080039
Phil Burk5ed503c2017-02-01 09:38:15 -080040namespace aaudio {
Phil Burk2355edb2016-12-26 13:54:02 -080041
Phil Burk39f02dd2017-08-04 09:13:31 -070042class AAudioServiceEndpoint;
43
Phil Burk2355edb2016-12-26 13:54:02 -080044// We expect the queue to only have a few commands.
45// This should be way more than we need.
46#define QUEUE_UP_CAPACITY_COMMANDS (128)
47
Phil Burkc0c70e32017-02-09 13:18:38 -080048/**
Phil Burk39f02dd2017-08-04 09:13:31 -070049 * Each instance of AAudioServiceStreamBase corresponds to a client stream.
50 * It uses a subclass of AAudioServiceEndpoint to communicate with the underlying device or port.
Phil Burkc0c70e32017-02-09 13:18:38 -080051 */
52class AAudioServiceStreamBase
Phil Burk11e8d332017-05-24 09:59:02 -070053 : public virtual android::RefBase
Phil Burk39f02dd2017-08-04 09:13:31 -070054 , public AAudioStreamParameters
Phil Burk11e8d332017-05-24 09:59:02 -070055 , public Runnable {
Phil Burk2355edb2016-12-26 13:54:02 -080056
57public:
Phil Burk19e990e2018-03-22 13:59:34 -070058 explicit AAudioServiceStreamBase(android::AAudioService &aAudioService);
Phil Burk39f02dd2017-08-04 09:13:31 -070059
Phil Burk5ed503c2017-02-01 09:38:15 -080060 virtual ~AAudioServiceStreamBase();
Phil Burk2355edb2016-12-26 13:54:02 -080061
62 enum {
63 ILLEGAL_THREAD_ID = 0
64 };
65
Phil Burka5222e22017-07-28 13:31:14 -070066 static std::string dumpHeader();
67
68 // does not include EOL
69 virtual std::string dump() const;
Phil Burk4501b352017-06-29 18:12:36 -070070
Phil Burkc0c70e32017-02-09 13:18:38 -080071 // -------------------------------------------------------------------
Phil Burk2355edb2016-12-26 13:54:02 -080072 /**
73 * Open the device.
74 */
Phil Burk39f02dd2017-08-04 09:13:31 -070075 virtual aaudio_result_t open(const aaudio::AAudioStreamRequest &request) = 0;
Phil Burkc0c70e32017-02-09 13:18:38 -080076
77 virtual aaudio_result_t close();
Phil Burk2355edb2016-12-26 13:54:02 -080078
79 /**
Phil Burk39f02dd2017-08-04 09:13:31 -070080 * Start the flow of audio data.
81 *
82 * This is not guaranteed to be synchronous but it currently is.
83 * An AAUDIO_SERVICE_EVENT_STARTED will be sent to the client when complete.
Phil Burk2355edb2016-12-26 13:54:02 -080084 */
Phil Burk71f35bb2017-04-13 16:05:07 -070085 virtual aaudio_result_t start();
Phil Burk2355edb2016-12-26 13:54:02 -080086
87 /**
Phil Burk39f02dd2017-08-04 09:13:31 -070088 * Stop the flow of data so that start() can resume without loss of data.
89 *
90 * This is not guaranteed to be synchronous but it currently is.
91 * An AAUDIO_SERVICE_EVENT_PAUSED will be sent to the client when complete.
92 */
Phil Burk71f35bb2017-04-13 16:05:07 -070093 virtual aaudio_result_t pause();
94
95 /**
Phil Burk39f02dd2017-08-04 09:13:31 -070096 * Stop the flow of data after the currently queued data has finished playing.
97 *
98 * This is not guaranteed to be synchronous but it currently is.
99 * An AAUDIO_SERVICE_EVENT_STOPPED will be sent to the client when complete.
100 *
Phil Burk71f35bb2017-04-13 16:05:07 -0700101 */
102 virtual aaudio_result_t stop();
Phil Burk2355edb2016-12-26 13:54:02 -0800103
Phil Burk98d6d922017-07-06 11:52:45 -0700104 aaudio_result_t stopTimestampThread();
105
Phil Burk2355edb2016-12-26 13:54:02 -0800106 /**
Phil Burk39f02dd2017-08-04 09:13:31 -0700107 * Discard any data held by the underlying HAL or Service.
108 *
109 * An AAUDIO_SERVICE_EVENT_FLUSHED will be sent to the client when complete.
Phil Burk2355edb2016-12-26 13:54:02 -0800110 */
Phil Burk71f35bb2017-04-13 16:05:07 -0700111 virtual aaudio_result_t flush();
Phil Burk2355edb2016-12-26 13:54:02 -0800112
Phil Burk39f02dd2017-08-04 09:13:31 -0700113
jiabind1f1cb62020-03-24 11:57:57 -0700114 virtual aaudio_result_t startClient(const android::AudioClient& client,
115 const audio_attributes_t *attr __unused,
Eric Laurentcb4dae22017-07-01 19:39:32 -0700116 audio_port_handle_t *clientHandle __unused) {
Phil Burk39f02dd2017-08-04 09:13:31 -0700117 ALOGD("AAudioServiceStreamBase::startClient(%p, ...) AAUDIO_ERROR_UNAVAILABLE", &client);
Eric Laurentcb4dae22017-07-01 19:39:32 -0700118 return AAUDIO_ERROR_UNAVAILABLE;
119 }
120
121 virtual aaudio_result_t stopClient(audio_port_handle_t clientHandle __unused) {
Phil Burk39f02dd2017-08-04 09:13:31 -0700122 ALOGD("AAudioServiceStreamBase::stopClient(%d) AAUDIO_ERROR_UNAVAILABLE", clientHandle);
Eric Laurentcb4dae22017-07-01 19:39:32 -0700123 return AAUDIO_ERROR_UNAVAILABLE;
124 }
125
Phil Burk11e8d332017-05-24 09:59:02 -0700126 bool isRunning() const {
127 return mState == AAUDIO_STREAM_STATE_STARTED;
128 }
Eric Laurentcb4dae22017-07-01 19:39:32 -0700129
Phil Burkc0c70e32017-02-09 13:18:38 -0800130 // -------------------------------------------------------------------
Phil Burk2355edb2016-12-26 13:54:02 -0800131
Phil Burkc0c70e32017-02-09 13:18:38 -0800132 /**
Phil Burk23296382017-11-20 15:45:11 -0800133 * Send a message to the client with an int64_t data value.
Phil Burkc0c70e32017-02-09 13:18:38 -0800134 */
135 aaudio_result_t sendServiceEvent(aaudio_service_event_t event,
Phil Burkc0c70e32017-02-09 13:18:38 -0800136 int64_t dataLong = 0);
Phil Burk23296382017-11-20 15:45:11 -0800137 /**
138 * Send a message to the client with an double data value.
139 */
140 aaudio_result_t sendServiceEvent(aaudio_service_event_t event,
141 double dataDouble);
Phil Burkdec33ab2017-01-17 14:48:16 -0800142
Phil Burkc0c70e32017-02-09 13:18:38 -0800143 /**
144 * Fill in a parcelable description of stream.
145 */
146 aaudio_result_t getDescription(AudioEndpointParcelable &parcelable);
Phil Burk2355edb2016-12-26 13:54:02 -0800147
Phil Burk2355edb2016-12-26 13:54:02 -0800148
Phil Burkc0c70e32017-02-09 13:18:38 -0800149 void setRegisteredThread(pid_t pid) {
Phil Burk2355edb2016-12-26 13:54:02 -0800150 mRegisteredClientThread = pid;
151 }
Phil Burkdec33ab2017-01-17 14:48:16 -0800152
Phil Burkc0c70e32017-02-09 13:18:38 -0800153 pid_t getRegisteredThread() const {
Phil Burk2355edb2016-12-26 13:54:02 -0800154 return mRegisteredClientThread;
155 }
156
Phil Burkc0c70e32017-02-09 13:18:38 -0800157 int32_t getFramesPerBurst() const {
158 return mFramesPerBurst;
159 }
160
Phil Burkc0c70e32017-02-09 13:18:38 -0800161 void run() override; // to implement Runnable
162
Phil Burk5ef003b2017-06-30 11:43:37 -0700163 void disconnect();
Phil Burkc0c70e32017-02-09 13:18:38 -0800164
Phil Burk39f02dd2017-08-04 09:13:31 -0700165 const android::AudioClient &getAudioClient() {
166 return mMmapClient;
167 }
168
Phil Burk2ac035f2017-06-23 14:51:14 -0700169 uid_t getOwnerUserId() const {
Eric Laurentcb4dae22017-07-01 19:39:32 -0700170 return mMmapClient.clientUid;
Phil Burk2ac035f2017-06-23 14:51:14 -0700171 }
172
Phil Burkb63320a2017-06-30 10:28:20 -0700173 pid_t getOwnerProcessId() const {
Eric Laurentcb4dae22017-07-01 19:39:32 -0700174 return mMmapClient.clientPid;
Phil Burkb63320a2017-06-30 10:28:20 -0700175 }
176
Phil Burk11e8d332017-05-24 09:59:02 -0700177 aaudio_handle_t getHandle() const {
178 return mHandle;
179 }
180 void setHandle(aaudio_handle_t handle) {
181 mHandle = handle;
182 }
183
Phil Burkbbd52862018-04-13 11:37:42 -0700184 audio_port_handle_t getPortHandle() const {
185 return mClientHandle;
186 }
187
Phil Burk5a26e662017-07-07 12:44:48 -0700188 aaudio_stream_state_t getState() const {
189 return mState;
190 }
191
Phil Burk39f02dd2017-08-04 09:13:31 -0700192 void onVolumeChanged(float volume);
193
Phil Burk23296382017-11-20 15:45:11 -0800194 /**
195 * Set false when the stream is started.
196 * Set true when data is first read from the stream.
197 * @param b
198 */
199 void setFlowing(bool b) {
200 mFlowing = b;
201 }
202
203 bool isFlowing() const {
204 return mFlowing;
205 }
206
Phil Burk94862522017-09-13 21:31:36 -0700207 /**
Phil Burk762365c2018-12-10 16:02:16 -0800208 * Set false when the stream should not longer be processed.
209 * This may be caused by a message queue overflow.
210 * Set true when stream is started.
211 * @param suspended
212 */
213 void setSuspended(bool suspended) {
214 mSuspended = suspended;
215 }
216
217 bool isSuspended() const {
218 return mSuspended;
219 }
220
221 /**
Phil Burk94862522017-09-13 21:31:36 -0700222 * Atomically increment the number of active references to the stream by AAudioService.
Phil Burk2fe718b2018-05-14 12:28:32 -0700223 *
224 * This is called under a global lock in AAudioStreamTracker.
225 *
Phil Burk94862522017-09-13 21:31:36 -0700226 * @return value after the increment
227 */
Phil Burk2fe718b2018-05-14 12:28:32 -0700228 int32_t incrementServiceReferenceCount_l();
Phil Burk94862522017-09-13 21:31:36 -0700229
230 /**
231 * Atomically decrement the number of active references to the stream by AAudioService.
Phil Burk2fe718b2018-05-14 12:28:32 -0700232 * This should only be called after incrementServiceReferenceCount_l().
233 *
234 * This is called under a global lock in AAudioStreamTracker.
235 *
Phil Burk94862522017-09-13 21:31:36 -0700236 * @return value after the decrement
237 */
Phil Burk2fe718b2018-05-14 12:28:32 -0700238 int32_t decrementServiceReferenceCount_l();
Phil Burk94862522017-09-13 21:31:36 -0700239
240 bool isCloseNeeded() const {
241 return mCloseNeeded.load();
242 }
243
Phil Burk2fe718b2018-05-14 12:28:32 -0700244 /**
245 * Mark this stream as needing to be closed.
246 * Once marked for closing, it cannot be unmarked.
247 */
248 void markCloseNeeded() {
249 mCloseNeeded.store(true);
Phil Burk94862522017-09-13 21:31:36 -0700250 }
251
Phil Burk19e990e2018-03-22 13:59:34 -0700252 virtual const char *getTypeText() const { return "Base"; }
253
Phil Burk2355edb2016-12-26 13:54:02 -0800254protected:
Phil Burk98d6d922017-07-06 11:52:45 -0700255
Phil Burk39f02dd2017-08-04 09:13:31 -0700256 /**
257 * Open the device.
258 */
259 aaudio_result_t open(const aaudio::AAudioStreamRequest &request,
260 aaudio_sharing_mode_t sharingMode);
261
Phil Burk5a26e662017-07-07 12:44:48 -0700262 void setState(aaudio_stream_state_t state) {
263 mState = state;
264 }
265
Phil Burkbcc36742017-08-31 17:24:51 -0700266 /**
267 * Device specific startup.
268 * @return AAUDIO_OK or negative error.
269 */
270 virtual aaudio_result_t startDevice();
271
Phil Burkc0c70e32017-02-09 13:18:38 -0800272 aaudio_result_t writeUpMessageQueue(AAudioServiceMessage *command);
273
274 aaudio_result_t sendCurrentTimestamp();
275
Phil Burk23296382017-11-20 15:45:11 -0800276 aaudio_result_t sendXRunCount(int32_t xRunCount);
277
Phil Burk940083c2017-07-17 17:00:02 -0700278 /**
279 * @param positionFrames
280 * @param timeNanos
281 * @return AAUDIO_OK or AAUDIO_ERROR_UNAVAILABLE or other negative error
282 */
Phil Burkc0c70e32017-02-09 13:18:38 -0800283 virtual aaudio_result_t getFreeRunningPosition(int64_t *positionFrames, int64_t *timeNanos) = 0;
284
Phil Burk97350f92017-07-21 15:59:44 -0700285 virtual aaudio_result_t getHardwareTimestamp(int64_t *positionFrames, int64_t *timeNanos) = 0;
286
Phil Burk523b3042017-09-13 13:03:08 -0700287 virtual aaudio_result_t getAudioDataDescription(AudioEndpointParcelable &parcelable) = 0;
Phil Burkc0c70e32017-02-09 13:18:38 -0800288
Phil Burkec89b2e2017-06-20 15:05:06 -0700289 aaudio_stream_state_t mState = AAUDIO_STREAM_STATE_UNINITIALIZED;
Phil Burk2355edb2016-12-26 13:54:02 -0800290
Eric Laurentcb4dae22017-07-01 19:39:32 -0700291 pid_t mRegisteredClientThread = ILLEGAL_THREAD_ID;
Phil Burk2355edb2016-12-26 13:54:02 -0800292
Eric Laurentcb4dae22017-07-01 19:39:32 -0700293 SharedRingBuffer* mUpMessageQueue;
Phil Burk39f02dd2017-08-04 09:13:31 -0700294 std::mutex mUpMessageQueueLock;
Phil Burk2355edb2016-12-26 13:54:02 -0800295
Phil Burkbcc36742017-08-31 17:24:51 -0700296 AAudioThread mTimestampThread;
Phil Burkc0c70e32017-02-09 13:18:38 -0800297 // This is used by one thread to tell another thread to exit. So it must be atomic.
Phil Burk39f02dd2017-08-04 09:13:31 -0700298 std::atomic<bool> mThreadEnabled{false};
Phil Burkc0c70e32017-02-09 13:18:38 -0800299
Eric Laurentcb4dae22017-07-01 19:39:32 -0700300 int32_t mFramesPerBurst = 0;
Phil Burk39f02dd2017-08-04 09:13:31 -0700301 android::AudioClient mMmapClient; // set in open, used in MMAP start()
Phil Burkbbd52862018-04-13 11:37:42 -0700302 // TODO rename mClientHandle to mPortHandle to be more consistent with AudioFlinger.
Eric Laurentcb4dae22017-07-01 19:39:32 -0700303 audio_port_handle_t mClientHandle = AUDIO_PORT_HANDLE_NONE;
304
Phil Burka53ffa62018-10-10 16:21:37 -0700305 SimpleDoubleBuffer<Timestamp> mAtomicStreamTimestamp;
Phil Burk97350f92017-07-21 15:59:44 -0700306
Phil Burk39f02dd2017-08-04 09:13:31 -0700307 android::AAudioService &mAudioService;
Phil Burk6e2770e2018-05-01 13:03:52 -0700308
309 // The mServiceEndpoint variable can be accessed by multiple threads.
310 // So we access it by locally promoting a weak pointer to a smart pointer,
311 // which is thread-safe.
Phil Burk39f02dd2017-08-04 09:13:31 -0700312 android::sp<AAudioServiceEndpoint> mServiceEndpoint;
Phil Burk6e2770e2018-05-01 13:03:52 -0700313 android::wp<AAudioServiceEndpoint> mServiceEndpointWeak;
Phil Burk39f02dd2017-08-04 09:13:31 -0700314
Phil Burk11e8d332017-05-24 09:59:02 -0700315private:
Phil Burkf878a8d2019-03-29 17:23:00 -0700316
317 /**
318 * @return true if the queue is getting full.
319 */
320 bool isUpMessageQueueBusy();
321
Eric Laurentcb4dae22017-07-01 19:39:32 -0700322 aaudio_handle_t mHandle = -1;
Phil Burk23296382017-11-20 15:45:11 -0800323 bool mFlowing = false;
Phil Burk94862522017-09-13 21:31:36 -0700324
Phil Burk2fe718b2018-05-14 12:28:32 -0700325 // This is modified under a global lock in AAudioStreamTracker.
326 int32_t mCallingCount = 0;
327
Phil Burk762365c2018-12-10 16:02:16 -0800328 // This indicates that a stream that is being referenced by a binder call needs to closed.
Phil Burk94862522017-09-13 21:31:36 -0700329 std::atomic<bool> mCloseNeeded{false};
Phil Burk762365c2018-12-10 16:02:16 -0800330
331 // This indicate that a running stream should not be processed because of an error,
332 // for example a full message queue. Note that this atomic is unrelated to mCloseNeeded.
333 std::atomic<bool> mSuspended{false};
Phil Burk2355edb2016-12-26 13:54:02 -0800334};
335
Phil Burk5ed503c2017-02-01 09:38:15 -0800336} /* namespace aaudio */
Phil Burk2355edb2016-12-26 13:54:02 -0800337
Phil Burk5ed503c2017-02-01 09:38:15 -0800338#endif //AAUDIO_AAUDIO_SERVICE_STREAM_BASE_H