#include "mov-internal.h" #include #include #include // 8.7.5 Chunk Offset Box (p58) /* aligned(8) class ChunkOffsetBox extends FullBox('stco', version = 0, 0) { unsigned int(32) entry_count; for (i=1; i <= entry_count; i++) { unsigned int(32) chunk_offset; } } aligned(8) class ChunkLargeOffsetBox extends FullBox('co64', version = 0, 0) { unsigned int(32) entry_count; for (i=1; i <= entry_count; i++) { unsigned int(64) chunk_offset; } } */ int mov_read_stco(struct mov_t* mov, const struct mov_box_t* box) { uint32_t i, entry_count; struct mov_stbl_t* stbl = &mov->track->stbl; mov_buffer_r8(&mov->io); /* version */ mov_buffer_r24(&mov->io); /* flags */ entry_count = mov_buffer_r32(&mov->io); assert(0 == stbl->stco_count && NULL == stbl->stco); if (stbl->stco_count < entry_count) { void* p = realloc(stbl->stco, sizeof(stbl->stco[0]) * entry_count); if (NULL == p) return -ENOMEM; stbl->stco = p; } stbl->stco_count = entry_count; if (MOV_TAG('s', 't', 'c', 'o') == box->type) { for (i = 0; i < entry_count; i++) stbl->stco[i] = mov_buffer_r32(&mov->io); // chunk_offset } else if (MOV_TAG('c', 'o', '6', '4') == box->type) { for (i = 0; i < entry_count; i++) stbl->stco[i] = mov_buffer_r64(&mov->io); // chunk_offset } else { i = 0; assert(0); } stbl->stco_count = i; return mov_buffer_error(&mov->io); } size_t mov_write_stco(const struct mov_t* mov, uint32_t count) { int co64; uint32_t size, i; const struct mov_sample_t* sample; const struct mov_track_t* track = mov->track; sample = track->sample_count > 0 ? &track->samples[track->sample_count - 1] : NULL; co64 = (sample && sample->offset + track->offset > UINT32_MAX) ? 1 : 0; size = 12/* full box */ + 4/* entry count */ + count * (co64 ? 8 : 4); mov_buffer_w32(&mov->io, size); /* size */ mov_buffer_write(&mov->io, co64 ? "co64" : "stco", 4); mov_buffer_w32(&mov->io, 0); /* version & flags */ mov_buffer_w32(&mov->io, count); /* entry count */ for (i = 0; i < track->sample_count; i++) { sample = track->samples + i; if(0 == sample->first_chunk) continue; if(0 == co64) mov_buffer_w32(&mov->io, (uint32_t)(sample->offset + track->offset)); else mov_buffer_w64(&mov->io, sample->offset + track->offset); } return size; } size_t mov_stco_size(const struct mov_track_t* track, uint64_t offset) { size_t i, j; uint64_t co64; const struct mov_sample_t* sample; if (track->sample_count < 1) return 0; sample = &track->samples[track->sample_count - 1]; co64 = sample->offset + track->offset; if (co64 > UINT32_MAX || co64 + offset <= UINT32_MAX) return 0; for (i = 0, j = 0; i < track->sample_count; i++) { sample = track->samples + i; if (0 != sample->first_chunk) j++; } return j * 4; } uint32_t mov_build_stco(struct mov_track_t* track) { size_t i; size_t bytes = 0; uint32_t count = 0; struct mov_sample_t* sample = NULL; assert(track->stsd.entry_count > 0); for (i = 0; i < track->sample_count; i++) { if (NULL != sample && sample->offset + bytes == track->samples[i].offset && sample->sample_description_index == track->samples[i].sample_description_index) { track->samples[i].first_chunk = 0; // mark invalid value bytes += track->samples[i].bytes; ++sample->samples_per_chunk; } else { sample = &track->samples[i]; sample->first_chunk = ++count; // chunk start from 1 sample->samples_per_chunk = 1; bytes = sample->bytes; } } return count; } void mov_apply_stco(struct mov_track_t* track) { uint32_t i, j, k; uint64_t n, chunk_offset; struct mov_stbl_t* stbl = &track->stbl; // sample offset assert(stbl->stsc_count > 0 && stbl->stco_count > 0); stbl->stsc[stbl->stsc_count].first_chunk = stbl->stco_count + 1; // fill stco count for (i = 0, n = 0; i < stbl->stsc_count; i++) { assert(stbl->stsc[i].first_chunk <= stbl->stco_count); for (j = stbl->stsc[i].first_chunk; j < stbl->stsc[i + 1].first_chunk; j++) { chunk_offset = stbl->stco[j - 1]; // chunk start from 1 for (k = 0; k < stbl->stsc[i].samples_per_chunk; k++, n++) { track->samples[n].sample_description_index = stbl->stsc[i].sample_description_index; track->samples[n].offset = chunk_offset; track->samples[n].data = NULL; chunk_offset += track->samples[n].bytes; assert(track->samples[n].bytes > 0); assert(0 == n || track->samples[n - 1].offset + track->samples[n - 1].bytes <= track->samples[n].offset); } } } assert(n == track->sample_count); }