| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 1 | /* | 
|  | 2 | ** | 
|  | 3 | ** Copyright 2012, The Android Open Source Project | 
|  | 4 | ** | 
|  | 5 | ** Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | 6 | ** you may not use this file except in compliance with the License. | 
|  | 7 | ** You may obtain a copy of the License at | 
|  | 8 | ** | 
|  | 9 | **     http://www.apache.org/licenses/LICENSE-2.0 | 
|  | 10 | ** | 
|  | 11 | ** Unless required by applicable law or agreed to in writing, software | 
|  | 12 | ** distributed under the License is distributed on an "AS IS" BASIS, | 
|  | 13 | ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | 14 | ** See the License for the specific language governing permissions and | 
|  | 15 | ** limitations under the License. | 
|  | 16 | */ | 
|  | 17 |  | 
|  | 18 | #ifndef INCLUDING_FROM_AUDIOFLINGER_H | 
|  | 19 | #error This header file should only be included from AudioFlinger.h | 
|  | 20 | #endif | 
|  | 21 |  | 
|  | 22 | // base for record and playback | 
|  | 23 | class TrackBase : public ExtendedAudioBufferProvider, public RefBase { | 
|  | 24 |  | 
|  | 25 | public: | 
|  | 26 | enum track_state { | 
|  | 27 | IDLE, | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 28 | FLUSHED, | 
|  | 29 | STOPPED, | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 30 | // next 2 states are currently used for fast tracks | 
|  | 31 | // and offloaded tracks only | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 32 | STOPPING_1,     // waiting for first underrun | 
|  | 33 | STOPPING_2,     // waiting for presentation complete | 
|  | 34 | RESUMING, | 
|  | 35 | ACTIVE, | 
|  | 36 | PAUSING, | 
| Glenn Kasten | 6dd62fb | 2013-12-05 16:35:58 -0800 | [diff] [blame] | 37 | PAUSED, | 
|  | 38 | STARTING_1,     // for RecordTrack only | 
|  | 39 | STARTING_2,     // for RecordTrack only | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 40 | }; | 
|  | 41 |  | 
| Glenn Kasten | 6181ffd | 2014-05-13 10:41:52 -0700 | [diff] [blame] | 42 | // where to allocate the data buffer | 
|  | 43 | enum alloc_type { | 
|  | 44 | ALLOC_CBLK,     // allocate immediately after control block | 
|  | 45 | ALLOC_READONLY, // allocate from a separate read-only heap per thread | 
|  | 46 | ALLOC_PIPE,     // do not allocate; use the pipe buffer | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 47 | ALLOC_LOCAL,    // allocate a local buffer | 
|  | 48 | ALLOC_NONE,     // do not allocate:use the buffer passed to TrackBase constructor | 
|  | 49 | }; | 
|  | 50 |  | 
|  | 51 | enum track_type { | 
|  | 52 | TYPE_DEFAULT, | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 53 | TYPE_OUTPUT, | 
|  | 54 | TYPE_PATCH, | 
| Glenn Kasten | 6181ffd | 2014-05-13 10:41:52 -0700 | [diff] [blame] | 55 | }; | 
|  | 56 |  | 
| Andy Hung | 1bc088a | 2018-02-09 15:57:31 -0800 | [diff] [blame] | 57 | enum { | 
|  | 58 | TRACK_NAME_PENDING = -1, | 
|  | 59 | TRACK_NAME_FAILURE = -2, | 
|  | 60 | }; | 
|  | 61 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 62 | TrackBase(ThreadBase *thread, | 
|  | 63 | const sp<Client>& client, | 
| Kevin Rocard | 1f564ac | 2018-03-29 13:53:10 -0700 | [diff] [blame] | 64 | const audio_attributes_t& mAttr, | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 65 | uint32_t sampleRate, | 
|  | 66 | audio_format_t format, | 
|  | 67 | audio_channel_mask_t channelMask, | 
|  | 68 | size_t frameCount, | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 69 | void *buffer, | 
| Andy Hung | 8fe6803 | 2017-06-05 16:17:51 -0700 | [diff] [blame] | 70 | size_t bufferSize, | 
| Glenn Kasten | d848eb4 | 2016-03-08 13:42:11 -0800 | [diff] [blame] | 71 | audio_session_t sessionId, | 
| Andy Hung | 1f12a8a | 2016-11-07 16:10:30 -0800 | [diff] [blame] | 72 | uid_t uid, | 
| Glenn Kasten | d776ac6 | 2014-05-07 09:16:09 -0700 | [diff] [blame] | 73 | bool isOut, | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 74 | alloc_type alloc = ALLOC_CBLK, | 
| Eric Laurent | 20b9ef0 | 2016-12-05 11:03:16 -0800 | [diff] [blame] | 75 | track_type type = TYPE_DEFAULT, | 
|  | 76 | audio_port_handle_t portId = AUDIO_PORT_HANDLE_NONE); | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 77 | virtual             ~TrackBase(); | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 78 | virtual status_t    initCheck() const; | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 79 |  | 
|  | 80 | virtual status_t    start(AudioSystem::sync_event_t event, | 
| Glenn Kasten | d848eb4 | 2016-03-08 13:42:11 -0800 | [diff] [blame] | 81 | audio_session_t triggerSession) = 0; | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 82 | virtual void        stop() = 0; | 
|  | 83 | sp<IMemory> getCblk() const { return mCblkMemory; } | 
|  | 84 | audio_track_cblk_t* cblk() const { return mCblk; } | 
| Glenn Kasten | d848eb4 | 2016-03-08 13:42:11 -0800 | [diff] [blame] | 85 | audio_session_t sessionId() const { return mSessionId; } | 
| Andy Hung | 1f12a8a | 2016-11-07 16:10:30 -0800 | [diff] [blame] | 86 | uid_t       uid() const { return mUid; } | 
| Eric Laurent | 6acd1d4 | 2017-01-04 14:23:29 -0800 | [diff] [blame] | 87 | audio_port_handle_t portId() const { return mPortId; } | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 88 | virtual status_t    setSyncEvent(const sp<SyncEvent>& event); | 
|  | 89 |  | 
| Glenn Kasten | d776ac6 | 2014-05-07 09:16:09 -0700 | [diff] [blame] | 90 | sp<IMemory> getBuffers() const { return mBufferMemory; } | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 91 | void*       buffer() const { return mBuffer; } | 
| Andy Hung | 8fe6803 | 2017-06-05 16:17:51 -0700 | [diff] [blame] | 92 | size_t      bufferSize() const { return mBufferSize; } | 
| Eric Laurent | 0506778 | 2016-06-01 18:27:28 -0700 | [diff] [blame] | 93 | virtual bool        isFastTrack() const = 0; | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 94 | bool        isOutputTrack() const { return (mType == TYPE_OUTPUT); } | 
|  | 95 | bool        isPatchTrack() const { return (mType == TYPE_PATCH); } | 
|  | 96 | bool        isExternalTrack() const { return !isOutputTrack() && !isPatchTrack(); } | 
| Glenn Kasten | d776ac6 | 2014-05-07 09:16:09 -0700 | [diff] [blame] | 97 |  | 
| Eric Laurent | 6acd1d4 | 2017-01-04 14:23:29 -0800 | [diff] [blame] | 98 | virtual void        invalidate() { mIsInvalid = true; } | 
|  | 99 | bool        isInvalid() const { return mIsInvalid; } | 
|  | 100 |  | 
| Kevin Rocard | 069c271 | 2018-03-29 19:09:14 -0700 | [diff] [blame] | 101 | audio_attributes_t  attributes() const { return mAttr; } | 
| Eric Laurent | 6acd1d4 | 2017-01-04 14:23:29 -0800 | [diff] [blame] | 102 |  | 
| Andy Hung | 8946a28 | 2018-04-19 20:04:56 -0700 | [diff] [blame^] | 103 | #ifdef TEE_SINK | 
|  | 104 | void         dumpTee(int fd, const std::string &reason) const { | 
|  | 105 | mTee.dump(fd, reason); | 
|  | 106 | } | 
|  | 107 | #endif | 
|  | 108 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 109 | protected: | 
| Mikhail Naganov | bf49308 | 2017-04-17 17:37:12 -0700 | [diff] [blame] | 110 | DISALLOW_COPY_AND_ASSIGN(TrackBase); | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 111 |  | 
|  | 112 | // AudioBufferProvider interface | 
| Glenn Kasten | d79072e | 2016-01-06 08:41:20 -0800 | [diff] [blame] | 113 | virtual status_t getNextBuffer(AudioBufferProvider::Buffer* buffer) = 0; | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 114 | virtual void releaseBuffer(AudioBufferProvider::Buffer* buffer); | 
|  | 115 |  | 
|  | 116 | // ExtendedAudioBufferProvider interface is only needed for Track, | 
|  | 117 | // but putting it in TrackBase avoids the complexity of virtual inheritance | 
|  | 118 | virtual size_t  framesReady() const { return SIZE_MAX; } | 
|  | 119 |  | 
| Glenn Kasten | c9b2e20 | 2013-02-26 11:32:32 -0800 | [diff] [blame] | 120 | audio_format_t format() const { return mFormat; } | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 121 |  | 
|  | 122 | uint32_t channelCount() const { return mChannelCount; } | 
|  | 123 |  | 
|  | 124 | audio_channel_mask_t channelMask() const { return mChannelMask; } | 
|  | 125 |  | 
| Glenn Kasten | 9f80dd2 | 2012-12-18 15:57:32 -0800 | [diff] [blame] | 126 | virtual uint32_t sampleRate() const { return mSampleRate; } | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 127 |  | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 128 | bool isStopped() const { | 
|  | 129 | return (mState == STOPPED || mState == FLUSHED); | 
|  | 130 | } | 
|  | 131 |  | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 132 | // for fast tracks and offloaded tracks only | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 133 | bool isStopping() const { | 
|  | 134 | return mState == STOPPING_1 || mState == STOPPING_2; | 
|  | 135 | } | 
|  | 136 | bool isStopping_1() const { | 
|  | 137 | return mState == STOPPING_1; | 
|  | 138 | } | 
|  | 139 | bool isStopping_2() const { | 
|  | 140 | return mState == STOPPING_2; | 
|  | 141 | } | 
|  | 142 |  | 
|  | 143 | bool isTerminated() const { | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 144 | return mTerminated; | 
|  | 145 | } | 
|  | 146 |  | 
|  | 147 | void terminate() { | 
|  | 148 | mTerminated = true; | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 149 | } | 
|  | 150 |  | 
| Andy Hung | 2c6c3bb | 2017-06-16 14:01:45 -0700 | [diff] [blame] | 151 | // Upper case characters are final states. | 
|  | 152 | // Lower case characters are transitory. | 
|  | 153 | const char *getTrackStateString() const { | 
|  | 154 | if (isTerminated()) { | 
|  | 155 | return "T "; | 
|  | 156 | } | 
|  | 157 | switch (mState) { | 
|  | 158 | case IDLE: | 
|  | 159 | return "I "; | 
|  | 160 | case STOPPING_1: // for Fast and Offload | 
|  | 161 | return "s1"; | 
|  | 162 | case STOPPING_2: // for Fast and Offload | 
|  | 163 | return "s2"; | 
|  | 164 | case STOPPED: | 
|  | 165 | return "S "; | 
|  | 166 | case RESUMING: | 
|  | 167 | return "r "; | 
|  | 168 | case ACTIVE: | 
|  | 169 | return "A "; | 
|  | 170 | case PAUSING: | 
|  | 171 | return "p "; | 
|  | 172 | case PAUSED: | 
|  | 173 | return "P "; | 
|  | 174 | case FLUSHED: | 
|  | 175 | return "F "; | 
|  | 176 | case STARTING_1: // for RecordTrack | 
|  | 177 | return "r1"; | 
|  | 178 | case STARTING_2: // for RecordTrack | 
|  | 179 | return "r2"; | 
|  | 180 | default: | 
|  | 181 | return "? "; | 
|  | 182 | } | 
|  | 183 | } | 
|  | 184 |  | 
| Glenn Kasten | e3aa659 | 2012-12-04 12:22:46 -0800 | [diff] [blame] | 185 | bool isOut() const { return mIsOut; } | 
| Glenn Kasten | d79072e | 2016-01-06 08:41:20 -0800 | [diff] [blame] | 186 | // true for Track, false for RecordTrack, | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 187 | // this could be a track type if needed later | 
|  | 188 |  | 
|  | 189 | const wp<ThreadBase> mThread; | 
|  | 190 | /*const*/ sp<Client> mClient;   // see explanation at ~TrackBase() why not const | 
|  | 191 | sp<IMemory>         mCblkMemory; | 
|  | 192 | audio_track_cblk_t* mCblk; | 
| Glenn Kasten | d776ac6 | 2014-05-07 09:16:09 -0700 | [diff] [blame] | 193 | sp<IMemory>         mBufferMemory;  // currently non-0 for fast RecordTrack only | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 194 | void*               mBuffer;    // start of track buffer, typically in shared memory | 
| Glenn Kasten | e3aa659 | 2012-12-04 12:22:46 -0800 | [diff] [blame] | 195 | // except for OutputTrack when it is in local memory | 
| Andy Hung | 8fe6803 | 2017-06-05 16:17:51 -0700 | [diff] [blame] | 196 | size_t              mBufferSize; // size of mBuffer in bytes | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 197 | // we don't really need a lock for these | 
|  | 198 | track_state         mState; | 
| Kevin Rocard | 1f564ac | 2018-03-29 13:53:10 -0700 | [diff] [blame] | 199 | const audio_attributes_t mAttr; | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 200 | const uint32_t      mSampleRate;    // initial sample rate only; for tracks which | 
|  | 201 | // support dynamic rates, the current value is in control block | 
|  | 202 | const audio_format_t mFormat; | 
|  | 203 | const audio_channel_mask_t mChannelMask; | 
| Glenn Kasten | f6ed423 | 2013-07-16 11:16:27 -0700 | [diff] [blame] | 204 | const uint32_t      mChannelCount; | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 205 | const size_t        mFrameSize; // AudioFlinger's view of frame size in shared memory, | 
|  | 206 | // where for AudioTrack (but not AudioRecord), | 
|  | 207 | // 8-bit PCM samples are stored as 16-bit | 
|  | 208 | const size_t        mFrameCount;// size of track buffer given at createTrack() or | 
| Eric Laurent | f14db3c | 2017-12-08 14:20:36 -0800 | [diff] [blame] | 209 | // createRecord(), and then adjusted as needed | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 210 |  | 
| Glenn Kasten | d848eb4 | 2016-03-08 13:42:11 -0800 | [diff] [blame] | 211 | const audio_session_t mSessionId; | 
| Andy Hung | 1f12a8a | 2016-11-07 16:10:30 -0800 | [diff] [blame] | 212 | uid_t               mUid; | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 213 | Vector < sp<SyncEvent> >mSyncEvents; | 
| Glenn Kasten | e3aa659 | 2012-12-04 12:22:46 -0800 | [diff] [blame] | 214 | const bool          mIsOut; | 
| Eric Laurent | 5bba2f6 | 2016-03-18 11:14:14 -0700 | [diff] [blame] | 215 | sp<ServerProxy>     mServerProxy; | 
| Glenn Kasten | da6ef13 | 2013-01-10 12:31:01 -0800 | [diff] [blame] | 216 | const int           mId; | 
| Andy Hung | 8946a28 | 2018-04-19 20:04:56 -0700 | [diff] [blame^] | 217 | #ifdef TEE_SINK | 
|  | 218 | NBAIO_Tee           mTee; | 
|  | 219 | #endif | 
| Eric Laurent | bfb1b83 | 2013-01-07 09:53:42 -0800 | [diff] [blame] | 220 | bool                mTerminated; | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 221 | track_type          mType;      // must be one of TYPE_DEFAULT, TYPE_OUTPUT, TYPE_PATCH ... | 
| Eric Laurent | aaa4447 | 2014-09-12 17:41:50 -0700 | [diff] [blame] | 222 | audio_io_handle_t   mThreadIoHandle; // I/O handle of the thread the track is attached to | 
| Eric Laurent | 20b9ef0 | 2016-12-05 11:03:16 -0800 | [diff] [blame] | 223 | audio_port_handle_t mPortId; // unique ID for this track used by audio policy | 
| Eric Laurent | 6acd1d4 | 2017-01-04 14:23:29 -0800 | [diff] [blame] | 224 | bool                mIsInvalid; // non-resettable latch, set by invalidate() | 
| Eric Laurent | 83b8808 | 2014-06-20 18:31:16 -0700 | [diff] [blame] | 225 | }; | 
|  | 226 |  | 
|  | 227 | // PatchProxyBufferProvider interface is implemented by PatchTrack and PatchRecord. | 
|  | 228 | // it provides buffer access methods that map those of a ClientProxy (see AudioTrackShared.h) | 
|  | 229 | class PatchProxyBufferProvider | 
|  | 230 | { | 
|  | 231 | public: | 
|  | 232 |  | 
|  | 233 | virtual ~PatchProxyBufferProvider() {} | 
|  | 234 |  | 
|  | 235 | virtual status_t    obtainBuffer(Proxy::Buffer* buffer, | 
|  | 236 | const struct timespec *requested = NULL) = 0; | 
|  | 237 | virtual void        releaseBuffer(Proxy::Buffer* buffer) = 0; | 
| Eric Laurent | 81784c3 | 2012-11-19 14:55:58 -0800 | [diff] [blame] | 238 | }; |