blob: a667bb20fde16b105a380264c46616f6cda82548 [file] [log] [blame]
// Copyright 2012 Google Inc. 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.
#include "cobalt/media/filters/shell_mp4_map.h"
#include <algorithm>
#include "base/stringprintf.h"
#include "cobalt/media/base/endian_util.h"
#include "cobalt/media/filters/shell_mp4_parser.h"
namespace cobalt {
namespace media {
// ==== TableCache =============================================================
ShellMP4Map::TableCache::TableCache(uint64 table_offset, uint32 entry_count,
uint32 entry_size,
uint32 cache_size_entries,
scoped_refptr<ShellDataSourceReader> reader)
: entry_size_(entry_size),
entry_count_(entry_count),
cache_size_entries_(cache_size_entries),
table_offset_(table_offset),
reader_(reader),
cache_first_entry_number_(-1),
cache_entry_count_(0) {}
uint8* ShellMP4Map::TableCache::GetBytesAtEntry(uint32 entry_number) {
// don't fetch the unfetchable
if (entry_number >= entry_count_) {
return NULL;
}
// this query within valid range for the current cache table?
if (entry_number < cache_first_entry_number_ ||
entry_number >= cache_first_entry_number_ + cache_entry_count_) {
// Calculate first entry in table keeping cache size alignment in table.
// Always cache one more entry as in stss we need to use the first entry
// of the next cache slot as an upper bound.
cache_entry_count_ = cache_size_entries_ + 1;
cache_first_entry_number_ =
(entry_number / cache_size_entries_) * cache_size_entries_;
// see if we have exceeded our table bounds
if (cache_first_entry_number_ + cache_entry_count_ > entry_count_) {
cache_entry_count_ = entry_count_ - cache_first_entry_number_;
}
// drop old data to allow ShellBufferFactory to defrag
cache_.clear();
DCHECK_GE(cache_entry_count_, 0);
int bytes_to_read = cache_entry_count_ * entry_size_;
cache_.resize(bytes_to_read);
uint64 file_offset =
table_offset_ + (cache_first_entry_number_ * entry_size_);
int bytes_read =
reader_->BlockingRead(file_offset, bytes_to_read, &cache_[0]);
if (bytes_read < bytes_to_read) {
cache_entry_count_ = 0;
return NULL;
}
}
// cache is assumed to be valid and to contain the entry from here on
DCHECK_GE(entry_number, cache_first_entry_number_);
DCHECK_LT(entry_number, cache_first_entry_number_ + cache_entry_count_);
uint32 cache_offset = entry_number - cache_first_entry_number_;
return &cache_[0] + (cache_offset * entry_size_);
}
bool ShellMP4Map::TableCache::ReadU32Entry(uint32 entry_number, uint32* entry) {
if (uint8* data = GetBytesAtEntry(entry_number)) {
*entry = endian_util::load_uint32_big_endian(data);
return true;
}
return false;
}
bool ShellMP4Map::TableCache::ReadU32PairEntry(uint32 entry_number,
uint32* first, uint32* second) {
if (uint8* data = GetBytesAtEntry(entry_number)) {
if (first) *first = endian_util::load_uint32_big_endian(data);
if (second) *second = endian_util::load_uint32_big_endian(data + 4);
return true;
}
return false;
}
bool ShellMP4Map::TableCache::ReadU32EntryIntoU64(uint32 entry_number,
uint64* entry) {
if (uint8* data = GetBytesAtEntry(entry_number)) {
*entry = endian_util::load_uint32_big_endian(data);
return true;
}
return false;
}
bool ShellMP4Map::TableCache::ReadU64Entry(uint32 entry_number, uint64* entry) {
if (uint8* data = GetBytesAtEntry(entry_number)) {
*entry = endian_util::load_uint64_big_endian(data);
return true;
}
return false;
}
// ==== ShellMP4Map ============================================================
// atom | name | size | description, (*) means optional table
// -----+-----------------------+------+----------------------------------------
// co64 | chunk offset (64-bit) | 8 | per-chunk list of chunk file offsets
// ctts | composition offset | 8 | (*) run-length sample number to cts
// stco | chunk offset (32-bit) | 4 | per-chunk list of chunk file offsets
// stsc | sample-to-chunk | 12 | chunk number to samples per chunk
// stss | sync sample | 4 | (*) list of keyframe sample numbers
// stts | time-to-sample | 8 | run-length sample number to duration
// stsz | sample size | 4 | per-sample list of sample sizes
ShellMP4Map::ShellMP4Map(scoped_refptr<ShellDataSourceReader> reader)
: reader_(reader),
current_chunk_sample_(0),
next_chunk_sample_(0),
current_chunk_offset_(0),
highest_valid_sample_number_(UINT32_MAX),
ctts_first_sample_(0),
ctts_sample_offset_(0),
ctts_next_first_sample_(0),
ctts_table_index_(0),
stsc_first_chunk_(0),
stsc_first_chunk_sample_(0),
stsc_samples_per_chunk_(0),
stsc_next_first_chunk_(0),
stsc_next_first_chunk_sample_(0),
stsc_table_index_(0),
stss_last_keyframe_(0),
stss_next_keyframe_(0),
stss_table_index_(0),
stts_first_sample_(0),
stts_first_sample_time_(0),
stts_sample_duration_(0),
stts_next_first_sample_(0),
stts_next_first_sample_time_(0),
stts_table_index_(0),
stsz_default_size_(0) {}
bool ShellMP4Map::IsComplete() {
// all required table pointers must be valid for map to function
return (co64_ || stco_) && stsc_ && stts_ && (stsz_ || stsz_default_size_);
}
// The sample size is a lookup in the stsz table, which is indexed per sample
// number.
bool ShellMP4Map::GetSize(uint32 sample_number, uint32* size_out) {
DCHECK(size_out);
DCHECK(stsz_ || stsz_default_size_);
if (sample_number > highest_valid_sample_number_) {
return false;
}
if (stsz_default_size_) {
*size_out = stsz_default_size_;
return true;
}
return stsz_->ReadU32Entry(sample_number, size_out);
}
// We first must integrate the stsc table to find the chunk number that the
// sample resides in, and the first sample number in that chunk. We look up that
// chunk offset from the stco or co64, which are indexed by chunk number. We
// then use the stsz to sum samples to the byte offset with that chunk. The sum
// of the chunk offset and the byte offset within the chunk is the offset of
// the sample.
bool ShellMP4Map::GetOffset(uint32 sample_number, uint64* offset_out) {
DCHECK(offset_out);
DCHECK(stsc_);
DCHECK(stco_ || co64_);
DCHECK(stsz_ || stsz_default_size_);
if (sample_number > highest_valid_sample_number_) {
return false;
}
// check for sequential access of sample numbers within the same chunk
if (sample_number < current_chunk_sample_ ||
sample_number >= next_chunk_sample_) {
// integrate through stsc until we find the chunk range containing sample
if (!stsc_AdvanceToSample(sample_number)) {
return false;
}
// make sure stsc advance did its job correctly
DCHECK_GE(sample_number, stsc_first_chunk_sample_);
// calculate chunk number based on chunk sample size for this range
uint32 sample_offset = sample_number - stsc_first_chunk_sample_;
uint32 chunk_range_offset = sample_offset / stsc_samples_per_chunk_;
uint32 chunk_number = stsc_first_chunk_ + chunk_range_offset;
// should be within the range of chunks with this sample size
DCHECK_LT(chunk_number, stsc_next_first_chunk_);
// update first sample number contained within this chunk
current_chunk_sample_ = stsc_first_chunk_sample_ +
(chunk_range_offset * stsc_samples_per_chunk_);
// update first sample number of next chunk
next_chunk_sample_ = current_chunk_sample_ + stsc_samples_per_chunk_;
// find offset of this chunk within the file from co64/stco
if (co64_) {
if (!co64_->ReadU64Entry(chunk_number, &current_chunk_offset_))
return false;
} else if (!stco_->ReadU32EntryIntoU64(chunk_number,
&current_chunk_offset_)) {
return false;
}
}
// at this point we should have sample_number within the range of our chunk
// offset summation saved state
DCHECK_LE(current_chunk_sample_, sample_number);
DCHECK_LT(sample_number, next_chunk_sample_);
if (stsz_default_size_ > 0) {
current_chunk_offset_ +=
(sample_number - current_chunk_sample_) * stsz_default_size_;
current_chunk_sample_ = sample_number;
} else {
// sum sample sizes within chunk to get to byte offset of sample
while (current_chunk_sample_ < sample_number) {
uint32 sample_size = 0;
if (!GetSize(current_chunk_sample_, &sample_size)) {
return false;
}
current_chunk_offset_ += sample_size;
current_chunk_sample_++;
}
}
*offset_out = current_chunk_offset_;
return true;
}
// Given a current sample number we integrate through the stts to find the
// duration of the current sample, and at the same time integrate through the
// durations to find the dts of that sample number. We then integrate sample
// numbers through the ctts to find the composition time offset, which we add to
// the dts to return the pts.
bool ShellMP4Map::GetTimestamp(uint32 sample_number, uint64* timestamp_out) {
DCHECK(timestamp_out);
if (sample_number > highest_valid_sample_number_) {
return false;
}
if (!stts_AdvanceToSample(sample_number)) {
return false;
}
DCHECK_LT(sample_number, stts_next_first_sample_);
DCHECK_GE(sample_number, stts_first_sample_);
uint64 dts = stts_first_sample_time_ +
(sample_number - stts_first_sample_) * stts_sample_duration_;
if (ctts_) {
if (!ctts_AdvanceToSample(sample_number)) {
return false;
}
DCHECK_LT(sample_number, ctts_next_first_sample_);
DCHECK_GE(sample_number, ctts_first_sample_);
}
*timestamp_out = dts + ctts_sample_offset_;
return true;
}
// Sum through the stts to find the duration of the given sample_number.
bool ShellMP4Map::GetDuration(uint32 sample_number, uint32* duration_out) {
DCHECK(duration_out);
if (sample_number > highest_valid_sample_number_) {
return false;
}
if (!stts_AdvanceToSample(sample_number)) {
return false;
}
DCHECK_LT(sample_number, stts_next_first_sample_);
DCHECK_GE(sample_number, stts_first_sample_);
*duration_out = stts_sample_duration_;
return true;
}
bool ShellMP4Map::GetIsKeyframe(uint32 sample_number, bool* is_keyframe_out) {
DCHECK(is_keyframe_out);
if (sample_number > highest_valid_sample_number_) {
return false;
}
// no stts means every frame is a keyframe
if (!stss_) {
*is_keyframe_out = true;
return true;
}
// check for keyframe match on either range value
if (sample_number == stss_next_keyframe_) {
*is_keyframe_out = true;
return stss_AdvanceStep();
} else if (sample_number == stss_last_keyframe_) {
*is_keyframe_out = true;
return true;
}
// this could be for a much earlier sample number, check if we are within
// current range of sample numbers
if (sample_number < stss_last_keyframe_ ||
sample_number > stss_next_keyframe_) {
// search for containing entry
if (!stss_FindNearestKeyframe(sample_number)) {
return false;
}
}
// sample number must be in range of keyframe states
DCHECK_GE(sample_number, stss_last_keyframe_);
DCHECK_LT(sample_number, stss_next_keyframe_);
// stss_FindNearestKeyframe returns exact matches to
// sample_number in the stss_last_keyframe_ variable, so
// we check that for equality
*is_keyframe_out = (sample_number == stss_last_keyframe_);
return true;
}
bool ShellMP4Map::IsEOS(uint32 sample_number) {
return (sample_number > highest_valid_sample_number_);
}
// First look up the sample number for the provided timestamp by integrating
// timestamps through the stts. Then do a binary search on the stss to find the
// keyframe nearest that sample number.
bool ShellMP4Map::GetKeyframe(uint64 timestamp, uint32* sample_out) {
DCHECK(sample_out);
// Advance stts to the provided timestamp range
if (!stts_AdvanceToTime(timestamp)) {
return false;
}
// ensure we got the correct sample duration range
DCHECK_LT(timestamp, stts_next_first_sample_time_);
DCHECK_GE(timestamp, stts_first_sample_time_);
// calculate sample number containing this timestamp
uint64 time_offset_within_range = timestamp - stts_first_sample_time_;
uint32 sample_number =
stts_first_sample_ + (time_offset_within_range / stts_sample_duration_);
// TODO: ctts?
// binary search on stts to find nearest keyframe beneath this sample number
if (stss_) {
if (!stss_FindNearestKeyframe(sample_number)) {
return false;
}
*sample_out = stss_last_keyframe_;
} else {
// an absent stts means every frame is a key frame, we can provide sample
// directly.
*sample_out = sample_number;
}
return true;
}
// Set up map state and load first part of table, or entire table if it is small
// enough, for each of the supporated atoms.
bool ShellMP4Map::SetAtom(uint32 four_cc, uint64 offset, uint64 size,
uint32 cache_size_entries, const uint8* atom) {
// All map atoms are variable-length tables starting with 4 bytes of
// version/flag info followed by a uint32 indicating the number of items in
// table. The stsz atom bucks tradition by putting an optional default value
// at index 4.
uint32 count = 0;
uint64 table_offset = offset + 8;
if (four_cc == kAtomType_stsz) {
if (size < 12) {
return false;
}
stsz_default_size_ = endian_util::load_uint32_big_endian(atom + 4);
count = endian_util::load_uint32_big_endian(atom + 8);
highest_valid_sample_number_ =
std::min(count - 1, highest_valid_sample_number_);
// if a non-zero default size is provided don't bother loading the table
if (stsz_default_size_) {
stsz_ = NULL;
return true;
}
table_offset += 4;
} else {
if (size < 8) {
return false;
}
count = endian_util::load_uint32_big_endian(atom + 4);
}
// if cache_size_entries is 0 we are to cache the entire table
if (cache_size_entries == 0) {
cache_size_entries = count;
}
bool atom_init = false;
// initialize the appropriate table cache dependent on table type
switch (four_cc) {
case kAtomType_co64:
co64_ = new TableCache(table_offset, count, kEntrySize_co64,
cache_size_entries, reader_);
if (co64_) atom_init = co64_Init();
break;
case kAtomType_ctts:
ctts_ = new TableCache(table_offset, count, kEntrySize_ctts,
cache_size_entries, reader_);
if (ctts_) atom_init = ctts_Init();
break;
case kAtomType_stco:
stco_ = new TableCache(table_offset, count, kEntrySize_stco,
cache_size_entries, reader_);
if (stco_) atom_init = stco_Init();
break;
case kAtomType_stsc:
stsc_ = new TableCache(table_offset, count, kEntrySize_stsc,
cache_size_entries, reader_);
if (stsc_) atom_init = stsc_Init();
break;
case kAtomType_stss:
stss_ = new TableCache(table_offset, count, kEntrySize_stss,
cache_size_entries, reader_);
if (stss_) atom_init = stss_Init();
break;
case kAtomType_stts:
stts_ = new TableCache(table_offset, count, kEntrySize_stts,
cache_size_entries, reader_);
if (stts_) atom_init = stts_Init();
break;
case kAtomType_stsz:
stsz_ = new TableCache(table_offset, count, kEntrySize_stsz,
cache_size_entries, reader_);
if (stsz_) atom_init = stsz_Init();
break;
default:
NOTREACHED() << "unknown atom type provided to mp4 map";
break;
}
return atom_init;
}
bool ShellMP4Map::co64_Init() {
DCHECK(co64_);
// load offset of first chunk into current_chunk_offset_
if (co64_->GetEntryCount() > 0) {
// can drop any stco table already allocated
stco_ = NULL;
// load initial value of current_chunk_offset_ for 0th chunk
return co64_->ReadU64Entry(0, &current_chunk_offset_);
}
co64_ = NULL;
return true;
}
// The ctts table has the following per-entry layout:
// uint32 sample count
// uint32 composition offset in ticks
//
bool ShellMP4Map::ctts_Init() {
DCHECK(ctts_);
// get cache segment vector to reserve table entries in advance
int cache_segments =
(ctts_->GetEntryCount() / ctts_->GetCacheSizeEntries()) + 1;
ctts_samples_.reserve(cache_segments);
if (ctts_->GetEntryCount() > 0) {
// save the start of the first table integration at 0
ctts_samples_.push_back(0);
ctts_table_index_ = 0;
ctts_first_sample_ = 0;
// load first entry in table, to start integration
return ctts_->ReadU32PairEntry(0, &ctts_next_first_sample_,
&ctts_sample_offset_);
}
// drop empty ctts_ table
ctts_ = NULL;
return true;
}
// To find the composition offset of a given sample number we must integrate
// through the ctts to find the range of samples containing sample_number. Note
// that the ctts is an optional table.
bool ShellMP4Map::ctts_AdvanceToSample(uint32 sample_number) {
// ctts table is optional, so treat not having one as non-fatal
if (!ctts_) {
return true;
}
// sample number could be before our saved first sample, meaning we've
// gone backward in sample numbers and will need to restart integration at
// the nearest saved sample count starting at a cache entry
if (sample_number < ctts_first_sample_) {
if (!ctts_SlipCacheToSample(sample_number, 0)) {
return false;
}
}
// sample_number could also be ahead of our current range, for example when
// seeking forward. See if we've calculated these values ahead of us before,
// and if we can slip forward to them
int next_cache_index = (ctts_table_index_ / ctts_->GetCacheSizeEntries()) + 1;
if ((next_cache_index < ctts_samples_.size()) &&
(sample_number >= ctts_samples_[next_cache_index])) {
if (!ctts_SlipCacheToSample(sample_number, next_cache_index)) {
return false;
}
}
// perform integration until sample number is within correct ctts range
while (ctts_next_first_sample_ <= sample_number) {
// next first sample is now our the first sample
ctts_first_sample_ = ctts_next_first_sample_;
// advance to next entry in table
ctts_table_index_++;
// If this would be a new cache entry, keep a record of integration up
// to this point so we don't have to start from 0 on seeking back
if (!(ctts_table_index_ % ctts_->GetCacheSizeEntries())) {
int cache_index = ctts_table_index_ / ctts_->GetCacheSizeEntries();
// check that this is our first time with these data
if (cache_index == ctts_samples_.size()) {
ctts_samples_.push_back(ctts_first_sample_);
}
// our integration at this point should always match any stored record
DCHECK_EQ(ctts_first_sample_, ctts_samples_[cache_index]);
}
if (ctts_table_index_ < ctts_->GetEntryCount()) {
// load the sample count to determine next first sample
uint32 sample_count;
if (!ctts_->ReadU32PairEntry(ctts_table_index_, &sample_count,
&ctts_sample_offset_))
return false;
ctts_next_first_sample_ = ctts_first_sample_ + sample_count;
} else {
// This means that the last entry in the table specified a sample range
// that this sample number has exceeded, and so the ctts of this sample
// number is undefined. While not a fatal error it's kind of a weird
// state, we set the offset back to zero and extend the next_first_sample
// to infinity
DLOG(WARNING) << base::StringPrintf(
"out of range sample number %d in ctts, last valid sample number: %d",
sample_number, ctts_next_first_sample_);
ctts_sample_offset_ = 0;
ctts_next_first_sample_ = UINT32_MAX;
break;
}
}
return true;
}
bool ShellMP4Map::ctts_SlipCacheToSample(uint32 sample_number,
int starting_cache_index) {
DCHECK_LT(starting_cache_index, ctts_samples_.size());
int cache_index = starting_cache_index;
for (; cache_index + 1 < ctts_samples_.size(); cache_index++) {
if (sample_number < ctts_samples_[cache_index + 1]) {
break;
}
}
ctts_first_sample_ = ctts_samples_[cache_index];
ctts_table_index_ = cache_index * ctts_->GetCacheSizeEntries();
// read sample count and duration to set next values
uint32 sample_count;
if (!ctts_->ReadU32PairEntry(ctts_table_index_, &sample_count,
&ctts_sample_offset_))
return false;
ctts_next_first_sample_ = ctts_first_sample_ + sample_count;
return true;
}
bool ShellMP4Map::stco_Init() {
DCHECK(stco_);
// load offset of first chunk into current_chunk_offset_
if (stco_->GetEntryCount() > 0) {
co64_ = NULL;
return stco_->ReadU32EntryIntoU64(0, &current_chunk_offset_);
}
stco_ = NULL;
return true;
}
// The stsc table has the following per-entry layout:
// uint32 first chunk number with this sample count
// uint32 samples-per-chunk
// uint32 sample description id (unused)
bool ShellMP4Map::stsc_Init() {
DCHECK(stsc_);
// set up vector to correct final size
int cache_segments =
(stsc_->GetEntryCount() / stsc_->GetCacheSizeEntries()) + 1;
stsc_sample_sums_.reserve(cache_segments);
// there must always be at least 1 entry in a valid stsc table
if (stsc_->GetEntryCount() > 0) {
stsc_first_chunk_ = 0;
stsc_first_chunk_sample_ = 0;
// first cached entry is always 0
stsc_sample_sums_.push_back(0);
if (!stsc_->ReadU32PairEntry(0, NULL, &stsc_samples_per_chunk_)) {
stsc_ = NULL;
return false;
}
// look up next first chunk at next index in table
if (stsc_->GetEntryCount() > 1) {
if (!stsc_->ReadU32PairEntry(1, &stsc_next_first_chunk_, NULL)) {
stsc_ = NULL;
return false;
}
--stsc_next_first_chunk_;
stsc_next_first_chunk_sample_ =
stsc_next_first_chunk_ * stsc_samples_per_chunk_;
} else {
// every chunk in the file has the sample sample count, set next first
// chunk to highest valid chunk number.
stsc_next_first_chunk_ = UINT32_MAX;
stsc_next_first_chunk_sample_ = UINT32_MAX;
}
stsc_table_index_ = 0;
// since we known the size of the first chunk we can set next_chunk_sample_
next_chunk_sample_ = stsc_samples_per_chunk_;
} else {
stsc_ = NULL;
}
return true;
}
// To find the chunk number of an abritrary sample we have to sum the
// samples-per-chunk value multiplied by the number of chunks with that sample
// count until the sum exceeds the sample number, then calculate the chunk
// number from that range of per-sample chunk sizes. Since this map is meant
// to be consumed incrementally and with minimal memory consumption we calculate
// this integration step only when needed, and save results for each cached
// piece of the table, to avoid having to recalculate needed data.
bool ShellMP4Map::stsc_AdvanceToSample(uint32 sample_number) {
DCHECK(stsc_);
// sample_number could be before first chunk, meaning that we are seeking
// backwards and have left the current chunk. Find the closest part of the
// cached table and integrate forward from there.
if (sample_number < stsc_first_chunk_sample_) {
if (!stsc_SlipCacheToSample(sample_number, 0)) {
return false;
}
}
// sample_number could also be well head of our current piece of the
// cache, so see if we can re-use any previously calculated summations to
// skip to the nearest cache entry
int next_cache_index = (stsc_table_index_ / stsc_->GetCacheSizeEntries()) + 1;
if ((next_cache_index < stsc_sample_sums_.size()) &&
(sample_number >= stsc_sample_sums_[next_cache_index])) {
if (!stsc_SlipCacheToSample(sample_number, next_cache_index)) {
return false;
}
}
// Integrate through each table entry until we find sample_number in range
while (stsc_next_first_chunk_sample_ <= sample_number) {
// advance to next chunk sample range
stsc_first_chunk_sample_ = stsc_next_first_chunk_sample_;
// our next_first_chunk is now our first chunk
stsc_first_chunk_ = stsc_next_first_chunk_;
// advance to next entry in table
stsc_table_index_++;
// if we've advanced to a new segment of the cache, update the saved
// integration values
if (!(stsc_table_index_ % stsc_->GetCacheSizeEntries())) {
int cache_index = stsc_table_index_ / stsc_->GetCacheSizeEntries();
// check that this is our first time with these data
if (cache_index == stsc_sample_sums_.size()) {
stsc_sample_sums_.push_back(stsc_first_chunk_sample_);
}
// our integration at this point should always match any stored record
DCHECK_EQ(stsc_first_chunk_sample_, stsc_sample_sums_[cache_index]);
}
if (stsc_table_index_ < stsc_->GetEntryCount()) {
// look up our new sample rate
if (!stsc_->ReadU32PairEntry(stsc_table_index_, NULL,
&stsc_samples_per_chunk_)) {
return false;
}
// we need to look up next table entry to determine next first chunk
if (stsc_table_index_ + 1 < stsc_->GetEntryCount()) {
// look up next first chunk
if (!stsc_->ReadU32PairEntry(stsc_table_index_ + 1,
&stsc_next_first_chunk_, NULL)) {
return false;
}
--stsc_next_first_chunk_;
// carry sum of first_samples forward to next chunk range
stsc_next_first_chunk_sample_ +=
(stsc_next_first_chunk_ - stsc_first_chunk_) *
stsc_samples_per_chunk_;
} else {
// this is the normal place to encounter the end of the chunk table.
// set the next chunk to the highest valid chunk number
stsc_next_first_chunk_ = UINT32_MAX;
stsc_next_first_chunk_sample_ = UINT32_MAX;
}
} else {
// We should normally encounter the end of the chunk table on lookup
// of the next_first_chunk_ within the if clause associated with this
// else. Something has gone wrong.
NOTREACHED();
return false;
}
}
return true;
}
bool ShellMP4Map::stsc_SlipCacheToSample(uint32 sample_number,
int starting_cache_index) {
DCHECK_LT(starting_cache_index, stsc_sample_sums_.size());
// look through old sample sums for the first entry that exceeds sample
// sample_number, we want the entry right before that
int cache_index = starting_cache_index;
for (; cache_index + 1 < stsc_sample_sums_.size(); cache_index++) {
if (sample_number < stsc_sample_sums_[cache_index + 1]) {
break;
}
}
// jump to new spot in table
stsc_first_chunk_sample_ = stsc_sample_sums_[cache_index];
stsc_table_index_ = cache_index * stsc_->GetCacheSizeEntries();
if (!stsc_->ReadU32PairEntry(stsc_table_index_, &stsc_first_chunk_,
&stsc_samples_per_chunk_)) {
return false;
}
// load current and next values
--stsc_first_chunk_;
if (stsc_table_index_ + 1 < stsc_->GetEntryCount()) {
if (!stsc_->ReadU32PairEntry(stsc_table_index_ + 1, &stsc_next_first_chunk_,
NULL)) {
return false;
}
--stsc_next_first_chunk_;
stsc_next_first_chunk_sample_ =
stsc_first_chunk_sample_ +
((stsc_next_first_chunk_ - stsc_first_chunk_) *
stsc_samples_per_chunk_);
} else {
// We seem to have cached an entry 1 entry before end of the table, and
// are seeking to a region contained in that last entry in the table.
stsc_next_first_chunk_ = UINT32_MAX;
stsc_next_first_chunk_sample_ = UINT32_MAX;
}
return true;
}
// stss is a list of sample numbers that are keyframes.
bool ShellMP4Map::stss_Init() {
int cache_segments =
(stss_->GetEntryCount() / stss_->GetCacheSizeEntries()) + 1;
stss_keyframes_.reserve(cache_segments);
// empty stss means every frame is a keyframe, same as not
// providing one
if (stss_->GetEntryCount() > 0) {
// identify first keyframe from first entry in stss
if (!stss_->ReadU32Entry(0, &stss_last_keyframe_)) {
stss_ = NULL;
return false;
}
--stss_last_keyframe_;
stss_keyframes_.push_back(stss_last_keyframe_);
stss_next_keyframe_ = stss_last_keyframe_;
stss_table_index_ = 0;
} else {
stss_ = NULL;
}
return true;
}
// advance by one table entry through stss, updating cache if necessary
bool ShellMP4Map::stss_AdvanceStep() {
DCHECK(stss_);
stss_last_keyframe_ = stss_next_keyframe_;
stss_table_index_++;
if (stss_table_index_ < stss_->GetEntryCount()) {
if (!stss_->ReadU32Entry(stss_table_index_, &stss_next_keyframe_)) {
return false;
}
--stss_next_keyframe_;
if (!(stss_table_index_ % stss_->GetCacheSizeEntries())) {
int cache_index = stss_table_index_ / stss_->GetCacheSizeEntries();
// only add if this is the first time we've encountered this number
if (cache_index == stss_keyframes_.size()) {
stss_keyframes_.push_back(stss_next_keyframe_);
}
DCHECK_EQ(stss_next_keyframe_, stss_keyframes_[cache_index]);
}
} else {
stss_next_keyframe_ = UINT32_MAX;
}
return true;
}
bool ShellMP4Map::stss_FindNearestKeyframe(uint32 sample_number) {
DCHECK(stss_);
// it is assumed that there's at least one cache entry created by
// stss_Init();
DCHECK_GT(stss_keyframes_.size(), 0);
int cache_entry_number = stss_keyframes_.size() - 1;
int total_cache_entries =
(stss_->GetEntryCount() + stss_->GetCacheSizeEntries() - 1) /
stss_->GetCacheSizeEntries();
// if there's more than one cache entry we can search the cached
// entries for the entry containing our keyframe, otherwise we skip
// directly to the binary search of the single cached entry
if (total_cache_entries > 1) {
// if the sample number resides within the range of cached entries
// we search those to find right table cache entry to load
if (sample_number < stss_keyframes_[cache_entry_number]) {
int lower_bound = 0;
int upper_bound = stss_keyframes_.size();
// binary search to find range
while (lower_bound <= upper_bound) {
cache_entry_number = lower_bound + ((upper_bound - lower_bound) / 2);
if (sample_number < stss_keyframes_[cache_entry_number]) {
upper_bound = cache_entry_number - 1;
} else { // sample_number >= stss_keyframes_[cache_entry_number]
// if we are at end of list or next cache entry is higher than sample
// number we consider it a match
if (cache_entry_number == stss_keyframes_.size() - 1 ||
sample_number < stss_keyframes_[cache_entry_number + 1]) {
break;
}
lower_bound = cache_entry_number + 1;
}
}
}
// We've gotten as close as we can using the cached values and must handle
// two cases. (a) is that we know that sample_number is contained in the
// cache_entry_number, because we know that:
// stts_keyframes_[cache_entry_number] <= sample_number <
// stts_keyframes_[cache_entry_number + 1]
// (b) is that we only know:
// stts_keyframes_[stts_keyframes_.size() - 1] <= sample_number
// because we have not cached an upper bound to sample_number.
// First step is to make (b) in to (a) by advancing through cache entries
// until last table entry in cache > sample_number or until we arrive
// at the cache entry in the table.
while ((cache_entry_number == stss_keyframes_.size() - 1) &&
cache_entry_number < total_cache_entries - 1) {
// Use the first key frame in next cache as upper bound.
int next_cached_entry_number =
(cache_entry_number + 1) * stss_->GetCacheSizeEntries();
uint32 next_cached_keyframe;
if (!stss_->ReadU32Entry(next_cached_entry_number,
&next_cached_keyframe)) {
return false;
}
--next_cached_keyframe;
// if this keyframe is higher than our sample number we're in the right
// table, stop
if (sample_number < next_cached_keyframe) {
break;
}
// ok, we need to look in to the next cache entry, advance
cache_entry_number++;
int first_table_entry_number =
cache_entry_number * stss_->GetCacheSizeEntries();
uint32 first_keyframe_in_cache_entry;
if (!stss_->ReadU32Entry(first_table_entry_number,
&first_keyframe_in_cache_entry)) {
return false;
}
--first_keyframe_in_cache_entry;
// save first entry in keyframe cache
stss_keyframes_.push_back(first_keyframe_in_cache_entry);
}
// make sure we have an upper bound
if (cache_entry_number != total_cache_entries - 1 &&
cache_entry_number == stss_keyframes_.size() - 1) {
int next_cached_entry_number =
((cache_entry_number + 1) * stss_->GetCacheSizeEntries());
uint32 next_cached_keyframe;
if (!stss_->ReadU32Entry(next_cached_entry_number,
&next_cached_keyframe)) {
return false;
}
--next_cached_keyframe;
stss_keyframes_.push_back(next_cached_keyframe);
}
// ok, now we assume we are in state (a), and that we're either
// at the end of the table or within the cache entry bounds for our
// sample number
DCHECK(stss_keyframes_[cache_entry_number] <= sample_number &&
(cache_entry_number == total_cache_entries - 1 ||
sample_number < stss_keyframes_[cache_entry_number + 1]));
}
// binary search within stss cache entry for keyframes bounding sample_number
int lower_bound = cache_entry_number * stss_->GetCacheSizeEntries();
int upper_bound = std::min(lower_bound + stss_->GetCacheSizeEntries(),
stss_->GetEntryCount());
while (lower_bound <= upper_bound) {
stss_table_index_ = lower_bound + ((upper_bound - lower_bound) / 2);
if (!stss_->ReadU32Entry(stss_table_index_, &stss_last_keyframe_)) {
return false;
}
--stss_last_keyframe_;
if (sample_number < stss_last_keyframe_) {
upper_bound = stss_table_index_ - 1;
} else { // sample_number >= last_keyframe
lower_bound = stss_table_index_ + 1;
// if this is the last entry in the table, we can stop here.
if (lower_bound == stss_->GetEntryCount()) {
stss_next_keyframe_ = UINT32_MAX;
break;
}
// load next entry in table, see if we actually found the upper bound
if (!stss_->ReadU32Entry(lower_bound, &stss_next_keyframe_)) {
return false;
}
--stss_next_keyframe_;
if (sample_number < stss_next_keyframe_) {
stss_table_index_ = lower_bound;
break;
}
}
}
return sample_number >= stss_last_keyframe_ &&
sample_number < stss_next_keyframe_;
}
// The stts table has the following per-entry layout:
// uint32 sample count - number of sequential samples with this duration
// uint32 sample duration - duration in ticks of this sample range
bool ShellMP4Map::stts_Init() {
int cache_segments =
(stts_->GetEntryCount() / stts_->GetCacheSizeEntries()) + 1;
stts_samples_.reserve(cache_segments);
stts_timestamps_.reserve(cache_segments);
// need at least one entry in valid stts
if (stts_->GetEntryCount() > 0) {
// integration starts at 0 for both cache entries
stts_samples_.push_back(0);
stts_timestamps_.push_back(0);
if (!stts_->ReadU32PairEntry(0, &stts_next_first_sample_,
&stts_sample_duration_)) {
stts_ = NULL;
return false;
}
stts_first_sample_ = 0;
stts_first_sample_time_ = 0;
stts_next_first_sample_time_ =
stts_next_first_sample_ * stts_sample_duration_;
stts_table_index_ = 0;
} else {
stts_ = NULL;
}
return true;
}
bool ShellMP4Map::stts_AdvanceToSample(uint32 sample_number) {
DCHECK(stts_);
// sample_number could be before our current sample range, in which case
// we skip to the nearest table entry before sample_number and integrate
// forward to the sample_number again.
if (sample_number < stts_first_sample_) {
if (!stts_SlipCacheToSample(sample_number, 0)) {
return false;
}
}
// sample number could also be well ahead of this cache segment, if we've
// previously calculated summations ahead let's skip to the correct one
int next_cache_index = (stts_table_index_ / stts_->GetCacheSizeEntries()) + 1;
if ((next_cache_index < stts_samples_.size()) &&
(sample_number >= stts_samples_[next_cache_index])) {
if (!stts_SlipCacheToSample(sample_number, next_cache_index)) {
return false;
}
}
// integrate through the stts until sample_number is within current range
while (stts_next_first_sample_ <= sample_number) {
if (!stts_IntegrateStep()) {
return false;
}
}
return true;
}
// Move our integration steps to a previously saved entry in the cache tables.
// Searches linearly through the vector of old cached values, so can accept a
// starting index to do the search from.
bool ShellMP4Map::stts_SlipCacheToSample(uint32 sample_number,
int starting_cache_index) {
DCHECK_LT(starting_cache_index, stts_samples_.size());
int cache_index = starting_cache_index;
for (; cache_index + 1 < stts_samples_.size(); cache_index++) {
if (sample_number < stts_samples_[cache_index + 1]) {
break;
}
}
stts_first_sample_ = stts_samples_[cache_index];
stts_first_sample_time_ = stts_timestamps_[cache_index];
stts_table_index_ = cache_index * stts_->GetCacheSizeEntries();
uint32 sample_count;
// read sample count and duration to set next values
if (!stts_->ReadU32PairEntry(stts_table_index_, &sample_count,
&stts_sample_duration_)) {
return false;
}
stts_next_first_sample_ = stts_first_sample_ + sample_count;
stts_next_first_sample_time_ =
stts_first_sample_time_ + (sample_count * stts_sample_duration_);
return true;
}
bool ShellMP4Map::stts_AdvanceToTime(uint64 timestamp) {
DCHECK(stts_);
if (timestamp < stts_first_sample_time_) {
if (!stts_SlipCacheToTime(timestamp, 0)) {
return false;
}
}
// sample number could also be well ahead of this cache segment, if we've
// previously calculated summations ahead let's skip to the correct one
int next_cache_index = (stts_table_index_ / stts_->GetCacheSizeEntries()) + 1;
if ((next_cache_index < stts_timestamps_.size()) &&
(timestamp >= stts_timestamps_[next_cache_index])) {
if (!stts_SlipCacheToTime(timestamp, next_cache_index)) {
return false;
}
}
// integrate through the stts until sample_number is within current range
while (stts_next_first_sample_time_ <= timestamp) {
if (!stts_IntegrateStep()) {
return false;
}
}
return true;
}
bool ShellMP4Map::stts_IntegrateStep() {
// advance time to next sample range
uint32 range_size = stts_next_first_sample_ - stts_first_sample_;
stts_first_sample_time_ += (range_size * stts_sample_duration_);
// advance sample counter to next range
stts_first_sample_ = stts_next_first_sample_;
// bump table counter to next entry
stts_table_index_++;
// see if we just crossed a cache boundary and should cache results
if (!(stts_table_index_ % stts_->GetCacheSizeEntries())) {
int cache_index = stts_table_index_ / stts_->GetCacheSizeEntries();
// check that this is our first time with these data
if (cache_index == stts_samples_.size()) {
// both tables should always grow together
DCHECK_EQ(stts_samples_.size(), stts_timestamps_.size());
stts_samples_.push_back(stts_first_sample_);
stts_timestamps_.push_back(stts_first_sample_time_);
}
// our integration at this point should always match any stored record
DCHECK_EQ(stts_first_sample_, stts_samples_[cache_index]);
DCHECK_EQ(stts_first_sample_time_, stts_timestamps_[cache_index]);
}
if (stts_table_index_ < stts_->GetEntryCount()) {
// load next entry data
uint32 sample_count;
if (!stts_->ReadU32PairEntry(stts_table_index_, &sample_count,
&stts_sample_duration_)) {
return false;
}
// calculate next sample number from range size
stts_next_first_sample_ = stts_first_sample_ + sample_count;
// and load duration of this sample range
stts_next_first_sample_time_ =
stts_first_sample_time_ + (sample_count * stts_sample_duration_);
} else {
// We've gone beyond the range defined by the last entry in the stts.
// this is an error.
highest_valid_sample_number_ =
std::min(highest_valid_sample_number_, stts_first_sample_ - 1);
return false;
}
return true;
}
bool ShellMP4Map::stts_SlipCacheToTime(uint64 timestamp,
int starting_cache_index) {
DCHECK_LT(starting_cache_index, stts_timestamps_.size());
int cache_index = starting_cache_index;
for (; cache_index + 1 < stts_timestamps_.size(); cache_index++) {
if (timestamp < stts_timestamps_[cache_index + 1]) {
break;
}
}
stts_first_sample_ = stts_samples_[cache_index];
stts_first_sample_time_ = stts_timestamps_[cache_index];
stts_table_index_ = cache_index * stts_->GetCacheSizeEntries();
// read sample count and duration to set next values
uint32 sample_count;
if (!stts_->ReadU32PairEntry(stts_table_index_, &sample_count,
&stts_sample_duration_)) {
return false;
}
stts_next_first_sample_ = stts_first_sample_ + sample_count;
stts_next_first_sample_time_ =
stts_first_sample_time_ + (sample_count * stts_sample_duration_);
return true;
}
bool ShellMP4Map::stsz_Init() { return stsz_->GetBytesAtEntry(0) != NULL; }
} // namespace media
} // namespace cobalt