mirror of
https://github.com/SickGear/SickGear.git
synced 2024-12-04 10:23:37 +00:00
980e05cc99
Backported 400 revisions from rev 1de4961-8897c5b (2018-2014). Move core/benchmark, core/cmd_line, core/memory, core/profiler and core/timeout to core/optional/* Remove metadata/qt* PORT: Version 2.0a3 (inline with 3.0a3 @ f80c7d5). Basic Support for XMP Packets. tga: improvements to adhere more closely to the spec. pdf: slightly improved parsing. rar: fix TypeError on unknown block types. Add MacRoman win32 codepage. tiff/exif: support SubIFDs and tiled images. Add method to export metadata in dictionary. mpeg_video: don't attempt to parse Stream past length. mpeg_video: parse ESCR correctly, add SCR value. Change centralise CustomFragments. field: don't set parser class if class is None, to enable autodetect. field: add value/display for CustomFragment. parser: inline warning to enable tracebacks in debug mode. Fix empty bytestrings in makePrintable. Fix contentSize in jpeg.py to account for image_data blocks. Fix the ELF parser. Enhance the AR archive parser. elf parser: fix wrong wrong fields order in parsing little endian section flags. elf parser: add s390 as a machine type. Flesh out mp4 parser. PORT: Version 2.0a1 (inline with 3.0a1). Major refactoring and PEP8. Fix ResourceWarning warnings on files. Add a close() method and support for the context manager protocol ("with obj: ...") to parsers, input and output streams. metadata: get comment from ZIP. Support for InputIOStream.read(0). Fix sizeGe when size is None. Remove unused new_seekable_field_set file. Remove parser Mapsforge .map. Remove parser Parallel Realities Starfighter .pak files. sevenzip: fix for newer archives. java: update access flags and modifiers for Java 1.7 and update description text for most recent Java. Support ustar prefix field in tar archives. Remove file_system* parsers. Remove misc parsers 3d0, 3ds, gnome_keyring, msoffice*, mstask, ole*, word*. Remove program parsers macho, nds, prc. Support non-8bit Character subclasses. Python parser supports Python 3.7. Enhance mpeg_ts parser to support MTS/M2TS. Support for creation date in tiff. Change don't hardcode errno constant. PORT: 1.9.1 Internal Only: The following are legacy reference to upstream commit messages. Relevant changes up to b0a115f8. Use integer division. Replace HACHOIR_ERRORS with Exception. Fix metadata.Data: make it sortable. Import fixes from e7de492. PORT: Version 2.0a1 (inline with 3.0a1 @ e9f8fad). Replace hachoir.core.field with hachoir.field Replace hachoir.core.stream with hachoir.stream Remove the compatibility module for PY1.5 to PY2.5. metadata: support TIFF picture. metadata: fix string normalization. metadata: fix datetime regex Fix hachoir bug #57. FileFromInputStream: fix comparison between None and an int. InputIOStream: open the file in binary mode.
467 lines
17 KiB
Python
467 lines
17 KiB
Python
"""
|
|
Parser for AVCHD/Blu-ray formats
|
|
|
|
Notice: This parser is based off reverse-engineering efforts.
|
|
It is NOT based on official specifications, and is subject to change as
|
|
more information becomes available. There's a lot of guesswork here, so if you find
|
|
that something disagrees with an official specification, please change it.
|
|
|
|
Notice: This parser has NOT been tested on Blu-ray disc data, only on files
|
|
taken from AVCHD camcorders.
|
|
|
|
Author: Robert Xiao
|
|
Creation: December 30, 2010
|
|
|
|
References:
|
|
- Wikipedia: http://en.wikipedia.org/wiki/AVCHD
|
|
- European patent EP1821310: http://www.freepatentsonline.com/EP1821310.html
|
|
"""
|
|
|
|
"""
|
|
File structure:
|
|
Root (/PRIVATE/AVCHD, /AVCHD, /, etc.)
|
|
AVCHDTN/: (AVCHD only)
|
|
THUMB.TDT: Thumbnail Data: stored as a series of 16KiB pages, where each thumbnail starts on a page boundary
|
|
THUMB.TID: Thumbnail Index (TIDX), unknown format
|
|
BDMV/:
|
|
INDEX.BDM|index.bdmv: Bluray Disc Metadata (INDX): Clip index file
|
|
MOVIEOBJ.BDM|MovieObject.bdmv: Bluray Disc Metadata (MOBJ): Clip description file
|
|
AUXDATA/: (Optional, Blu-ray only)
|
|
sound.bdmv: Sound(s) associated with HDMV Interactive Graphic streams applications
|
|
?????.otf: Font(s) associated with Text subtitle applications
|
|
BACKUP/: (Optional)
|
|
[Copies of *.bdmv, CLIPINF/* and PLAYLIST/*]
|
|
CLIPINF/:
|
|
?????.CPI/?????.clpi: Clip information (HDMV)
|
|
PLAYLIST/:
|
|
?????.MPL/?????.mpls: Movie Playlist information (MPLS)
|
|
STREAM/:
|
|
?????.MTS|?????.m2ts: BDAV MPEG-2 Transport Stream (video file)
|
|
SSIF/: (Blu-ray 3D only)
|
|
?????.ssif: Stereoscopic Interleaved file
|
|
IISVPL/: (Optional?, AVCHD only?)
|
|
?????.VPL: Virtual Playlist? (MPLS)
|
|
"""
|
|
|
|
from hachoir.parser import HachoirParser
|
|
from hachoir.field import (RootSeekableFieldSet, FieldSet,
|
|
RawBytes, Bytes, String, Bits, UInt8, UInt16, UInt32, PascalString8, Enum)
|
|
from hachoir.core.endian import BIG_ENDIAN
|
|
from hachoir.core.iso639 import ISO639_2
|
|
from hachoir.core.text_handler import textHandler, hexadecimal
|
|
from datetime import datetime
|
|
|
|
|
|
def fromhex(field):
|
|
return int('%x' % field.value)
|
|
|
|
|
|
class AVCHDTimestamp(FieldSet):
|
|
static_size = 8 * 8
|
|
|
|
def createFields(self):
|
|
yield textHandler(UInt8(self, "unknown", description="0x1E"), hexadecimal)
|
|
yield textHandler(UInt8(self, "century"), hexadecimal)
|
|
yield textHandler(UInt8(self, "year"), hexadecimal)
|
|
yield textHandler(UInt8(self, "month"), hexadecimal)
|
|
yield textHandler(UInt8(self, "day"), hexadecimal)
|
|
yield textHandler(UInt8(self, "hour"), hexadecimal)
|
|
yield textHandler(UInt8(self, "minute"), hexadecimal)
|
|
yield textHandler(UInt8(self, "second"), hexadecimal)
|
|
|
|
def createValue(self):
|
|
return datetime(fromhex(self['century']) * 100 + fromhex(self['year']),
|
|
fromhex(self['month']), fromhex(self['day']),
|
|
fromhex(self['hour']), fromhex(self['minute']), fromhex(self['second']))
|
|
|
|
|
|
class AVCHDGenericChunk(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield RawBytes(self, "raw[]", self['size'].value)
|
|
|
|
|
|
class AVCHDINDX_0(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield RawBytes(self, "unknown[]", 22)
|
|
yield UInt32(self, "count")
|
|
for i in xrange(self['count'].value):
|
|
yield RawBytes(self, "data[]", 12)
|
|
|
|
|
|
class AVCHDIDEX_0(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield RawBytes(self, "unknown[]", 40)
|
|
yield AVCHDTimestamp(self, "last_modified")
|
|
yield RawBytes(self, "unknown[]", self._size // 8 - 52)
|
|
|
|
|
|
class AVCHDMOBJ_Chunk(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "unknown[]")
|
|
yield UInt32(self, "index")
|
|
yield UInt32(self, "unknown[]")
|
|
yield textHandler(UInt32(self, "unknown_id"), hexadecimal)
|
|
yield UInt32(self, "unknown[]")
|
|
yield textHandler(UInt32(self, "playlist_id"), lambda field: '%05d' % field.value)
|
|
yield UInt32(self, "unknown[]")
|
|
|
|
|
|
class AVCHDMPLS_StreamEntry(FieldSet):
|
|
ENTRYTYPE = {1: 'PlayItem on disc',
|
|
2: 'SubPath on disc',
|
|
3: 'PlayItem in local storage',
|
|
4: 'SubPath in local storage'}
|
|
|
|
def createFields(self):
|
|
yield UInt8(self, "size")
|
|
self._size = (self['size'].value + 1) * 8
|
|
yield Enum(UInt8(self, "type"), self.ENTRYTYPE)
|
|
if self['type'].value in (1, 3):
|
|
yield textHandler(UInt16(self, "pid", "PID of item in clip stream m2ts file"), hexadecimal)
|
|
else: # 2,4
|
|
'''
|
|
The patent says:
|
|
ref_to_SubPath_id
|
|
ref_to_SubClip_entry_id
|
|
ref_to_Stream_PID_of_subClip
|
|
Sizes aren't given, though, so I cannot determine the format without a sample.
|
|
'''
|
|
pass
|
|
|
|
|
|
class AVCHDMPLS_StreamAttribs(FieldSet):
|
|
STREAMTYPE = {
|
|
0x01: "V_MPEG1",
|
|
0x02: "V_MPEG2",
|
|
0x1B: "V_AVC",
|
|
0xEA: "V_VC1",
|
|
0x03: "A_MPEG1",
|
|
0x04: "A_MPEG2",
|
|
0x80: "A_LPCM",
|
|
0x81: "A_AC3",
|
|
0x84: "A_AC3_PLUS",
|
|
0xA1: "A_AC3_PLUS_SEC",
|
|
0x83: "A_TRUEHD",
|
|
0x82: "A_DTS",
|
|
0x85: "A_DTS-HD",
|
|
0xA2: "A_DTS-HD_SEC",
|
|
0x86: "A_DTS-MA",
|
|
0x90: "S_PGS",
|
|
0x91: "S_IGS",
|
|
0x92: "T_SUBTITLE",
|
|
}
|
|
|
|
# Enumerations taken from "ClownBD's CLIPINF Editor". Values may not be accurate.
|
|
def createFields(self):
|
|
yield UInt8(self, "size")
|
|
self._size = (self['size'].value + 1) * 8
|
|
yield Enum(UInt8(self, "type"), self.STREAMTYPE)
|
|
if self['type'].display.startswith('V'): # Video
|
|
yield Enum(Bits(self, "resolution", 4),
|
|
{1: '480i', 2: '576i', 3: '480p', 4: '1080i', 5: '720p', 6: '1080p', 7: '576p'})
|
|
yield Enum(Bits(self, "fps", 4), {1: '24/1.001', 2: '24', 3: '25', 4: '30/1.001', 6: '50', 7: '60/1.001'})
|
|
yield Enum(UInt8(self, "aspect_ratio"), {0x20: '4:3', 0x30: '16:9'})
|
|
elif self['type'].display.startswith('A'): # Audio
|
|
yield Enum(Bits(self, "channel_layout", 4), {1: 'Mono', 3: 'Stereo', 6: 'Multi', 12: 'Combi'})
|
|
yield Enum(Bits(self, "sample_rate", 4),
|
|
{1: '48KHz', 4: '96KHz', 5: '192KHz', 12: '48-192KHz', 14: '48-96KHz'})
|
|
yield Enum(String(self, "language", 3), ISO639_2)
|
|
elif self['type'].display.startswith('T'): # Text subtitle
|
|
yield UInt8(self, "unknown[]")
|
|
yield Enum(String(self, "language", 3), ISO639_2)
|
|
elif self['type'].display.startswith('S'): # Graphics
|
|
yield Enum(String(self, "language", 3), ISO639_2)
|
|
else:
|
|
pass
|
|
|
|
|
|
class AVCHDMPLS_Stream(FieldSet):
|
|
def createFields(self):
|
|
yield AVCHDMPLS_StreamEntry(self, "entry")
|
|
yield AVCHDMPLS_StreamAttribs(self, "attribs")
|
|
|
|
|
|
class AVCHDMPLS_PlayItem(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield UInt16(self, "unknown[]")
|
|
yield UInt8(self, "video_count", "Number of video stream entries")
|
|
yield UInt8(self, "audio_count", "Number of video stream entries")
|
|
yield UInt8(self, "subtitle_count", "Number of presentation graphics/text subtitle entries")
|
|
yield UInt8(self, "ig_count", "Number of interactive graphics entries")
|
|
yield RawBytes(self, "unknown[]", 8)
|
|
for i in xrange(self['video_count'].value):
|
|
yield AVCHDMPLS_Stream(self, "video[]")
|
|
for i in xrange(self['audio_count'].value):
|
|
yield AVCHDMPLS_Stream(self, "audio[]")
|
|
for i in xrange(self['subtitle_count'].value):
|
|
yield AVCHDMPLS_Stream(self, "subtitle[]")
|
|
for i in xrange(self['ig_count'].value):
|
|
yield AVCHDMPLS_Stream(self, "ig[]")
|
|
|
|
|
|
class AVCHDMPLS_0_Chunk(FieldSet):
|
|
def createFields(self):
|
|
yield UInt16(self, "size")
|
|
self._size = (self['size'].value + 2) * 8
|
|
yield Bytes(self, "clip_id", 5)
|
|
yield Bytes(self, "clip_type", 4)
|
|
yield RawBytes(self, "unknown[]", 3)
|
|
yield UInt32(self, "clip_start_time[]", "clip start time (units unknown)")
|
|
yield UInt32(self, "clip_end_time[]", "clip end time (units unknown)")
|
|
yield RawBytes(self, "unknown[]", 10)
|
|
yield AVCHDMPLS_PlayItem(self, "playitem")
|
|
|
|
|
|
class AVCHDMPLS_0(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield UInt32(self, "count")
|
|
yield UInt16(self, "unknown[]")
|
|
for i in xrange(self['count'].value):
|
|
yield AVCHDMPLS_0_Chunk(self, "chunk[]")
|
|
|
|
|
|
class AVCHDMPLS_PlayItemMark(FieldSet):
|
|
def createFields(self):
|
|
yield UInt16(self, "unknown[]")
|
|
yield UInt16(self, "playitem_idx", "Index of the associated PlayItem")
|
|
yield UInt32(self, "mark_time", "Marker time in clip (units unknown)")
|
|
yield RawBytes(self, "unknown", 6)
|
|
|
|
|
|
class AVCHDMPLS_1(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield UInt16(self, "count")
|
|
for i in xrange(self['count'].value):
|
|
yield AVCHDMPLS_PlayItemMark(self, "chunk[]")
|
|
|
|
|
|
class AVCHDPLEX_1_Chunk(FieldSet):
|
|
static_size = 66 * 8
|
|
|
|
def createFields(self):
|
|
yield RawBytes(self, "unknown[]", 10)
|
|
yield AVCHDTimestamp(self, "date")
|
|
yield RawBytes(self, "unknown[]", 1)
|
|
yield PascalString8(self, "date")
|
|
|
|
def createValue(self):
|
|
return self['date'].value
|
|
|
|
|
|
class AVCHDPLEX_0(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield RawBytes(self, "unknown[]", 10)
|
|
yield AVCHDTimestamp(self, "last_modified")
|
|
yield RawBytes(self, "unknown[]", 2)
|
|
yield PascalString8(self, "date")
|
|
|
|
|
|
class AVCHDPLEX_1(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield UInt16(self, "count")
|
|
for i in xrange(self['count'].value):
|
|
yield AVCHDPLEX_1_Chunk(self, "chunk[]")
|
|
|
|
|
|
class AVCHDCLPI_1(FieldSet):
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield RawBytes(self, "unknown[]", 10)
|
|
yield textHandler(UInt16(self, "video_pid", "PID of video data in stream file"), hexadecimal)
|
|
yield AVCHDMPLS_StreamAttribs(self, "video_attribs")
|
|
yield textHandler(UInt16(self, "audio_pid", "PID of audio data in stream file"), hexadecimal)
|
|
yield AVCHDMPLS_StreamAttribs(self, "audio_attribs")
|
|
|
|
|
|
def AVCHDIDEX(self):
|
|
yield AVCHDIDEX_0(self, "chunk[]")
|
|
yield AVCHDGenericChunk(self, "chunk[]")
|
|
|
|
|
|
def AVCHDPLEX(self):
|
|
yield AVCHDPLEX_0(self, "chunk[]")
|
|
yield AVCHDPLEX_1(self, "chunk[]")
|
|
yield AVCHDGenericChunk(self, "chunk[]")
|
|
|
|
|
|
def AVCHDCLEX(self):
|
|
yield AVCHDGenericChunk(self, "chunk[]")
|
|
yield AVCHDGenericChunk(self, "chunk[]")
|
|
|
|
|
|
class AVCHDChunkWithHeader(FieldSet):
|
|
TYPES = {'IDEX': AVCHDIDEX,
|
|
'PLEX': AVCHDPLEX,
|
|
'CLEX': AVCHDCLEX, }
|
|
|
|
def createFields(self):
|
|
yield UInt32(self, "size")
|
|
self._size = (self['size'].value + 4) * 8
|
|
yield UInt32(self, "unknown[]", "24")
|
|
yield UInt32(self, "unknown[]", "1")
|
|
yield UInt32(self, "unknown[]", "0x10000100")
|
|
yield UInt32(self, "unknown[]", "24")
|
|
yield UInt32(self, "size2")
|
|
assert self['size'].value == self['size2'].value + 20
|
|
yield Bytes(self, "magic", 4)
|
|
yield RawBytes(self, "unknown[]", 36)
|
|
for field in self.TYPES[self['magic'].value](self):
|
|
yield field
|
|
|
|
|
|
class AVCHDINDX(HachoirParser, RootSeekableFieldSet):
|
|
endian = BIG_ENDIAN
|
|
MAGIC = "INDX0"
|
|
PARSER_TAGS = {
|
|
"id": "bdmv_index",
|
|
"category": "video",
|
|
"file_ext": ("bdm", "bdmv"),
|
|
"magic": ((MAGIC, 0),),
|
|
"min_size": 8, # INDX0?00
|
|
"description": "INDEX.BDM",
|
|
}
|
|
|
|
def __init__(self, stream, **args):
|
|
RootSeekableFieldSet.__init__(self, None, "root", stream, None, stream.askSize(self))
|
|
HachoirParser.__init__(self, stream, **args)
|
|
|
|
def validate(self):
|
|
if self.stream.readBytes(0, len(self.MAGIC)) != self.MAGIC:
|
|
return "Invalid magic"
|
|
return True
|
|
|
|
def createFields(self):
|
|
yield Bytes(self, "filetype", 4, "File type (INDX)")
|
|
yield Bytes(self, "fileversion", 4, "File version (0?00)")
|
|
yield UInt32(self, "offset[0]")
|
|
yield UInt32(self, "offset[1]")
|
|
self.seekByte(self['offset[0]'].value)
|
|
yield AVCHDINDX_0(self, "chunk[]")
|
|
self.seekByte(self['offset[1]'].value)
|
|
yield AVCHDChunkWithHeader(self, "chunk[]")
|
|
|
|
|
|
class AVCHDMOBJ(HachoirParser, RootSeekableFieldSet):
|
|
endian = BIG_ENDIAN
|
|
MAGIC = "MOBJ0"
|
|
PARSER_TAGS = {
|
|
"id": "bdmv_mobj",
|
|
"category": "video",
|
|
"file_ext": ("bdm", "bdmv"),
|
|
"magic": ((MAGIC, 0),),
|
|
"min_size": 8, # MOBJ0?00
|
|
"description": "MOVIEOBJ.BDM",
|
|
}
|
|
|
|
def __init__(self, stream, **args):
|
|
RootSeekableFieldSet.__init__(self, None, "root", stream, None, stream.askSize(self))
|
|
HachoirParser.__init__(self, stream, **args)
|
|
|
|
def validate(self):
|
|
if self.stream.readBytes(0, len(self.MAGIC)) != self.MAGIC:
|
|
return "Invalid magic"
|
|
return True
|
|
|
|
def createFields(self):
|
|
yield Bytes(self, "filetype", 4, "File type (MOBJ)")
|
|
yield Bytes(self, "fileversion", 4, "File version (0?00)")
|
|
yield RawBytes(self, "unknown[]", 32)
|
|
yield UInt32(self, "size")
|
|
yield UInt32(self, "unknown[]")
|
|
yield UInt16(self, "count")
|
|
yield textHandler(UInt32(self, "unknown_id"), hexadecimal)
|
|
for i in xrange(1, self['count'].value):
|
|
yield AVCHDMOBJ_Chunk(self, "movie_object[]")
|
|
|
|
|
|
class AVCHDMPLS(HachoirParser, RootSeekableFieldSet):
|
|
endian = BIG_ENDIAN
|
|
MAGIC = "MPLS0"
|
|
PARSER_TAGS = {
|
|
"id": "bdmv_mpls",
|
|
"category": "video",
|
|
"file_ext": ("mpl", "mpls", "vpl"),
|
|
"magic": ((MAGIC, 0),),
|
|
"min_size": 8, # MPLS0?00
|
|
"description": "MPLS",
|
|
}
|
|
|
|
def __init__(self, stream, **args):
|
|
RootSeekableFieldSet.__init__(self, None, "root", stream, None, stream.askSize(self))
|
|
HachoirParser.__init__(self, stream, **args)
|
|
|
|
def validate(self):
|
|
if self.stream.readBytes(0, len(self.MAGIC)) != self.MAGIC:
|
|
return "Invalid magic"
|
|
return True
|
|
|
|
def createFields(self):
|
|
yield Bytes(self, "filetype", 4, "File type (MPLS)")
|
|
yield Bytes(self, "fileversion", 4, "File version (0?00)")
|
|
yield UInt32(self, "offset[0]")
|
|
yield UInt32(self, "offset[1]")
|
|
yield UInt32(self, "offset[2]")
|
|
self.seekByte(self['offset[0]'].value)
|
|
yield AVCHDMPLS_0(self, "chunk[]")
|
|
self.seekByte(self['offset[1]'].value)
|
|
yield AVCHDMPLS_1(self, "chunk[]")
|
|
self.seekByte(self['offset[2]'].value)
|
|
yield AVCHDChunkWithHeader(self, "chunk[]")
|
|
|
|
|
|
class AVCHDCLPI(HachoirParser, RootSeekableFieldSet):
|
|
endian = BIG_ENDIAN
|
|
MAGIC = "HDMV0"
|
|
PARSER_TAGS = {
|
|
"id": "bdmv_clpi",
|
|
"category": "video",
|
|
"file_ext": ("cpi", "clpi"),
|
|
"magic": ((MAGIC, 0),),
|
|
"min_size": 8, # HDMV0?00
|
|
"description": "HDMV",
|
|
}
|
|
|
|
def __init__(self, stream, **args):
|
|
RootSeekableFieldSet.__init__(self, None, "root", stream, None, stream.askSize(self))
|
|
HachoirParser.__init__(self, stream, **args)
|
|
|
|
def validate(self):
|
|
if self.stream.readBytes(0, len(self.MAGIC)) != self.MAGIC:
|
|
return "Invalid magic"
|
|
return True
|
|
|
|
def createFields(self):
|
|
yield Bytes(self, "filetype", 4, "File type (HDMV)")
|
|
yield Bytes(self, "fileversion", 4, "File version (0?00)")
|
|
yield UInt32(self, "offset[]")
|
|
yield UInt32(self, "offset[]")
|
|
yield UInt32(self, "offset[]")
|
|
yield UInt32(self, "offset[]")
|
|
yield UInt32(self, "offset[]")
|
|
self.seekByte(self['offset[0]'].value)
|
|
yield AVCHDGenericChunk(self, "chunk[]")
|
|
self.seekByte(self['offset[1]'].value)
|
|
yield AVCHDCLPI_1(self, "chunk[]")
|
|
self.seekByte(self['offset[2]'].value)
|
|
yield AVCHDGenericChunk(self, "chunk[]")
|
|
self.seekByte(self['offset[3]'].value)
|
|
yield AVCHDGenericChunk(self, "chunk[]")
|
|
self.seekByte(self['offset[4]'].value)
|
|
yield AVCHDChunkWithHeader(self, "chunk[]")
|