1 /*****************************************************************************
3 *****************************************************************************
4 * Copyright (C) 2001, 2002 VideoLAN
5 * $Id: mpeg4video.c,v 1.1 2002/12/14 21:32:41 fenrir Exp $
7 * Authors: Laurent Aimar <fenrir@via.ecp.fr>
8 * Eric Petit <titer@videolan.org>
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA.
23 *****************************************************************************/
25 /*****************************************************************************
27 *****************************************************************************/
30 #include <vlc/decoder.h>
31 #include <vlc/input.h>
34 #include <stdlib.h> /* malloc(), free() */
35 #include <string.h> /* strdup() */
37 /*****************************************************************************
39 *****************************************************************************/
40 typedef struct packetizer_thread_s
42 /* Input properties */
43 decoder_fifo_t *p_fifo;
44 bit_stream_t bit_stream;
48 /* Output properties */
49 sout_input_t *p_sout_input;
50 sout_packet_format_t output_format;
55 } packetizer_thread_t;
57 static int Open ( vlc_object_t * );
58 static int Run ( decoder_fifo_t * );
60 static int InitThread ( packetizer_thread_t * );
61 static void PacketizeThread ( packetizer_thread_t * );
62 static void EndThread ( packetizer_thread_t * );
64 /*****************************************************************************
66 *****************************************************************************/
69 set_description( _("MPEG-4 packetizer") );
70 set_capability( "packetizer", 50 );
71 set_callbacks( Open, NULL );
74 #define VIDEO_OBJECT_MASK 0x01f
75 #define VIDEO_OBJECT_LAYER_MASK 0x00f
77 #define VIDEO_OBJECT_START_CODE 0x100
78 #define VIDEO_OBJECT_LAYER_START_CODE 0x120
79 #define VISUAL_OBJECT_SEQUENCE_START_CODE 0x1b0
80 #define VISUAL_OBJECT_SEQUENCE_END_CODE 0x1b1
81 #define USER_DATA_START_CODE 0x1b2
82 #define GROUP_OF_VOP_START_CODE 0x1b3
83 #define VIDEO_SESSION_ERROR_CODE 0x1b4
84 #define VISUAL_OBJECT_START_CODE 0x1b5
85 #define VOP_START_CODE 0x1b6
86 #define FACE_OBJECT_START_CODE 0x1ba
87 #define FACE_OBJECT_PLANE_START_CODE 0x1bb
88 #define MESH_OBJECT_START_CODE 0x1bc
89 #define MESH_OBJECT_PLANE_START_CODE 0x1bd
90 #define STILL_TEXTURE_OBJECT_START_CODE 0x1be
91 #define TEXTURE_SPATIAL_LAYER_START_CODE 0x1bf
92 #define TEXTURE_SNR_LAYER_START_CODE 0x1c0
94 /*****************************************************************************
95 * OpenDecoder: probe the packetizer and return score
96 *****************************************************************************
97 * Tries to launch a decoder and return score so that the interface is able
99 *****************************************************************************/
100 static int Open( vlc_object_t *p_this )
102 decoder_fifo_t *p_fifo = (decoder_fifo_t*) p_this;
104 p_fifo->pf_run = Run;
106 switch( p_fifo->i_fourcc )
108 case VLC_FOURCC( 'm', 'p', '4', 'v'):
109 case VLC_FOURCC( 'D', 'I', 'V', 'X'):
110 case VLC_FOURCC( 'd', 'i', 'v', 'x'):
111 case VLC_FOURCC( 'X', 'V', 'I', 'D'):
112 case VLC_FOURCC( 'X', 'v', 'i', 'D'):
113 case VLC_FOURCC( 'x', 'v', 'i', 'd'):
114 case VLC_FOURCC( 'D', 'X', '5', '0'):
121 /*****************************************************************************
122 * RunDecoder: this function is called just after the thread is created
123 *****************************************************************************/
124 static int Run( decoder_fifo_t *p_fifo )
126 packetizer_thread_t *p_pack;
129 msg_Info( p_fifo, "Running MPEG-4 packetizer" );
130 if( !( p_pack = malloc( sizeof( packetizer_thread_t ) ) ) )
132 msg_Err( p_fifo, "out of memory" );
133 DecoderError( p_fifo );
136 memset( p_pack, 0, sizeof( packetizer_thread_t ) );
138 p_pack->p_fifo = p_fifo;
140 if( InitThread( p_pack ) != 0 )
142 DecoderError( p_fifo );
146 while( ( !p_pack->p_fifo->b_die )&&( !p_pack->p_fifo->b_error ) )
148 PacketizeThread( p_pack );
152 if( ( b_error = p_pack->p_fifo->b_error ) )
154 DecoderError( p_pack->p_fifo );
167 #define FREE( p ) if( p ) free( p ); p = NULL
169 static int CopyUntilNextStartCode( packetizer_thread_t *p_pack,
170 sout_buffer_t *p_sout_buffer,
177 p_sout_buffer->p_buffer[(*pi_pos)++] =
178 GetBits( &p_pack->bit_stream, 8 );
181 if( *pi_pos + 2048 > p_sout_buffer->i_allocated_size )
183 sout_BufferRealloc( p_pack->p_sout_input->p_sout,
185 p_sout_buffer->i_allocated_size + 50 * 1024);
188 } while( ShowBits( &p_pack->bit_stream, 24 ) != 0x01 &&
189 !p_pack->p_fifo->b_die && !p_pack->p_fifo->b_error );
194 static int sout_BufferAddMem( sout_instance_t *p_sout,
195 sout_buffer_t *p_buffer,
199 if( p_buffer->i_size + i_mem >= p_buffer->i_allocated_size )
201 sout_BufferRealloc( p_sout,
203 p_buffer->i_size + i_mem + 1024 );
205 memcpy( p_buffer->p_buffer + p_buffer->i_size, p_mem, i_mem );
206 p_buffer->i_size += i_mem;
211 /*****************************************************************************
212 * InitThread: initialize data before entering main loop
213 *****************************************************************************/
215 static int InitThread( packetizer_thread_t *p_pack )
218 p_pack->p_vol = NULL;
219 p_pack->i_vop_since_vol = 0;
220 p_pack->output_format.i_cat = VIDEO_ES;
221 p_pack->output_format.i_fourcc = VLC_FOURCC( 'm', 'p', '4', 'v' );
223 if( InitBitstream( &p_pack->bit_stream, p_pack->p_fifo,
224 NULL, NULL ) != VLC_SUCCESS )
226 msg_Err( p_pack->p_fifo, "cannot initialize bitstream" );
230 p_pack->p_sout_input =
231 sout_InputNew( p_pack->p_fifo,
232 &p_pack->output_format );
234 if( !p_pack->p_sout_input )
236 msg_Err( p_pack->p_fifo,
237 "cannot add a new stream" );
244 /*****************************************************************************
245 * PacketizeThread: packetize an unit (here copy a complete pes)
246 *****************************************************************************/
247 static void PacketizeThread( packetizer_thread_t *p_pack )
249 sout_instance_t *p_sout = p_pack->p_sout_input->p_sout;
250 sout_buffer_t *p_frame;
252 uint32_t i_startcode;
254 /* Idea: Copy until a vop has been found
255 * Once a videoobject & videoobjectlayer has been found we save it
258 p_frame = sout_BufferNew( p_sout, 20*1024 ); // FIXME
263 while( ( ( i_startcode = ShowBits( &p_pack->bit_stream, 32 ) )&0xffffff00 ) != 0x00000100 )
265 RemoveBits( &p_pack->bit_stream, 8 );
268 if( i_startcode == VISUAL_OBJECT_SEQUENCE_START_CODE )
270 msg_Dbg( p_pack->p_fifo, "<visuel_object_sequence>" );
271 RemoveBits32( &p_pack->bit_stream );
273 else if( i_startcode == VISUAL_OBJECT_SEQUENCE_END_CODE )
275 msg_Dbg( p_pack->p_fifo, "</visuel_object_sequence>" );
276 RemoveBits32( &p_pack->bit_stream );
280 msg_Dbg( p_pack->p_fifo, "start code:0x%8.8x", i_startcode );
282 if( ( i_startcode & ~VIDEO_OBJECT_MASK ) == VIDEO_OBJECT_START_CODE )
284 msg_Dbg( p_pack->p_fifo, "<video_object>" );
285 CopyUntilNextStartCode( p_pack, p_frame, &p_frame->i_size );
287 else if( ( i_startcode & ~VIDEO_OBJECT_LAYER_MASK ) == VIDEO_OBJECT_LAYER_START_CODE )
290 if( p_pack->p_vol == NULL )
292 p_pack->p_vol = sout_BufferNew( p_sout, 1024 );
294 p_pack->p_vol->i_size = 0;
295 CopyUntilNextStartCode( p_pack, p_pack->p_vol, &p_pack->p_vol->i_size );
296 p_pack->i_vop_since_vol = 0;
298 /* then: add it to p_frame */
299 sout_BufferAddMem( p_sout, p_frame,
300 p_pack->p_vol->i_size,
301 p_pack->p_vol->p_buffer );
303 else if( i_startcode == GROUP_OF_VOP_START_CODE )
305 msg_Dbg( p_pack->p_fifo, "<group_of_vop>" );
306 if( p_pack->p_vol && p_pack->i_vop_since_vol > 100 ) // FIXME
308 sout_BufferAddMem( p_sout, p_frame,
309 p_pack->p_vol->i_size,
310 p_pack->p_vol->p_buffer );
311 p_pack->i_vop_since_vol = 0;
313 CopyUntilNextStartCode( p_pack, p_frame, &p_frame->i_size );
315 else if( i_startcode == VOP_START_CODE )
317 msg_Dbg( p_pack->p_fifo, "<vop>" );
318 if( p_pack->p_vol && p_pack->i_vop_since_vol > 100 ) // FIXME
320 sout_BufferAddMem( p_sout, p_frame,
321 p_pack->p_vol->i_size,
322 p_pack->p_vol->p_buffer );
323 p_pack->i_vop_since_vol = 0;
325 CopyUntilNextStartCode( p_pack, p_frame, &p_frame->i_size );
326 p_pack->i_vop_since_vol++;
331 msg_Dbg( p_pack->p_fifo, "unknown start code" );
332 CopyUntilNextStartCode( p_pack, p_frame, &p_frame->i_size );
338 p_frame->i_length = 1000000 / 25;
339 p_frame->i_bitrate= 0;
340 p_frame->i_dts = p_pack->i_dts;
341 p_frame->i_pts = p_pack->i_dts;
343 p_pack->i_dts += 1000000 / 25;
344 sout_InputSendBuffer( p_pack->p_sout_input, p_frame );
348 /*****************************************************************************
349 * EndThread : packetizer thread destruction
350 *****************************************************************************/
351 static void EndThread ( packetizer_thread_t *p_pack)
353 if( p_pack->p_sout_input )
355 sout_InputDelete( p_pack->p_sout_input );