mirror of
https://gitlab.com/mbunkus/mkvtoolnix.git
synced 2024-12-24 20:01:53 +00:00
230 lines
4.8 KiB
C++
230 lines
4.8 KiB
C++
// VirtualDub - Video processing and capture application
|
|
// Copyright (C) 1998-2001 Avery Lee
|
|
//
|
|
// This program is free software; you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation; either version 2 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program; if not, write to the Free Software
|
|
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
//
|
|
// Modified by Julien 'Cyrius' Coloos
|
|
// 20-09-2003
|
|
//
|
|
|
|
#include "AVIIndex.h"
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
class AVIIndexChainNode {
|
|
public:
|
|
enum { ENTS=2048 };
|
|
|
|
AVIIndexChainNode *next;
|
|
|
|
AVIIndexEntry2 ient[ENTS];
|
|
int num_ents;
|
|
|
|
AVIIndexChainNode() {
|
|
num_ents = 0;
|
|
next = NULL;
|
|
}
|
|
|
|
bool add(uint32_le ckid, sint64 pos, long size, bool is_keyframe) {
|
|
if (num_ents < ENTS) {
|
|
ient[num_ents].ckid = ckid;
|
|
ient[num_ents].pos = pos;
|
|
ient[num_ents].size = is_keyframe ? size : 0x80000000+size;
|
|
++num_ents;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void put(w32AVIINDEXENTRY *&avieptr) {
|
|
int i;
|
|
|
|
for(i=0; i<num_ents; i++) {
|
|
avieptr->ckid = ient[i].ckid;
|
|
avieptr->dwFlags = ient[i].size & 0x80000000 ? 0 : AVIIF_KEYFRAME;
|
|
avieptr->dwChunkOffset = ient[i].pos;
|
|
avieptr->dwChunkLength = ient[i].size & 0x7FFFFFFF;
|
|
|
|
++avieptr;
|
|
}
|
|
}
|
|
|
|
void put(AVIIndexEntry2 *&avie2ptr) {
|
|
int i;
|
|
|
|
for(i=0; i<num_ents; i++)
|
|
*avie2ptr++ = ient[i];
|
|
}
|
|
|
|
void put(AVIIndexEntry3 *&avie3ptr, sint64 offset) {
|
|
int i;
|
|
|
|
for(i=0; i<num_ents; i++) {
|
|
avie3ptr->dwSizeKeyframe = ient[i].size;
|
|
avie3ptr->dwOffset = (uint32)(ient[i].pos - offset);
|
|
|
|
++avie3ptr;
|
|
}
|
|
}
|
|
};
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
AVIIndexChain::AVIIndexChain() {
|
|
head = tail = NULL;
|
|
total_ents = 0;
|
|
}
|
|
|
|
void AVIIndexChain::delete_chain() {
|
|
AVIIndexChainNode *aicn = head,*aicn2;
|
|
|
|
while(aicn) {
|
|
aicn2 = aicn->next;
|
|
delete aicn;
|
|
aicn = aicn2;
|
|
}
|
|
|
|
head = tail = NULL;
|
|
}
|
|
|
|
AVIIndexChain::~AVIIndexChain() {
|
|
delete_chain();
|
|
}
|
|
|
|
bool AVIIndexChain::add(w32AVIINDEXENTRY *avie) {
|
|
if (!tail || !tail->add(avie->ckid, avie->dwChunkOffset, avie->dwChunkLength, !!(avie->dwFlags & AVIIF_KEYFRAME))) {
|
|
AVIIndexChainNode *aicn = new AVIIndexChainNode();
|
|
|
|
if (tail) tail->next = aicn; else head=aicn;
|
|
tail = aicn;
|
|
|
|
if (tail->add(avie->ckid, avie->dwChunkOffset, avie->dwChunkLength, !!(avie->dwFlags & AVIIF_KEYFRAME))) {
|
|
++total_ents;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
++total_ents;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool AVIIndexChain::add(AVIIndexEntry2 *avie2) {
|
|
return add(avie2->ckid, avie2->pos, avie2->size & 0x7FFFFFFF, !!(avie2->size & 0x80000000));
|
|
}
|
|
|
|
bool AVIIndexChain::add(uint32_le ckid, sint64 pos, long size, bool is_keyframe) {
|
|
if (!tail || !tail->add(ckid, pos, size, is_keyframe)) {
|
|
AVIIndexChainNode *aicn = new AVIIndexChainNode();
|
|
|
|
if (tail) tail->next = aicn; else head=aicn;
|
|
tail = aicn;
|
|
|
|
if (tail->add(ckid, pos, size, is_keyframe)) {
|
|
++total_ents;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
++total_ents;
|
|
|
|
return true;
|
|
}
|
|
|
|
void AVIIndexChain::put(w32AVIINDEXENTRY *avietbl) {
|
|
AVIIndexChainNode *aicn = head;
|
|
|
|
while(aicn) {
|
|
aicn->put(avietbl);
|
|
aicn=aicn->next;
|
|
}
|
|
|
|
delete_chain();
|
|
}
|
|
|
|
void AVIIndexChain::put(AVIIndexEntry2 *avie2tbl) {
|
|
AVIIndexChainNode *aicn = head;
|
|
|
|
while(aicn) {
|
|
aicn->put(avie2tbl);
|
|
aicn=aicn->next;
|
|
}
|
|
|
|
delete_chain();
|
|
}
|
|
|
|
void AVIIndexChain::put(AVIIndexEntry3 *avie3tbl, sint64 offset) {
|
|
AVIIndexChainNode *aicn = head;
|
|
|
|
while(aicn) {
|
|
aicn->put(avie3tbl, offset);
|
|
aicn=aicn->next;
|
|
}
|
|
|
|
delete_chain();
|
|
}
|
|
|
|
AVIIndex::AVIIndex() {
|
|
index = NULL;
|
|
index2 = NULL;
|
|
index3 = NULL;
|
|
}
|
|
|
|
AVIIndex::~AVIIndex() {
|
|
delete[] index;
|
|
delete[] index2;
|
|
delete[] index3;
|
|
}
|
|
|
|
bool AVIIndex::makeIndex() {
|
|
if (!allocateIndex(total_ents)) return false;
|
|
|
|
put(indexPtr());
|
|
|
|
return true;
|
|
}
|
|
|
|
bool AVIIndex::makeIndex2() {
|
|
if (!allocateIndex2(total_ents)) return false;
|
|
|
|
put(index2Ptr());
|
|
|
|
return true;
|
|
}
|
|
|
|
bool AVIIndex::makeIndex3(sint64 offset) {
|
|
if (!allocateIndex3(total_ents)) return false;
|
|
|
|
put(index3Ptr(), offset);
|
|
|
|
return true;
|
|
}
|
|
|
|
void AVIIndex::clear() {
|
|
delete_chain();
|
|
delete[] index;
|
|
delete[] index2;
|
|
delete[] index3;
|
|
index = NULL;
|
|
index2 = NULL;
|
|
index3 = NULL;
|
|
total_ents = 0;
|
|
}
|