blob: dafba1ee9d02e9cacf8c9af92d206bc1107e2542 [file] [log] [blame]
Eric Laurent81784c32012-11-19 14:55:58 -08001/*
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
23class TrackBase : public ExtendedAudioBufferProvider, public RefBase {
24
25public:
26 enum track_state {
27 IDLE,
Eric Laurent81784c32012-11-19 14:55:58 -080028 FLUSHED,
29 STOPPED,
Eric Laurentbfb1b832013-01-07 09:53:42 -080030 // next 2 states are currently used for fast tracks
31 // and offloaded tracks only
Eric Laurent81784c32012-11-19 14:55:58 -080032 STOPPING_1, // waiting for first underrun
33 STOPPING_2, // waiting for presentation complete
34 RESUMING,
35 ACTIVE,
36 PAUSING,
Glenn Kasten6dd62fb2013-12-05 16:35:58 -080037 PAUSED,
38 STARTING_1, // for RecordTrack only
39 STARTING_2, // for RecordTrack only
Eric Laurent81784c32012-11-19 14:55:58 -080040 };
41
Glenn Kasten6181ffd2014-05-13 10:41:52 -070042 // 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 Laurent83b88082014-06-20 18:31:16 -070047 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 Laurent83b88082014-06-20 18:31:16 -070053 TYPE_OUTPUT,
54 TYPE_PATCH,
Glenn Kasten6181ffd2014-05-13 10:41:52 -070055 };
56
Andy Hung1bc088a2018-02-09 15:57:31 -080057 enum {
58 TRACK_NAME_PENDING = -1,
59 TRACK_NAME_FAILURE = -2,
60 };
61
Eric Laurent81784c32012-11-19 14:55:58 -080062 TrackBase(ThreadBase *thread,
63 const sp<Client>& client,
Kevin Rocard1f564ac2018-03-29 13:53:10 -070064 const audio_attributes_t& mAttr,
Eric Laurent81784c32012-11-19 14:55:58 -080065 uint32_t sampleRate,
66 audio_format_t format,
67 audio_channel_mask_t channelMask,
68 size_t frameCount,
Eric Laurent83b88082014-06-20 18:31:16 -070069 void *buffer,
Andy Hung8fe68032017-06-05 16:17:51 -070070 size_t bufferSize,
Glenn Kastend848eb42016-03-08 13:42:11 -080071 audio_session_t sessionId,
Andy Hung1f12a8a2016-11-07 16:10:30 -080072 uid_t uid,
Glenn Kastend776ac62014-05-07 09:16:09 -070073 bool isOut,
Eric Laurent83b88082014-06-20 18:31:16 -070074 alloc_type alloc = ALLOC_CBLK,
Eric Laurent20b9ef02016-12-05 11:03:16 -080075 track_type type = TYPE_DEFAULT,
76 audio_port_handle_t portId = AUDIO_PORT_HANDLE_NONE);
Eric Laurent81784c32012-11-19 14:55:58 -080077 virtual ~TrackBase();
Eric Laurent83b88082014-06-20 18:31:16 -070078 virtual status_t initCheck() const;
Eric Laurent81784c32012-11-19 14:55:58 -080079
80 virtual status_t start(AudioSystem::sync_event_t event,
Glenn Kastend848eb42016-03-08 13:42:11 -080081 audio_session_t triggerSession) = 0;
Eric Laurent81784c32012-11-19 14:55:58 -080082 virtual void stop() = 0;
83 sp<IMemory> getCblk() const { return mCblkMemory; }
84 audio_track_cblk_t* cblk() const { return mCblk; }
Glenn Kastend848eb42016-03-08 13:42:11 -080085 audio_session_t sessionId() const { return mSessionId; }
Andy Hung1f12a8a2016-11-07 16:10:30 -080086 uid_t uid() const { return mUid; }
Eric Laurent6acd1d42017-01-04 14:23:29 -080087 audio_port_handle_t portId() const { return mPortId; }
Eric Laurent81784c32012-11-19 14:55:58 -080088 virtual status_t setSyncEvent(const sp<SyncEvent>& event);
89
Glenn Kastend776ac62014-05-07 09:16:09 -070090 sp<IMemory> getBuffers() const { return mBufferMemory; }
Eric Laurent83b88082014-06-20 18:31:16 -070091 void* buffer() const { return mBuffer; }
Andy Hung8fe68032017-06-05 16:17:51 -070092 size_t bufferSize() const { return mBufferSize; }
Eric Laurent05067782016-06-01 18:27:28 -070093 virtual bool isFastTrack() const = 0;
Eric Laurent83b88082014-06-20 18:31:16 -070094 bool isOutputTrack() const { return (mType == TYPE_OUTPUT); }
95 bool isPatchTrack() const { return (mType == TYPE_PATCH); }
96 bool isExternalTrack() const { return !isOutputTrack() && !isPatchTrack(); }
Glenn Kastend776ac62014-05-07 09:16:09 -070097
Eric Laurent6acd1d42017-01-04 14:23:29 -080098 virtual void invalidate() { mIsInvalid = true; }
99 bool isInvalid() const { return mIsInvalid; }
100
Kevin Rocard069c2712018-03-29 19:09:14 -0700101 audio_attributes_t attributes() const { return mAttr; }
Eric Laurent6acd1d42017-01-04 14:23:29 -0800102
Andy Hung8946a282018-04-19 20:04:56 -0700103#ifdef TEE_SINK
104 void dumpTee(int fd, const std::string &reason) const {
105 mTee.dump(fd, reason);
106 }
107#endif
108
Andy Hungcef2daa2018-06-01 15:31:49 -0700109 /** returns the buffer contents size converted to time in milliseconds
110 * for PCM Playback or Record streaming tracks. The return value is zero for
111 * PCM static tracks and not defined for non-PCM tracks.
112 *
113 * This may be called without the thread lock.
114 */
115 virtual double bufferLatencyMs() const {
116 return mServerProxy->framesReadySafe() * 1000 / sampleRate();
117 }
118
119 /** returns whether the track supports server latency computation.
120 * This is set in the constructor and constant throughout the track lifetime.
121 */
122
123 bool isServerLatencySupported() const { return mServerLatencySupported; }
124
125 /** computes the server latency for PCM Playback or Record track
126 * to the device sink/source. This is the time for the next frame in the track buffer
127 * written or read from the server thread to the device source or sink.
128 *
129 * This may be called without the thread lock, but latencyMs and fromTrack
130 * may be not be synchronized. For example PatchPanel may not obtain the
131 * thread lock before calling.
132 *
133 * \param latencyMs on success is set to the latency in milliseconds of the
134 * next frame written/read by the server thread to/from the track buffer
135 * from the device source/sink.
136 * \param fromTrack on success is set to true if latency was computed directly
137 * from the track timestamp; otherwise set to false if latency was
138 * estimated from the server timestamp.
139 * fromTrack may be nullptr or omitted if not required.
140 *
141 * \returns OK or INVALID_OPERATION on failure.
142 */
143 status_t getServerLatencyMs(double *latencyMs, bool *fromTrack = nullptr) const {
144 if (!isServerLatencySupported()) {
145 return INVALID_OPERATION;
146 }
147
148 // if no thread lock is acquired, these atomics are not
149 // synchronized with each other, considered a benign race.
150
151 const double serverLatencyMs = mServerLatencyMs.load();
152 if (serverLatencyMs == 0.) {
153 return INVALID_OPERATION;
154 }
155 if (fromTrack != nullptr) {
156 *fromTrack = mServerLatencyFromTrack.load();
157 }
158 *latencyMs = serverLatencyMs;
159 return OK;
160 }
161
162 /** computes the total client latency for PCM Playback or Record tracks
163 * for the next client app access to the device sink/source; i.e. the
164 * server latency plus the buffer latency.
165 *
166 * This may be called without the thread lock, but latencyMs and fromTrack
167 * may be not be synchronized. For example PatchPanel may not obtain the
168 * thread lock before calling.
169 *
170 * \param latencyMs on success is set to the latency in milliseconds of the
171 * next frame written/read by the client app to/from the track buffer
172 * from the device sink/source.
173 * \param fromTrack on success is set to true if latency was computed directly
174 * from the track timestamp; otherwise set to false if latency was
175 * estimated from the server timestamp.
176 * fromTrack may be nullptr or omitted if not required.
177 *
178 * \returns OK or INVALID_OPERATION on failure.
179 */
180 status_t getTrackLatencyMs(double *latencyMs, bool *fromTrack = nullptr) const {
181 double serverLatencyMs;
182 status_t status = getServerLatencyMs(&serverLatencyMs, fromTrack);
183 if (status == OK) {
184 *latencyMs = serverLatencyMs + bufferLatencyMs();
185 }
186 return status;
187 }
188
Eric Laurent81784c32012-11-19 14:55:58 -0800189protected:
Mikhail Naganovbf493082017-04-17 17:37:12 -0700190 DISALLOW_COPY_AND_ASSIGN(TrackBase);
Eric Laurent81784c32012-11-19 14:55:58 -0800191
192 // AudioBufferProvider interface
Glenn Kastend79072e2016-01-06 08:41:20 -0800193 virtual status_t getNextBuffer(AudioBufferProvider::Buffer* buffer) = 0;
Eric Laurent81784c32012-11-19 14:55:58 -0800194 virtual void releaseBuffer(AudioBufferProvider::Buffer* buffer);
195
196 // ExtendedAudioBufferProvider interface is only needed for Track,
197 // but putting it in TrackBase avoids the complexity of virtual inheritance
198 virtual size_t framesReady() const { return SIZE_MAX; }
199
Glenn Kastenc9b2e202013-02-26 11:32:32 -0800200 audio_format_t format() const { return mFormat; }
Eric Laurent81784c32012-11-19 14:55:58 -0800201
202 uint32_t channelCount() const { return mChannelCount; }
203
204 audio_channel_mask_t channelMask() const { return mChannelMask; }
205
Glenn Kasten9f80dd22012-12-18 15:57:32 -0800206 virtual uint32_t sampleRate() const { return mSampleRate; }
Eric Laurent81784c32012-11-19 14:55:58 -0800207
Eric Laurent81784c32012-11-19 14:55:58 -0800208 bool isStopped() const {
209 return (mState == STOPPED || mState == FLUSHED);
210 }
211
Eric Laurentbfb1b832013-01-07 09:53:42 -0800212 // for fast tracks and offloaded tracks only
Eric Laurent81784c32012-11-19 14:55:58 -0800213 bool isStopping() const {
214 return mState == STOPPING_1 || mState == STOPPING_2;
215 }
216 bool isStopping_1() const {
217 return mState == STOPPING_1;
218 }
219 bool isStopping_2() const {
220 return mState == STOPPING_2;
221 }
222
223 bool isTerminated() const {
Eric Laurentbfb1b832013-01-07 09:53:42 -0800224 return mTerminated;
225 }
226
227 void terminate() {
228 mTerminated = true;
Eric Laurent81784c32012-11-19 14:55:58 -0800229 }
230
Andy Hung2c6c3bb2017-06-16 14:01:45 -0700231 // Upper case characters are final states.
232 // Lower case characters are transitory.
233 const char *getTrackStateString() const {
234 if (isTerminated()) {
235 return "T ";
236 }
237 switch (mState) {
238 case IDLE:
239 return "I ";
240 case STOPPING_1: // for Fast and Offload
241 return "s1";
242 case STOPPING_2: // for Fast and Offload
243 return "s2";
244 case STOPPED:
245 return "S ";
246 case RESUMING:
247 return "r ";
248 case ACTIVE:
249 return "A ";
250 case PAUSING:
251 return "p ";
252 case PAUSED:
253 return "P ";
254 case FLUSHED:
255 return "F ";
256 case STARTING_1: // for RecordTrack
257 return "r1";
258 case STARTING_2: // for RecordTrack
259 return "r2";
260 default:
261 return "? ";
262 }
263 }
264
Glenn Kastene3aa6592012-12-04 12:22:46 -0800265 bool isOut() const { return mIsOut; }
Glenn Kastend79072e2016-01-06 08:41:20 -0800266 // true for Track, false for RecordTrack,
Eric Laurent81784c32012-11-19 14:55:58 -0800267 // this could be a track type if needed later
268
269 const wp<ThreadBase> mThread;
270 /*const*/ sp<Client> mClient; // see explanation at ~TrackBase() why not const
271 sp<IMemory> mCblkMemory;
272 audio_track_cblk_t* mCblk;
Glenn Kastend776ac62014-05-07 09:16:09 -0700273 sp<IMemory> mBufferMemory; // currently non-0 for fast RecordTrack only
Eric Laurent81784c32012-11-19 14:55:58 -0800274 void* mBuffer; // start of track buffer, typically in shared memory
Glenn Kastene3aa6592012-12-04 12:22:46 -0800275 // except for OutputTrack when it is in local memory
Andy Hung8fe68032017-06-05 16:17:51 -0700276 size_t mBufferSize; // size of mBuffer in bytes
Eric Laurent81784c32012-11-19 14:55:58 -0800277 // we don't really need a lock for these
278 track_state mState;
Kevin Rocard1f564ac2018-03-29 13:53:10 -0700279 const audio_attributes_t mAttr;
Eric Laurent81784c32012-11-19 14:55:58 -0800280 const uint32_t mSampleRate; // initial sample rate only; for tracks which
281 // support dynamic rates, the current value is in control block
282 const audio_format_t mFormat;
283 const audio_channel_mask_t mChannelMask;
Glenn Kastenf6ed4232013-07-16 11:16:27 -0700284 const uint32_t mChannelCount;
Eric Laurent81784c32012-11-19 14:55:58 -0800285 const size_t mFrameSize; // AudioFlinger's view of frame size in shared memory,
286 // where for AudioTrack (but not AudioRecord),
287 // 8-bit PCM samples are stored as 16-bit
288 const size_t mFrameCount;// size of track buffer given at createTrack() or
Eric Laurentf14db3c2017-12-08 14:20:36 -0800289 // createRecord(), and then adjusted as needed
Eric Laurent81784c32012-11-19 14:55:58 -0800290
Glenn Kastend848eb42016-03-08 13:42:11 -0800291 const audio_session_t mSessionId;
Andy Hung1f12a8a2016-11-07 16:10:30 -0800292 uid_t mUid;
Eric Laurent81784c32012-11-19 14:55:58 -0800293 Vector < sp<SyncEvent> >mSyncEvents;
Glenn Kastene3aa6592012-12-04 12:22:46 -0800294 const bool mIsOut;
Eric Laurent5bba2f62016-03-18 11:14:14 -0700295 sp<ServerProxy> mServerProxy;
Glenn Kastenda6ef132013-01-10 12:31:01 -0800296 const int mId;
Andy Hung8946a282018-04-19 20:04:56 -0700297#ifdef TEE_SINK
298 NBAIO_Tee mTee;
299#endif
Eric Laurentbfb1b832013-01-07 09:53:42 -0800300 bool mTerminated;
Eric Laurent83b88082014-06-20 18:31:16 -0700301 track_type mType; // must be one of TYPE_DEFAULT, TYPE_OUTPUT, TYPE_PATCH ...
Eric Laurentaaa44472014-09-12 17:41:50 -0700302 audio_io_handle_t mThreadIoHandle; // I/O handle of the thread the track is attached to
Eric Laurent20b9ef02016-12-05 11:03:16 -0800303 audio_port_handle_t mPortId; // unique ID for this track used by audio policy
Eric Laurent6acd1d42017-01-04 14:23:29 -0800304 bool mIsInvalid; // non-resettable latch, set by invalidate()
Andy Hungcef2daa2018-06-01 15:31:49 -0700305
306 bool mServerLatencySupported = false;
307 std::atomic<bool> mServerLatencyFromTrack{}; // latency from track or server timestamp.
308 std::atomic<double> mServerLatencyMs{}; // last latency pushed from server thread.
Eric Laurent83b88082014-06-20 18:31:16 -0700309};
310
311// PatchProxyBufferProvider interface is implemented by PatchTrack and PatchRecord.
312// it provides buffer access methods that map those of a ClientProxy (see AudioTrackShared.h)
313class PatchProxyBufferProvider
314{
315public:
316
317 virtual ~PatchProxyBufferProvider() {}
318
319 virtual status_t obtainBuffer(Proxy::Buffer* buffer,
320 const struct timespec *requested = NULL) = 0;
321 virtual void releaseBuffer(Proxy::Buffer* buffer) = 0;
Eric Laurent81784c32012-11-19 14:55:58 -0800322};