/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef LIVE_SESSION_H_ #define LIVE_SESSION_H_ #include #include namespace android { struct ABuffer; struct AnotherPacketSource; struct DataSource; struct HTTPBase; struct LiveDataSource; struct M3UParser; struct PlaylistFetcher; struct Parcel; struct LiveSession : public AHandler { enum Flags { // Don't log any URLs. kFlagIncognito = 1, }; LiveSession( const sp ¬ify, uint32_t flags = 0, bool uidValid = false, uid_t uid = 0); enum StreamIndex { kAudioIndex = 0, kVideoIndex = 1, kSubtitleIndex = 2, kMaxStreams = 3, }; enum StreamType { STREAMTYPE_AUDIO = 1 << kAudioIndex, STREAMTYPE_VIDEO = 1 << kVideoIndex, STREAMTYPE_SUBTITLES = 1 << kSubtitleIndex, }; status_t dequeueAccessUnit(StreamType stream, sp *accessUnit); status_t getStreamFormat(StreamType stream, sp *format); void connectAsync( const char *url, const KeyedVector *headers = NULL); status_t disconnect(); // Blocks until seek is complete. status_t seekTo(int64_t timeUs); status_t getDuration(int64_t *durationUs) const; status_t getTrackInfo(Parcel *reply) const; status_t selectTrack(size_t index, bool select); bool isSeekable() const; bool hasDynamicDuration() const; enum { kWhatStreamsChanged, kWhatError, kWhatPrepared, kWhatPreparationFailed, }; // create a format-change discontinuity // // swap: // whether is format-change discontinuity should trigger a buffer swap sp createFormatChangeBuffer(bool swap = true); protected: virtual ~LiveSession(); virtual void onMessageReceived(const sp &msg); private: friend struct PlaylistFetcher; enum { kWhatConnect = 'conn', kWhatDisconnect = 'disc', kWhatSeek = 'seek', kWhatFetcherNotify = 'notf', kWhatCheckBandwidth = 'bndw', kWhatChangeConfiguration = 'chC0', kWhatChangeConfiguration2 = 'chC2', kWhatChangeConfiguration3 = 'chC3', kWhatFinishDisconnect2 = 'fin2', kWhatSwapped = 'swap', }; struct BandwidthItem { size_t mPlaylistIndex; unsigned long mBandwidth; }; struct FetcherInfo { sp mFetcher; int64_t mDurationUs; bool mIsPrepared; bool mToBeRemoved; }; struct StreamItem { const char *mType; AString mUri; StreamItem() : mType("") {} StreamItem(const char *type) : mType(type) {} AString uriKey() { AString key(mType); key.append("URI"); return key; } }; StreamItem mStreams[kMaxStreams]; sp mNotify; uint32_t mFlags; bool mUIDValid; uid_t mUID; bool mInPreparationPhase; sp mHTTPDataSource; KeyedVector mExtraHeaders; AString mMasterURL; Vector mBandwidthItems; ssize_t mPrevBandwidthIndex; sp mPlaylist; KeyedVector mFetcherInfos; uint32_t mStreamMask; // Masks used during reconfiguration: // mNewStreamMask: streams in the variant playlist we're switching to; // we don't want to immediately overwrite the original value. uint32_t mNewStreamMask; // mSwapMask: streams that have started to playback content in the new variant playlist; // we use this to track reconfiguration progress. uint32_t mSwapMask; KeyedVector > mPacketSources; // A second set of packet sources that buffer content for the variant we're switching to. KeyedVector > mPacketSources2; // A mutex used to serialize two sets of events: // * the swapping of packet sources in dequeueAccessUnit on the player thread, AND // * a forced bandwidth switch termination in cancelSwitch on the live looper. Mutex mSwapMutex; int32_t mCheckBandwidthGeneration; int32_t mSwitchGeneration; size_t mContinuationCounter; sp mContinuation; sp mSeekReply; int64_t mLastDequeuedTimeUs; int64_t mRealTimeBaseUs; bool mReconfigurationInProgress; bool mSwitchInProgress; uint32_t mDisconnectReplyID; uint32_t mSeekReplyID; sp addFetcher(const char *uri); void onConnect(const sp &msg); status_t onSeek(const sp &msg); void onFinishDisconnect2(); // If given a non-zero block_size (default 0), it is used to cap the number of // bytes read in from the DataSource. If given a non-NULL buffer, new content // is read into the end. // // The DataSource we read from is responsible for signaling error or EOF to help us // break out of the read loop. The DataSource can be returned to the caller, so // that the caller can reuse it for subsequent fetches (within the initially // requested range). // // For reused HTTP sources, the caller must download a file sequentially without // any overlaps or gaps to prevent reconnection. ssize_t fetchFile( const char *url, sp *out, /* request/open a file starting at range_offset for range_length bytes */ int64_t range_offset = 0, int64_t range_length = -1, /* download block size */ uint32_t block_size = 0, /* reuse DataSource if doing partial fetch */ sp *source = NULL, String8 *actualUrl = NULL); sp fetchPlaylist( const char *url, uint8_t *curPlaylistHash, bool *unchanged); size_t getBandwidthIndex(); static int SortByBandwidth(const BandwidthItem *, const BandwidthItem *); static StreamType indexToType(int idx); void changeConfiguration( int64_t timeUs, size_t bandwidthIndex, bool pickTrack = false); void onChangeConfiguration(const sp &msg); void onChangeConfiguration2(const sp &msg); void onChangeConfiguration3(const sp &msg); void onSwapped(const sp &msg); void tryToFinishBandwidthSwitch(); void scheduleCheckBandwidthEvent(); void cancelCheckBandwidthEvent(); // cancelBandwidthSwitch is atomic wrt swapPacketSource; call it to prevent packet sources // from being swapped out on stale discontinuities while manipulating // mPacketSources/mPacketSources2. void cancelBandwidthSwitch(); bool canSwitchBandwidthTo(size_t bandwidthIndex); void onCheckBandwidth(); void finishDisconnect(); void postPrepared(status_t err); void swapPacketSource(StreamType stream); bool canSwitchUp(); DISALLOW_EVIL_CONSTRUCTORS(LiveSession); }; } // namespace android #endif // LIVE_SESSION_H_