demuxer
This commit is contained in:
		
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@ -173,4 +173,4 @@ cython_debug/
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
# PyPI configuration file
 | 
					# PyPI configuration file
 | 
				
			||||||
.pypirc
 | 
					.pypirc
 | 
				
			||||||
 | 
					/test.mp4
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										0
									
								
								mp4/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								mp4/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										20
									
								
								mp4/codec.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								mp4/codec.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,20 @@
 | 
				
			|||||||
 | 
					# RozK
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class NullCodec:
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def name(self):
 | 
				
			||||||
 | 
					        return b"<None>"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Codec:
 | 
				
			||||||
 | 
					    __slots__ = '_ref'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self, ref):
 | 
				
			||||||
 | 
					        self._ref = ref
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def _as_parameter_(self):
 | 
				
			||||||
 | 
					        return self._ref
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def name(self):
 | 
				
			||||||
 | 
					        return self._ref.contents.name
 | 
				
			||||||
							
								
								
									
										54
									
								
								mp4/context.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								mp4/context.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,54 @@
 | 
				
			|||||||
 | 
					# RozK
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from . import libav
 | 
				
			||||||
 | 
					from .codec import Codec
 | 
				
			||||||
 | 
					from .stream import NullStream, Stream
 | 
				
			||||||
 | 
					from .packet import Packet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Context:
 | 
				
			||||||
 | 
					    __slots__ = '_ref'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self):
 | 
				
			||||||
 | 
					        self._ref = libav.alloc_context()
 | 
				
			||||||
 | 
					        if not self._ref:
 | 
				
			||||||
 | 
					            raise MemoryError
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __del__(self):
 | 
				
			||||||
 | 
					        if self._ref:
 | 
				
			||||||
 | 
					            libav.free_context(self._ref)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def _as_parameter_(self):
 | 
				
			||||||
 | 
					        return self._ref
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def open_input(self, url):
 | 
				
			||||||
 | 
					        if not self._ref:
 | 
				
			||||||
 | 
					            return
 | 
				
			||||||
 | 
					        errcode = libav.open_input(self._ref, url)
 | 
				
			||||||
 | 
					        if errcode < 0:
 | 
				
			||||||
 | 
					            raise Exception(f"Failed to open: {url}")
 | 
				
			||||||
 | 
					        errcode = libav.find_stream_info(self._ref)
 | 
				
			||||||
 | 
					        if errcode < 0:
 | 
				
			||||||
 | 
					            libav.close_input(self._ref)
 | 
				
			||||||
 | 
					            raise Exception("Failed to find stream info")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def close_input(self):
 | 
				
			||||||
 | 
					        if self._ref:
 | 
				
			||||||
 | 
					            libav.close_input(self._ref)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def find_stream(self, type):
 | 
				
			||||||
 | 
					        if not self._ref:
 | 
				
			||||||
 | 
					            return NullStream()
 | 
				
			||||||
 | 
					        index, codec_ref = libav.find_best_stream(self._ref, type)
 | 
				
			||||||
 | 
					        if index < 0 or not codec_ref:
 | 
				
			||||||
 | 
					            return NullStream()
 | 
				
			||||||
 | 
					        return Stream(index, Codec(codec_ref))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def read_packet(self):
 | 
				
			||||||
 | 
					        if not self._ref:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					        packet = Packet()
 | 
				
			||||||
 | 
					        errcode = libav.read_frame(self._ref, packet)
 | 
				
			||||||
 | 
					        if errcode < 0:
 | 
				
			||||||
 | 
					            return None
 | 
				
			||||||
 | 
					        return packet
 | 
				
			||||||
							
								
								
									
										19
									
								
								mp4/demuxer.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								mp4/demuxer.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,19 @@
 | 
				
			|||||||
 | 
					# RozK
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from . import libav
 | 
				
			||||||
 | 
					from .context import Context
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Demuxer:
 | 
				
			||||||
 | 
					    __slots__ = 'context', 'video_stream', 'audio_stream'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self, path):
 | 
				
			||||||
 | 
					        self.context = Context()
 | 
				
			||||||
 | 
					        self.context.open_input("file:" + path)
 | 
				
			||||||
 | 
					        self.video_stream = self.context.find_stream(libav.AVMEDIA_TYPE_VIDEO)
 | 
				
			||||||
 | 
					        self.audio_stream = self.context.find_stream(libav.AVMEDIA_TYPE_AUDIO)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def read_packet(self):
 | 
				
			||||||
 | 
					        return self.context.read_packet()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def close(self):
 | 
				
			||||||
 | 
					        self.context.close_input()
 | 
				
			||||||
							
								
								
									
										112
									
								
								mp4/libav.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								mp4/libav.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,112 @@
 | 
				
			|||||||
 | 
					# RozK
 | 
				
			||||||
 | 
					# https://www.ffmpeg.org/doxygen/trunk/group__libavf.html
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import ctypes
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat = ctypes.cdll.LoadLibrary('libavformat.so')
 | 
				
			||||||
 | 
					_avcodec = ctypes.cdll.LoadLibrary('libavcodec.so')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class AVFormatContext(ctypes.Structure):
 | 
				
			||||||
 | 
					    pass
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					AVFormatContext_p = ctypes.POINTER(AVFormatContext)
 | 
				
			||||||
 | 
					AVFormatContext_pp = ctypes.POINTER(AVFormatContext_p)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.avformat_alloc_context.restype = AVFormatContext_p
 | 
				
			||||||
 | 
					_avformat.avformat_alloc_context.argtypes = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def alloc_context():
 | 
				
			||||||
 | 
					    return _avformat.avformat_alloc_context()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.avformat_free_context.restype = None
 | 
				
			||||||
 | 
					_avformat.avformat_free_context.argtypes = [AVFormatContext_p] # context
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def free_context(context):
 | 
				
			||||||
 | 
					    _avformat.avformat_free_context(context)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.avformat_open_input.restype = ctypes.c_int
 | 
				
			||||||
 | 
					_avformat.avformat_open_input.argtypes = [
 | 
				
			||||||
 | 
					    AVFormatContext_pp, # context
 | 
				
			||||||
 | 
					    ctypes.c_char_p, # url
 | 
				
			||||||
 | 
					    ctypes.c_void_p, # format
 | 
				
			||||||
 | 
					    ctypes.POINTER(ctypes.c_void_p)] # options
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def open_input(context, url):
 | 
				
			||||||
 | 
					    return _avformat.avformat_open_input(ctypes.byref(context), url.encode('ascii', 'ignore'), None, None)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.avformat_close_input.restype = None
 | 
				
			||||||
 | 
					_avformat.avformat_close_input.argtypes = [AVFormatContext_pp] # context
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def close_input(context):
 | 
				
			||||||
 | 
					    _avformat.avformat_close_input(ctypes.byref(context))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.avformat_find_stream_info.restype = ctypes.c_int
 | 
				
			||||||
 | 
					_avformat.avformat_find_stream_info.argtypes = [
 | 
				
			||||||
 | 
					    AVFormatContext_p, # context
 | 
				
			||||||
 | 
					    ctypes.POINTER(ctypes.c_void_p)] # options
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def find_stream_info(context):
 | 
				
			||||||
 | 
					    return _avformat.avformat_find_stream_info(context, None)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					AVMEDIA_TYPE_UNKNOWN = -1
 | 
				
			||||||
 | 
					AVMEDIA_TYPE_VIDEO = 0
 | 
				
			||||||
 | 
					AVMEDIA_TYPE_AUDIO = 1
 | 
				
			||||||
 | 
					AVMEDIA_TYPE_DATA = 2
 | 
				
			||||||
 | 
					AVMEDIA_TYPE_SUBTITLE = 3
 | 
				
			||||||
 | 
					AVMEDIA_TYPE_ATTACHMENT = 4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class AVCodec(ctypes.Structure):
 | 
				
			||||||
 | 
					    _fields_ = [
 | 
				
			||||||
 | 
					        ("name", ctypes.c_char_p),
 | 
				
			||||||
 | 
					        ("long_name", ctypes.c_char_p)]
 | 
				
			||||||
 | 
					        # ...
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					AVCodec_p = ctypes.POINTER(AVCodec)
 | 
				
			||||||
 | 
					AVCodec_pp = ctypes.POINTER(AVCodec_p)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.av_find_best_stream.restype = ctypes.c_int
 | 
				
			||||||
 | 
					_avformat.av_find_best_stream.argtypes = [
 | 
				
			||||||
 | 
					    AVFormatContext_p, # context
 | 
				
			||||||
 | 
					    ctypes.c_int, # type
 | 
				
			||||||
 | 
					    ctypes.c_int, # wanted stream
 | 
				
			||||||
 | 
					    ctypes.c_int, # related stream
 | 
				
			||||||
 | 
					    AVCodec_pp, # decoder
 | 
				
			||||||
 | 
					    ctypes.c_int] # flags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def find_best_stream(context, type):
 | 
				
			||||||
 | 
					    codec = AVCodec_p()
 | 
				
			||||||
 | 
					    index = _avformat.av_find_best_stream(context, type, -1, -1, ctypes.byref(codec), 0)
 | 
				
			||||||
 | 
					    return index, codec
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class AVPacket(ctypes.Structure):
 | 
				
			||||||
 | 
					    _fields_ = [
 | 
				
			||||||
 | 
					        ("buf", ctypes.c_void_p),
 | 
				
			||||||
 | 
					        ("pts", ctypes.c_int64),
 | 
				
			||||||
 | 
					        ("dts", ctypes.c_int64),
 | 
				
			||||||
 | 
					        ("data", ctypes.c_void_p),
 | 
				
			||||||
 | 
					        ("size", ctypes.c_int),
 | 
				
			||||||
 | 
					        ("stream_index", ctypes.c_int)]
 | 
				
			||||||
 | 
					        # ...
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					AVPacket_p = ctypes.POINTER(AVPacket)
 | 
				
			||||||
 | 
					AVPacket_pp = ctypes.POINTER(AVPacket_p)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.av_packet_alloc.restype = AVPacket_p
 | 
				
			||||||
 | 
					_avformat.av_packet_alloc.argtypes = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def packet_alloc():
 | 
				
			||||||
 | 
					    return _avformat.av_packet_alloc()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.av_packet_free.restype = None
 | 
				
			||||||
 | 
					_avformat.av_packet_free.argtypes = [AVPacket_pp] # packet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def packet_free(packet):
 | 
				
			||||||
 | 
					    _avformat.av_packet_free(ctypes.byref(packet))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					_avformat.av_read_frame.restype = ctypes.c_int
 | 
				
			||||||
 | 
					_avformat.av_read_frame.argtypes = [
 | 
				
			||||||
 | 
					    AVFormatContext_p, # context
 | 
				
			||||||
 | 
					    AVPacket_p] # packet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					def read_frame(context, packet):
 | 
				
			||||||
 | 
					    return _avformat.av_read_frame(context, packet)
 | 
				
			||||||
							
								
								
									
										25
									
								
								mp4/packet.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								mp4/packet.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,25 @@
 | 
				
			|||||||
 | 
					# RozK
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from . import libav
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Packet:
 | 
				
			||||||
 | 
					    __slots__ = '_ref'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self):
 | 
				
			||||||
 | 
					        self._ref = libav.packet_alloc()
 | 
				
			||||||
 | 
					        if not self._ref:
 | 
				
			||||||
 | 
					            raise MemoryError
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __del__(self):
 | 
				
			||||||
 | 
					        if self._ref:
 | 
				
			||||||
 | 
					            libav.packet_free(self._ref)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def _as_parameter_(self):
 | 
				
			||||||
 | 
					        return self._ref
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def stream_index(self):
 | 
				
			||||||
 | 
					        if self._ref:
 | 
				
			||||||
 | 
					            return self._ref.contents.stream_index
 | 
				
			||||||
 | 
					        return -1
 | 
				
			||||||
							
								
								
									
										25
									
								
								mp4/stream.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								mp4/stream.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,25 @@
 | 
				
			|||||||
 | 
					# RozK
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from .codec import NullCodec
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class NullStream:
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def index(self):
 | 
				
			||||||
 | 
					        return -1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    @property
 | 
				
			||||||
 | 
					    def codec(self):
 | 
				
			||||||
 | 
					        return NullCodec()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def contains(self, packet):
 | 
				
			||||||
 | 
					        return False
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Stream:
 | 
				
			||||||
 | 
					    __slots__ = 'index', 'codec'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def __init__(self, index, codec):
 | 
				
			||||||
 | 
					        self.index = index
 | 
				
			||||||
 | 
					        self.codec = codec
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def contains(self, packet):
 | 
				
			||||||
 | 
					        return (self.index == packet.stream_index)
 | 
				
			||||||
							
								
								
									
										21
									
								
								pve.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								pve.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,21 @@
 | 
				
			|||||||
 | 
					# RozK
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from mp4.demuxer import Demuxer
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					demuxer = Demuxer('test.mp4')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					print(demuxer.video_stream.codec.name)
 | 
				
			||||||
 | 
					print(demuxer.audio_stream.codec.name)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					while True:
 | 
				
			||||||
 | 
					    packet = demuxer.read_packet()
 | 
				
			||||||
 | 
					    if packet is None:
 | 
				
			||||||
 | 
					        break
 | 
				
			||||||
 | 
					    if demuxer.video_stream.contains(packet):
 | 
				
			||||||
 | 
					        continue
 | 
				
			||||||
 | 
					    elif demuxer.audio_stream.contains(packet):
 | 
				
			||||||
 | 
					        continue
 | 
				
			||||||
 | 
					    else:
 | 
				
			||||||
 | 
					        print("unkown packet")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					demuxer.close()
 | 
				
			||||||
		Reference in New Issue
	
	Block a user