| // Copyright 2012 The Cobalt Authors. All Rights Reserved. |
| // |
| // 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 COBALT_MEDIA_PROGRESSIVE_MP4_MAP_H_ |
| #define COBALT_MEDIA_PROGRESSIVE_MP4_MAP_H_ |
| |
| #include <vector> |
| |
| #include "base/callback.h" |
| #include "base/memory/ref_counted.h" |
| #include "cobalt/media/progressive/data_source_reader.h" |
| |
| namespace cobalt { |
| namespace media { |
| |
| // per-atom sizes of individual table entries |
| static const int kEntrySize_co64 = 8; |
| static const int kEntrySize_ctts = 8; |
| static const int kEntrySize_stco = 4; |
| static const int kEntrySize_stts = 8; |
| static const int kEntrySize_stsc = 12; |
| static const int kEntrySize_stss = 4; |
| static const int kEntrySize_stsz = 4; |
| |
| // Utility class to parse the various subatoms of the stbl mp4 atom and use |
| // them to provide byte offsets, sizes, and timestamps of a mp4 atom. The |
| // caching design benefits from, but does not require, sequential access |
| // in sample numbers. |
| class MP4Map : public base::RefCountedThreadSafe<MP4Map> { |
| public: |
| explicit MP4Map(scoped_refptr<DataSourceReader> reader); |
| |
| bool IsComplete(); |
| |
| // All Get() methods return true on success and save their values by |
| // reference in the latter argument. |
| bool GetSize(uint32 sample_number, uint32* size_out); |
| bool GetOffset(uint32 sample_number, uint64* offset_out); |
| // all time values in *ticks* as defined by the mp4 container |
| bool GetTimestamp(uint32 sample_number, uint64* timestamp_out); |
| bool GetDuration(uint32 sample_number, uint32* duration_out); |
| bool GetIsKeyframe(uint32 sample_number, bool* is_keyframe_out); |
| // Used to determine if the failure reported by any of the above methods |
| // is due to EOS or other (fatal) error. The length of a mp4 file in samples |
| // may not be known until iterating through almost the entire map, in the |
| // case of a default sample size (rare in compressed media) |
| bool IsEOS(uint32 sample_number); |
| |
| // Returns the keyframe sample number nearest the provided timestamp |
| bool GetKeyframe(uint64 timestamp, uint32* sample_out); |
| |
| // pass 0 as cache_size_entries to force caching of the entire map. |
| bool SetAtom(uint32 four_cc, // fourCC code ascii code as big-endian uint32 |
| uint64 offset, // offset of atom body in file |
| uint64 size, // total size of atom in bytes |
| uint32 cache_size_entries, // num of entries to cache in memory |
| const uint8* atom); // pointer to atom body start |
| |
| private: |
| bool co64_Init(); |
| |
| bool ctts_Init(); |
| // advance the ctts cache and integration state to contain sample number. |
| bool ctts_AdvanceToSample(uint32 sample_number); |
| bool ctts_SlipCacheToSample(uint32 sample_number, int starting_cache_index); |
| |
| bool stco_Init(); |
| |
| bool stsc_Init(); |
| // advance the stsc cache and integration state to contain sample number. |
| bool stsc_AdvanceToSample(uint32 sample_number); |
| // re-use previously calculated sums to jump through the table to get to the |
| // nearest cache entry that contains given sample number. Starts the search |
| // from the starting_cache_index. |
| bool stsc_SlipCacheToSample(uint32 sample_number, int starting_cache_index); |
| |
| bool stss_Init(); |
| // step through table by one table entry, return false on error |
| bool stss_AdvanceStep(); |
| // search for nearest keyframe, update state to contain it |
| bool stss_FindNearestKeyframe(uint32 sample_number); |
| |
| bool stts_Init(); |
| bool stts_AdvanceToSample(uint32 sample_number); |
| bool stts_SlipCacheToSample(uint32 sample_number, int starting_cache_index); |
| bool stts_AdvanceToTime(uint64 timestamp); |
| bool stts_SlipCacheToTime(uint64 timestamp, int starting_cache_index); |
| // step through the stts table by one table entry, return false on error |
| bool stts_IntegrateStep(); |
| |
| bool stsz_Init(); |
| |
| // TableCache manages the caching of each atom table in a separate instance. |
| // As each atom has a different per-entry byte size, and may want different |
| // caching behavior based on consumption rate of entries and the overall size |
| // of the table, it allows each atom to use its own policy for caching. |
| // To keep things relatively simple it always keep a table of size of the |
| // minimum of cache_size_entries or entry_count in memory, and that cached |
| // table is always aligned with the full table in cache_size_entries, so |
| // that the position in the cache is trivially calculated from |
| // entry_number % cache_size_entries, and the cache index is similarly |
| // calculated from entry_number / cache_size_entries. |
| class TableCache : public base::RefCountedThreadSafe<TableCache> { |
| public: |
| TableCache(uint64 table_offset, // byte offset of start of table in file |
| uint32 entry_count, // number of entries in table |
| uint32 entry_size, // size in bytes of each entry in table |
| uint32 cache_size_entries, // number of entries to cache in mem |
| scoped_refptr<DataSourceReader> reader); // reader to use |
| |
| // The following Read* functions all read values in big endian. |
| bool ReadU32Entry(uint32 entry_number, uint32* entry); |
| bool ReadU32PairEntry(uint32 entry_number, uint32* first, uint32* second); |
| bool ReadU32EntryIntoU64(uint32 entry_number, uint64* entry); |
| bool ReadU64Entry(uint32 entry_number, uint64* entry); |
| |
| uint8* GetBytesAtEntry(uint32 entry_number); |
| |
| // how many entries total in the table? |
| inline uint32 GetEntryCount() const { return entry_count_; } |
| // how many entries are we caching in memory at once? |
| inline uint32 GetCacheSizeEntries() const { return cache_size_entries_; } |
| |
| private: |
| friend class base::RefCountedThreadSafe<TableCache>; |
| uint32 entry_size_; // size of entry in bytes |
| uint32 entry_count_; // size of table in entries |
| uint32 cache_size_entries_; // max number of entries to fit in memory |
| uint64 table_offset_; // offset of table in stream |
| scoped_refptr<DataSourceReader> reader_; // means to read more table |
| |
| // current cache state |
| std::vector<uint8> cache_; // the cached part of the table |
| uint32 cache_first_entry_number_; // first table entry number in cache |
| uint32 cache_entry_count_; // number of valid entries in cache |
| }; |
| |
| scoped_refptr<DataSourceReader> reader_; |
| |
| // current integration state for GetOffset(), we save the sum of sample sizes |
| // within the current chunk. |
| uint32 current_chunk_sample_; // sample number last included in summation |
| uint32 next_chunk_sample_; // first sample number of next chunk |
| uint64 current_chunk_offset_; // file byte offset of current_chunk_sample_ |
| |
| // Can be set by a stsz entry count but an stsz may provide a default size, |
| // in which case this number may not be known until iteration through |
| // the ctts, or stts has completed. In the event that one of those tables |
| // ends at a lower number than the others this number will be amended |
| // to return the lower number. |
| uint32 highest_valid_sample_number_; |
| |
| // ==== c064 - per-chunk list of file offsets (64-bit) |
| scoped_refptr<TableCache> co64_; |
| |
| // ==== ctts - run-length sample number to composition time offset |
| scoped_refptr<TableCache> ctts_; |
| uint32 ctts_first_sample_; |
| uint32 ctts_sample_offset_; |
| uint32 ctts_next_first_sample_; |
| uint32 ctts_table_index_; |
| std::vector<uint32> ctts_samples_; |
| |
| // ==== stco - per-chunk list of chunk file offsets (32-bit) |
| scoped_refptr<TableCache> stco_; |
| |
| // ==== stsc - chunk-number to samples-per-chunk |
| scoped_refptr<TableCache> stsc_; |
| uint32 stsc_first_chunk_; // first chunk of the current sample size range |
| uint32 stsc_first_chunk_sample_; // sum of samples of all prev chunk ranges |
| uint32 stsc_samples_per_chunk_; // current samples-per-chunk in this range |
| uint32 stsc_next_first_chunk_; // the chunk number the next region begins in |
| uint32 stsc_next_first_chunk_sample_; // sample number next region begins in |
| uint32 stsc_table_index_; // the index in the table of the current range |
| std::vector<uint32> stsc_sample_sums_; // saved sums of cache segments |
| |
| // ==== stss - list of keyframe sample numbers |
| scoped_refptr<TableCache> stss_; |
| uint32 stss_last_keyframe_; |
| uint32 stss_next_keyframe_; |
| uint32 stss_table_index_; // index of stss_next_keyframe_ in table |
| std::vector<uint32> stss_keyframes_; |
| |
| // ==== stts - run-length sample number to time duration |
| scoped_refptr<TableCache> stts_; |
| uint32 stts_first_sample_; // first sample of the current duration range |
| uint64 stts_first_sample_time_; // sum of all durations of previous ranges |
| uint32 stts_sample_duration_; // current duration of samples in this range |
| uint32 stts_next_first_sample_; // first sample number of next range |
| uint64 stts_next_first_sample_time_; // first timestamp of next range |
| uint32 stts_table_index_; // index in the table of the next entry |
| std::vector<uint32> stts_samples_; |
| std::vector<uint64> stts_timestamps_; |
| |
| // ==== stsz - per-sample list of sample sizes |
| scoped_refptr<TableCache> stsz_; |
| uint32 stsz_default_size_; |
| }; |
| |
| } // namespace media |
| } // namespace cobalt |
| |
| #endif // COBALT_MEDIA_PROGRESSIVE_MP4_MAP_H_ |