mirror of
https://github.com/WinampDesktop/winamp.git
synced 2025-01-01 14:36:00 +00:00
262 lines
7.7 KiB
C++
262 lines
7.7 KiB
C++
|
/*
|
||
|
* The contents of this file are subject to the Mozilla Public
|
||
|
* License Version 1.1 (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.mozilla.org/MPL/
|
||
|
*
|
||
|
* Software distributed under the License is distributed on an "AS
|
||
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
||
|
* implied. See the License for the specific language governing
|
||
|
* rights and limitations under the License.
|
||
|
*
|
||
|
* The Original Code is MPEG4IP.
|
||
|
*
|
||
|
* The Initial Developer of the Original Code is Cisco Systems Inc.
|
||
|
* Portions created by Cisco Systems Inc. are
|
||
|
* Copyright (C) Cisco Systems Inc. 2004. All Rights Reserved.
|
||
|
*
|
||
|
* Contributor(s):
|
||
|
* Bill May wmay@cisco.com
|
||
|
*/
|
||
|
|
||
|
#include "mp4common.h"
|
||
|
|
||
|
/*
|
||
|
* SizeTableProperty is a special version of the MP4TableProperty -
|
||
|
* the BytesProperty will need to set the value before it can read
|
||
|
* from the file
|
||
|
*/
|
||
|
class SizeTableProperty : public MP4TableProperty
|
||
|
{
|
||
|
public:
|
||
|
SizeTableProperty(char *name, MP4IntegerProperty *pCountProperty) :
|
||
|
MP4TableProperty(name, pCountProperty) {};
|
||
|
protected:
|
||
|
void ReadEntry(MP4File *pFile, u_int32_t index) {
|
||
|
// Each table has a size, followed by the length field
|
||
|
// first, read the length
|
||
|
m_pProperties[0]->Read(pFile, index);
|
||
|
MP4IntegerProperty *pIntProp = (MP4IntegerProperty *)m_pProperties[0];
|
||
|
// set the size in the bytes property
|
||
|
MP4BytesProperty *pBytesProp = (MP4BytesProperty *)m_pProperties[1];
|
||
|
pBytesProp->SetValueSize(pIntProp->GetValue(index), index);
|
||
|
// And read the bytes
|
||
|
m_pProperties[1]->Read(pFile, index);
|
||
|
};
|
||
|
};
|
||
|
|
||
|
MP4AvcCAtom::MP4AvcCAtom()
|
||
|
: MP4Atom("avcC")
|
||
|
{
|
||
|
MP4BitfieldProperty *pCount;
|
||
|
MP4TableProperty *pTable;
|
||
|
|
||
|
AddProperty( new MP4Integer8Property("configurationVersion")); /* 0 */
|
||
|
|
||
|
AddProperty( new MP4Integer8Property("AVCProfileIndication")); /* 1 */
|
||
|
|
||
|
AddProperty( new MP4Integer8Property("profile_compatibility")); /* 2 */
|
||
|
|
||
|
AddProperty( new MP4Integer8Property("AVCLevelIndication")); /* 3 */
|
||
|
|
||
|
AddProperty( new MP4BitfieldProperty("reserved", 6)); /* 4 */
|
||
|
AddProperty( new MP4BitfieldProperty("lengthSizeMinusOne", 2)); /* 5 */
|
||
|
AddProperty( new MP4BitfieldProperty("reserved1", 3)); /* 6 */
|
||
|
pCount = new MP4BitfieldProperty("numOfSequenceParameterSets", 5);
|
||
|
AddProperty(pCount); /* 7 */
|
||
|
|
||
|
pTable = new SizeTableProperty("sequenceEntries", pCount);
|
||
|
AddProperty(pTable); /* 8 */
|
||
|
pTable->AddProperty(new MP4Integer16Property("sequenceParameterSetLength"));
|
||
|
pTable->AddProperty(new MP4BytesProperty("sequenceParameterSetNALUnit"));
|
||
|
|
||
|
MP4Integer8Property *pCount2 = new MP4Integer8Property("numOfPictureParameterSets");
|
||
|
AddProperty(pCount2); /* 9 */
|
||
|
|
||
|
pTable = new SizeTableProperty("pictureEntries", pCount2);
|
||
|
AddProperty(pTable); /* 10 */
|
||
|
pTable->AddProperty(new MP4Integer16Property("pictureParameterSetLength"));
|
||
|
pTable->AddProperty(new MP4BytesProperty("pictureParameterSetNALUnit"));
|
||
|
}
|
||
|
|
||
|
void MP4AvcCAtom::Generate()
|
||
|
{
|
||
|
MP4Atom::Generate();
|
||
|
|
||
|
((MP4Integer8Property*)m_pProperties[0])->SetValue(1);
|
||
|
|
||
|
m_pProperties[4]->SetReadOnly(false);
|
||
|
((MP4BitfieldProperty*)m_pProperties[4])->SetValue(0x3f);
|
||
|
m_pProperties[4]->SetReadOnly(true);
|
||
|
|
||
|
m_pProperties[6]->SetReadOnly(false);
|
||
|
((MP4BitfieldProperty*)m_pProperties[6])->SetValue(0x7);
|
||
|
m_pProperties[6]->SetReadOnly(true);
|
||
|
#if 0
|
||
|
// property reserved4 has non-zero fixed values
|
||
|
static u_int8_t reserved4[4] = {
|
||
|
0x00, 0x18, 0xFF, 0xFF,
|
||
|
};
|
||
|
m_pProperties[7]->SetReadOnly(false);
|
||
|
((MP4BytesProperty*)m_pProperties[7])->
|
||
|
SetValue(reserved4, sizeof(reserved4));
|
||
|
m_pProperties[7]->SetReadOnly(true);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Clone - clone my properties to destination atom
|
||
|
//
|
||
|
// this method simplifies duplicating avcC atom properties from
|
||
|
// source to destination file using a single API rather than
|
||
|
// having to copy each property. This API encapsulates the object
|
||
|
// so the application layer need not concern with each property
|
||
|
// thereby isolating any future changes to atom properties.
|
||
|
//
|
||
|
// ----------------------------------------
|
||
|
// property description
|
||
|
// ----------------------------------------
|
||
|
//
|
||
|
// 0 configurationVersion
|
||
|
// 1 AVCProfileIndication
|
||
|
// 2 profile_compatibility
|
||
|
// 3 AVCLevelIndication
|
||
|
// 4 reserved
|
||
|
// 5 lengthSizeMinusOne
|
||
|
// 6 reserved
|
||
|
// 7 number of SPS
|
||
|
// 8 SPS entries
|
||
|
// 9 number of PPS
|
||
|
// 10 PPS entries
|
||
|
//
|
||
|
//
|
||
|
void MP4AvcCAtom::Clone(MP4AvcCAtom *dstAtom)
|
||
|
{
|
||
|
|
||
|
MP4Property *dstProperty;
|
||
|
MP4TableProperty *pTable;
|
||
|
u_int16_t i16;
|
||
|
u_int64_t i32;
|
||
|
u_int64_t i64;
|
||
|
u_int8_t *tmp;
|
||
|
|
||
|
// source pointer Property I16
|
||
|
MP4Integer16Property *spPI16;
|
||
|
// source pointer Property Bytes
|
||
|
MP4BytesProperty *spPB;
|
||
|
|
||
|
// dest pointer Property I16
|
||
|
MP4Integer16Property *dpPI16;
|
||
|
// dest pointer Property Bytes
|
||
|
MP4BytesProperty *dpPB;
|
||
|
|
||
|
|
||
|
// start with defaults and reserved fields
|
||
|
dstAtom->Generate();
|
||
|
|
||
|
// 0, 4, 6 are now generated from defaults
|
||
|
// leaving 1, 2, 3, 5, 7, 8, 9, 10 to export
|
||
|
|
||
|
dstProperty=dstAtom->GetProperty(1);
|
||
|
((MP4Integer8Property *)dstProperty)->SetValue(
|
||
|
((MP4Integer8Property *)m_pProperties[1])->GetValue());
|
||
|
|
||
|
dstProperty=dstAtom->GetProperty(2);
|
||
|
((MP4Integer8Property *)dstProperty)->SetValue(
|
||
|
((MP4Integer8Property *)m_pProperties[2])->GetValue());
|
||
|
|
||
|
dstProperty=dstAtom->GetProperty(3);
|
||
|
((MP4Integer8Property *)dstProperty)->SetValue(
|
||
|
((MP4Integer8Property *)m_pProperties[3])->GetValue());
|
||
|
|
||
|
dstProperty=dstAtom->GetProperty(5);
|
||
|
((MP4BitfieldProperty *)dstProperty)->SetValue(
|
||
|
((MP4BitfieldProperty *)m_pProperties[5])->GetValue());
|
||
|
|
||
|
//
|
||
|
// 7 and 8 are related SPS (one set of sequence parameters)
|
||
|
//
|
||
|
// first the count bitfield
|
||
|
//
|
||
|
dstProperty=dstAtom->GetProperty(7);
|
||
|
dstProperty->SetReadOnly(false);
|
||
|
((MP4BitfieldProperty *)dstProperty)->SetValue(
|
||
|
((MP4BitfieldProperty *)m_pProperties[7])->GetValue());
|
||
|
dstProperty->SetReadOnly(true);
|
||
|
|
||
|
// next export SPS Length and NAL bytes */
|
||
|
|
||
|
// first source pointers
|
||
|
pTable = (MP4TableProperty *) m_pProperties[8];
|
||
|
spPI16 = (MP4Integer16Property *)pTable->GetProperty(0);
|
||
|
spPB = (MP4BytesProperty *)pTable->GetProperty(1);
|
||
|
|
||
|
// now dest pointers
|
||
|
dstProperty=dstAtom->GetProperty(8);
|
||
|
pTable = (MP4TableProperty *) dstProperty;
|
||
|
dpPI16 = (MP4Integer16Property *)pTable->GetProperty(0);
|
||
|
dpPB = (MP4BytesProperty *)pTable->GetProperty(1);
|
||
|
|
||
|
// sps length
|
||
|
i16 = spPI16->GetValue();
|
||
|
i64 = i16;
|
||
|
// FIXME - this leaves m_maxNumElements =2
|
||
|
// but src atom m_maxNumElements is 1
|
||
|
dpPI16->InsertValue(i64, 0);
|
||
|
|
||
|
// export byte array
|
||
|
i32 = i16;
|
||
|
// copy bytes to local buffer
|
||
|
tmp = (u_int8_t *)MP4Malloc(i32);
|
||
|
ASSERT(tmp != NULL);
|
||
|
spPB->CopyValue(tmp, 0);
|
||
|
// set element count
|
||
|
dpPB->SetCount(1);
|
||
|
// copy bytes
|
||
|
dpPB->SetValue(tmp, i32, 0);
|
||
|
MP4Free((void *)tmp);
|
||
|
|
||
|
//
|
||
|
// 9 and 10 are related PPS (one set of picture parameters)
|
||
|
//
|
||
|
// first the integer8 count
|
||
|
//
|
||
|
dstProperty=dstAtom->GetProperty(9);
|
||
|
dstProperty->SetReadOnly(false);
|
||
|
((MP4Integer8Property *)dstProperty)->SetValue(
|
||
|
((MP4Integer8Property *)m_pProperties[9])->GetValue());
|
||
|
dstProperty->SetReadOnly(true);
|
||
|
|
||
|
// next export PPS Length and NAL bytes */
|
||
|
|
||
|
// first source pointers
|
||
|
pTable = (MP4TableProperty *) m_pProperties[10];
|
||
|
spPI16 = (MP4Integer16Property *)pTable->GetProperty(0);
|
||
|
spPB = (MP4BytesProperty *)pTable->GetProperty(1);
|
||
|
|
||
|
// now dest pointers
|
||
|
dstProperty=dstAtom->GetProperty(10);
|
||
|
pTable = (MP4TableProperty *) dstProperty;
|
||
|
dpPI16 = (MP4Integer16Property *)pTable->GetProperty(0);
|
||
|
dpPB = (MP4BytesProperty *)pTable->GetProperty(1);
|
||
|
|
||
|
// pps length
|
||
|
i16 = spPI16->GetValue();
|
||
|
i64 = i16;
|
||
|
dpPI16->InsertValue(i64, 0);
|
||
|
|
||
|
// export byte array
|
||
|
i32 = i16;
|
||
|
// copy bytes to local buffer
|
||
|
tmp = (u_int8_t *)MP4Malloc(i32);
|
||
|
ASSERT(tmp != NULL);
|
||
|
spPB->CopyValue(tmp, 0);
|
||
|
// set element count
|
||
|
dpPB->SetCount(1);
|
||
|
// copy bytes
|
||
|
dpPB->SetValue(tmp, i32, 0);
|
||
|
MP4Free((void *)tmp);
|
||
|
}
|
||
|
|
||
|
|