Phil Burk | 204a163 | 2017-01-03 17:23:43 -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 | |
Phil Burk | 5204d31 | 2017-05-04 17:16:13 -0700 | [diff] [blame] | 17 | #ifndef ANDROID_AAUDIO_AUDIO_STREAM_INTERNAL_H |
| 18 | #define ANDROID_AAUDIO_AUDIO_STREAM_INTERNAL_H |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 19 | |
| 20 | #include <stdint.h> |
Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 21 | #include <aaudio/AAudio.h> |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 22 | |
Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 23 | #include "binding/IAAudioService.h" |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 24 | #include "binding/AudioEndpointParcelable.h" |
Phil Burk | e572f46 | 2017-04-20 13:03:19 -0700 | [diff] [blame] | 25 | #include "binding/AAudioServiceInterface.h" |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 26 | #include "client/IsochronousClockModel.h" |
| 27 | #include "client/AudioEndpoint.h" |
| 28 | #include "core/AudioStream.h" |
Phil Burk | fd34a93 | 2017-07-19 07:03:52 -0700 | [diff] [blame] | 29 | #include "utility/AudioClock.h" |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 30 | |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 31 | using android::sp; |
Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 32 | using android::IAAudioService; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 33 | |
Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 34 | namespace aaudio { |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 35 | |
Phil Burk | 6479d50 | 2017-11-20 09:32:52 -0800 | [diff] [blame] | 36 | // These are intended to be outside the range of what is normally encountered. |
| 37 | // TODO MAXes should probably be much bigger. |
| 38 | constexpr int32_t MIN_FRAMES_PER_BURST = 16; // arbitrary |
| 39 | constexpr int32_t MAX_FRAMES_PER_BURST = 16 * 1024; // arbitrary |
| 40 | constexpr int32_t MAX_BUFFER_CAPACITY_IN_FRAMES = 32 * 1024; // arbitrary |
| 41 | |
Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 42 | // A stream that talks to the AAudioService or directly to a HAL. |
Phil Burk | 965650e | 2017-09-07 21:00:09 -0700 | [diff] [blame] | 43 | class AudioStreamInternal : public AudioStream { |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 44 | |
| 45 | public: |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 46 | AudioStreamInternal(AAudioServiceInterface &serviceInterface, bool inService); |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 47 | virtual ~AudioStreamInternal(); |
| 48 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 49 | aaudio_result_t requestStart() override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 50 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 51 | aaudio_result_t requestStop() override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 52 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 53 | aaudio_result_t getTimestamp(clockid_t clockId, |
Phil Burk | 3316d5e | 2017-02-15 11:23:01 -0800 | [diff] [blame] | 54 | int64_t *framePosition, |
| 55 | int64_t *timeNanoseconds) override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 56 | |
Phil Burk | 0befec6 | 2017-07-28 15:12:13 -0700 | [diff] [blame] | 57 | virtual aaudio_result_t updateStateMachine() override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 58 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 59 | aaudio_result_t open(const AudioStreamBuilder &builder) override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 60 | |
Phil Burk | 8b4e05e | 2019-12-17 12:12:09 -0800 | [diff] [blame] | 61 | aaudio_result_t release_l() override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 62 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 63 | aaudio_result_t setBufferSize(int32_t requestedFrames) override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 64 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 65 | int32_t getBufferSize() const override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 66 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 67 | int32_t getBufferCapacity() const override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 68 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 69 | int32_t getFramesPerBurst() const override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 70 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 71 | int32_t getXRunCount() const override { |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 72 | return mXRunCount; |
| 73 | } |
| 74 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 75 | aaudio_result_t registerThread() override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 76 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 77 | aaudio_result_t unregisterThread() override; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 78 | |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 79 | aaudio_result_t joinThread(void** returnArg); |
| 80 | |
Phil Burk | e4d7bb4 | 2017-03-28 11:32:39 -0700 | [diff] [blame] | 81 | // Called internally from 'C' |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 82 | virtual void *callbackLoop() = 0; |
Phil Burk | e4d7bb4 | 2017-03-28 11:32:39 -0700 | [diff] [blame] | 83 | |
Phil Burk | e2fbb59 | 2017-05-01 15:05:52 -0700 | [diff] [blame] | 84 | |
| 85 | bool isMMap() override { |
| 86 | return true; |
| 87 | } |
| 88 | |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 89 | // Calculate timeout based on framesPerBurst |
| 90 | int64_t calculateReasonableTimeout(); |
| 91 | |
Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 92 | aaudio_result_t startClient(const android::AudioClient& client, |
jiabin | d1f1cb6 | 2020-03-24 11:57:57 -0700 | [diff] [blame] | 93 | const audio_attributes_t *attr, |
Eric Laurent | cb4dae2 | 2017-07-01 19:39:32 -0700 | [diff] [blame] | 94 | audio_port_handle_t *clientHandle); |
| 95 | |
| 96 | aaudio_result_t stopClient(audio_port_handle_t clientHandle); |
| 97 | |
Phil Burk | 39f02dd | 2017-08-04 09:13:31 -0700 | [diff] [blame] | 98 | aaudio_handle_t getServiceHandle() const { |
| 99 | return mServiceStreamHandle; |
| 100 | } |
| 101 | |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 102 | protected: |
| 103 | |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 104 | aaudio_result_t processData(void *buffer, |
| 105 | int32_t numFrames, |
| 106 | int64_t timeoutNanoseconds); |
| 107 | |
| 108 | /** |
| 109 | * Low level data processing that will not block. It will just read or write as much as it can. |
| 110 | * |
| 111 | * It passed back a recommended time to wake up if wakeTimePtr is not NULL. |
| 112 | * |
| 113 | * @return the number of frames processed or a negative error code. |
| 114 | */ |
| 115 | virtual aaudio_result_t processDataNow(void *buffer, |
| 116 | int32_t numFrames, |
| 117 | int64_t currentTimeNanos, |
| 118 | int64_t *wakeTimePtr) = 0; |
| 119 | |
Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 120 | aaudio_result_t drainTimestampsFromService(); |
| 121 | |
Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 122 | aaudio_result_t processCommands(); |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 123 | |
Phil Burk | e4d7bb4 | 2017-03-28 11:32:39 -0700 | [diff] [blame] | 124 | aaudio_result_t stopCallback(); |
| 125 | |
Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 126 | virtual void advanceClientToMatchServerPosition() = 0; |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 127 | |
Phil Burk | b336e89 | 2017-07-05 15:35:43 -0700 | [diff] [blame] | 128 | virtual void onFlushFromServer() {} |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 129 | |
Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 130 | aaudio_result_t onEventFromServer(AAudioServiceMessage *message); |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 131 | |
Phil Burk | 97350f9 | 2017-07-21 15:59:44 -0700 | [diff] [blame] | 132 | aaudio_result_t onTimestampService(AAudioServiceMessage *message); |
| 133 | |
| 134 | aaudio_result_t onTimestampHardware(AAudioServiceMessage *message); |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 135 | |
Phil Burk | ec89b2e | 2017-06-20 15:05:06 -0700 | [diff] [blame] | 136 | void logTimestamp(AAudioServiceMessage &message); |
| 137 | |
Phil Burk | e4d7bb4 | 2017-03-28 11:32:39 -0700 | [diff] [blame] | 138 | // Calculate timeout for an operation involving framesPerOperation. |
| 139 | int64_t calculateReasonableTimeout(int32_t framesPerOperation); |
| 140 | |
Phil Burk | 41f19d8 | 2018-02-13 14:59:10 -0800 | [diff] [blame] | 141 | int32_t getDeviceChannelCount() const { return mDeviceChannelCount; } |
| 142 | |
| 143 | /** |
| 144 | * @return true if running in audio service, versus in app process |
| 145 | */ |
| 146 | bool isInService() const { return mInService; } |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 147 | |
Phil Burk | 377c1c2 | 2018-12-12 16:06:54 -0800 | [diff] [blame] | 148 | /** |
| 149 | * Is the service FIFO position currently controlled by the AAudio service or HAL, |
| 150 | * or set based on the Clock Model. |
| 151 | * |
| 152 | * @return true if the ClockModel is currently determining the FIFO position |
| 153 | */ |
| 154 | bool isClockModelInControl() const; |
| 155 | |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 156 | IsochronousClockModel mClockModel; // timing model for chasing the HAL |
| 157 | |
Phil Burk | 5edc4ea | 2020-04-17 08:15:42 -0700 | [diff] [blame] | 158 | std::unique_ptr<AudioEndpoint> mAudioEndpoint; // source for reads or sink for writes |
| 159 | |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 160 | aaudio_handle_t mServiceStreamHandle; // opaque handle returned from service |
| 161 | |
Phil Burk | 6479d50 | 2017-11-20 09:32:52 -0800 | [diff] [blame] | 162 | int32_t mFramesPerBurst = MIN_FRAMES_PER_BURST; // frames per HAL transfer |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 163 | int32_t mXRunCount = 0; // how many underrun events? |
| 164 | |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 165 | // Offset from underlying frame position. |
| 166 | int64_t mFramesOffsetFromService = 0; // offset for timestamps |
| 167 | |
Phil Burk | bf821e2 | 2020-04-17 11:51:43 -0700 | [diff] [blame] | 168 | std::unique_ptr<uint8_t[]> mCallbackBuffer; |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 169 | int32_t mCallbackFrames = 0; |
| 170 | |
Phil Burk | ec89b2e | 2017-06-20 15:05:06 -0700 | [diff] [blame] | 171 | // The service uses this for SHARED mode. |
| 172 | bool mInService = false; // Is this running in the client or the service? |
| 173 | |
Phil Burk | b336e89 | 2017-07-05 15:35:43 -0700 | [diff] [blame] | 174 | AAudioServiceInterface &mServiceInterface; // abstract interface to the service |
| 175 | |
Phil Burk | a53ffa6 | 2018-10-10 16:21:37 -0700 | [diff] [blame] | 176 | SimpleDoubleBuffer<Timestamp> mAtomicInternalTimestamp; |
Phil Burk | bcc3674 | 2017-08-31 17:24:51 -0700 | [diff] [blame] | 177 | |
| 178 | AtomicRequestor mNeedCatchUp; // Ask read() or write() to sync on first timestamp. |
| 179 | |
Phil Burk | 965650e | 2017-09-07 21:00:09 -0700 | [diff] [blame] | 180 | float mStreamVolume = 1.0f; |
| 181 | |
Phil Burk | 5edc4ea | 2020-04-17 08:15:42 -0700 | [diff] [blame] | 182 | int64_t mLastFramesWritten = 0; |
| 183 | int64_t mLastFramesRead = 0; |
| 184 | |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 185 | private: |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 186 | /* |
| 187 | * Asynchronous write with data conversion. |
| 188 | * @param buffer |
| 189 | * @param numFrames |
| 190 | * @return fdrames written or negative error |
| 191 | */ |
| 192 | aaudio_result_t writeNowWithConversion(const void *buffer, |
| 193 | int32_t numFrames); |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 194 | |
Phil Burk | 87c9f64 | 2017-05-17 07:22:39 -0700 | [diff] [blame] | 195 | // Adjust timing model based on timestamp from service. |
| 196 | void processTimestamp(uint64_t position, int64_t time); |
Phil Burk | 71f35bb | 2017-04-13 16:05:07 -0700 | [diff] [blame] | 197 | |
Phil Burk | fd34a93 | 2017-07-19 07:03:52 -0700 | [diff] [blame] | 198 | // Thread on other side of FIFO will have wakeup jitter. |
| 199 | // By delaying slightly we can avoid waking up before other side is ready. |
| 200 | const int32_t mWakeupDelayNanos; // delay past typical wakeup jitter |
| 201 | const int32_t mMinimumSleepNanos; // minimum sleep while polling |
Phil Burk | b31b66f | 2019-09-30 09:33:41 -0700 | [diff] [blame] | 202 | int32_t mTimeOffsetNanos = 0; // add to time part of an MMAP timestamp |
Phil Burk | fd34a93 | 2017-07-19 07:03:52 -0700 | [diff] [blame] | 203 | |
Phil Burk | c0c70e3 | 2017-02-09 13:18:38 -0800 | [diff] [blame] | 204 | AudioEndpointParcelable mEndPointParcelable; // description of the buffers filled by service |
| 205 | EndpointDescriptor mEndpointDescriptor; // buffer description with resolved addresses |
Phil Burk | 97350f9 | 2017-07-21 15:59:44 -0700 | [diff] [blame] | 206 | |
Phil Burk | 97350f9 | 2017-07-21 15:59:44 -0700 | [diff] [blame] | 207 | int64_t mServiceLatencyNanos = 0; |
Phil Burk | 41f19d8 | 2018-02-13 14:59:10 -0800 | [diff] [blame] | 208 | |
Phil Burk | 0127c1b | 2018-03-29 13:48:06 -0700 | [diff] [blame] | 209 | // Sometimes the hardware is operating with a different channel count from the app. |
| 210 | // Then we require conversion in AAudio. |
Phil Burk | 41f19d8 | 2018-02-13 14:59:10 -0800 | [diff] [blame] | 211 | int32_t mDeviceChannelCount = 0; |
Phil Burk | 8d4f006 | 2019-10-03 15:55:41 -0700 | [diff] [blame] | 212 | |
| 213 | int32_t mBufferSizeInFrames = 0; // local threshold to control latency |
Phil Burk | 5edc4ea | 2020-04-17 08:15:42 -0700 | [diff] [blame] | 214 | int32_t mBufferCapacityInFrames = 0; |
| 215 | |
Phil Burk | 8d4f006 | 2019-10-03 15:55:41 -0700 | [diff] [blame] | 216 | |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 217 | }; |
| 218 | |
Phil Burk | 5ed503c | 2017-02-01 09:38:15 -0800 | [diff] [blame] | 219 | } /* namespace aaudio */ |
Phil Burk | 204a163 | 2017-01-03 17:23:43 -0800 | [diff] [blame] | 220 | |
Phil Burk | 5204d31 | 2017-05-04 17:16:13 -0700 | [diff] [blame] | 221 | #endif //ANDROID_AAUDIO_AUDIO_STREAM_INTERNAL_H |