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, |
| 28 | TERMINATED, |
| 29 | FLUSHED, |
| 30 | STOPPED, |
| 31 | // next 2 states are currently used for fast tracks only |
| 32 | STOPPING_1, // waiting for first underrun |
| 33 | STOPPING_2, // waiting for presentation complete |
| 34 | RESUMING, |
| 35 | ACTIVE, |
| 36 | PAUSING, |
| 37 | PAUSED |
| 38 | }; |
| 39 | |
| 40 | TrackBase(ThreadBase *thread, |
| 41 | const sp<Client>& client, |
| 42 | uint32_t sampleRate, |
| 43 | audio_format_t format, |
| 44 | audio_channel_mask_t channelMask, |
| 45 | size_t frameCount, |
| 46 | const sp<IMemory>& sharedBuffer, |
| 47 | int sessionId); |
| 48 | virtual ~TrackBase(); |
| 49 | |
| 50 | virtual status_t start(AudioSystem::sync_event_t event, |
| 51 | int triggerSession) = 0; |
| 52 | virtual void stop() = 0; |
| 53 | sp<IMemory> getCblk() const { return mCblkMemory; } |
| 54 | audio_track_cblk_t* cblk() const { return mCblk; } |
| 55 | int sessionId() const { return mSessionId; } |
| 56 | virtual status_t setSyncEvent(const sp<SyncEvent>& event); |
| 57 | |
| 58 | protected: |
| 59 | TrackBase(const TrackBase&); |
| 60 | TrackBase& operator = (const TrackBase&); |
| 61 | |
| 62 | // AudioBufferProvider interface |
| 63 | virtual status_t getNextBuffer(AudioBufferProvider::Buffer* buffer, int64_t pts) = 0; |
| 64 | virtual void releaseBuffer(AudioBufferProvider::Buffer* buffer); |
| 65 | |
| 66 | // ExtendedAudioBufferProvider interface is only needed for Track, |
| 67 | // but putting it in TrackBase avoids the complexity of virtual inheritance |
| 68 | virtual size_t framesReady() const { return SIZE_MAX; } |
| 69 | |
| 70 | audio_format_t format() const { |
| 71 | return mFormat; |
| 72 | } |
| 73 | |
| 74 | uint32_t channelCount() const { return mChannelCount; } |
| 75 | |
| 76 | audio_channel_mask_t channelMask() const { return mChannelMask; } |
| 77 | |
| 78 | uint32_t sampleRate() const; // FIXME inline after cblk sr moved |
| 79 | |
| 80 | // Return a pointer to the start of a contiguous slice of the track buffer. |
| 81 | // Parameter 'offset' is the requested start position, expressed in |
| 82 | // monotonically increasing frame units relative to the track epoch. |
| 83 | // Parameter 'frames' is the requested length, also in frame units. |
| 84 | // Always returns non-NULL. It is the caller's responsibility to |
| 85 | // verify that this will be successful; the result of calling this |
| 86 | // function with invalid 'offset' or 'frames' is undefined. |
| 87 | void* getBuffer(uint32_t offset, uint32_t frames) const; |
| 88 | |
| 89 | bool isStopped() const { |
| 90 | return (mState == STOPPED || mState == FLUSHED); |
| 91 | } |
| 92 | |
| 93 | // for fast tracks only |
| 94 | bool isStopping() const { |
| 95 | return mState == STOPPING_1 || mState == STOPPING_2; |
| 96 | } |
| 97 | bool isStopping_1() const { |
| 98 | return mState == STOPPING_1; |
| 99 | } |
| 100 | bool isStopping_2() const { |
| 101 | return mState == STOPPING_2; |
| 102 | } |
| 103 | |
| 104 | bool isTerminated() const { |
| 105 | return mState == TERMINATED; |
| 106 | } |
| 107 | |
| 108 | bool step(); // mStepCount is an implicit input |
| 109 | void reset(); |
| 110 | |
| 111 | virtual bool isOut() const = 0; // true for Track and TimedTrack, false for RecordTrack, |
| 112 | // this could be a track type if needed later |
| 113 | |
| 114 | const wp<ThreadBase> mThread; |
| 115 | /*const*/ sp<Client> mClient; // see explanation at ~TrackBase() why not const |
| 116 | sp<IMemory> mCblkMemory; |
| 117 | audio_track_cblk_t* mCblk; |
| 118 | void* mBuffer; // start of track buffer, typically in shared memory |
| 119 | void* mBufferEnd; // &mBuffer[mFrameCount * frameSize], where frameSize |
| 120 | // is based on mChannelCount and 16-bit samples |
| 121 | uint32_t mStepCount; // saves AudioBufferProvider::Buffer::frameCount as of |
| 122 | // time of releaseBuffer() for later use by step() |
| 123 | // we don't really need a lock for these |
| 124 | track_state mState; |
| 125 | const uint32_t mSampleRate; // initial sample rate only; for tracks which |
| 126 | // support dynamic rates, the current value is in control block |
| 127 | const audio_format_t mFormat; |
| 128 | const audio_channel_mask_t mChannelMask; |
| 129 | const uint8_t mChannelCount; |
| 130 | const size_t mFrameSize; // AudioFlinger's view of frame size in shared memory, |
| 131 | // where for AudioTrack (but not AudioRecord), |
| 132 | // 8-bit PCM samples are stored as 16-bit |
| 133 | const size_t mFrameCount;// size of track buffer given at createTrack() or |
| 134 | // openRecord(), and then adjusted as needed |
| 135 | |
| 136 | bool mStepServerFailed; |
| 137 | const int mSessionId; |
| 138 | Vector < sp<SyncEvent> >mSyncEvents; |
| 139 | }; |