]> git.sesse.net Git - vlc/blob - src/input/input.c
. le d�codeur de sous-titres s'appelle maintenant spu_decoder
[vlc] / src / input / input.c
1 /*******************************************************************************
2  * input.c: input thread 
3  * (c)1998 VideoLAN
4  *******************************************************************************
5  * Read an MPEG2 stream, demultiplex and parse it before sending it to
6  * decoders.
7  ******************************************************************************/
8
9 /*******************************************************************************
10  * Preamble
11  *******************************************************************************/
12 #include "vlc.h"
13
14 /*
15 #include <errno.h>
16 #include <sys/uio.h>                                               
17 #include <string.h>
18
19 #include <stdlib.h>                            
20 #include <stdio.h>
21 #include <sys/ioctl.h>                                            
22 #include <net/if.h>                                                
23 #include <netinet/in.h>
24
25 #include "common.h"
26 #include "config.h"
27 #include "mtime.h"
28 #include "vlc_thread.h"
29 #include "intf_msg.h"
30 #include "debug.h"
31
32 #include "input.h"
33 #include "input_psi.h"
34 #include "input_pcr.h"
35 #include "input_netlist.h"
36 #include "decoder_fifo.h"
37 #include "input_file.h"
38 #include "input_network.h"
39
40 #include "audio_output.h"
41 #include "audio_decoder.h"
42
43 #include "video.h"
44 #include "video_output.h"
45 #include "video_decoder.h"
46 */
47
48 /******************************************************************************
49  * Local prototypes
50  ******************************************************************************/
51 static void RunThread   ( input_thread_t *p_input );
52 static void ErrorThread ( input_thread_t *p_input );
53 static void EndThread   ( input_thread_t *p_input );
54
55 static __inline__ int   input_ReadPacket( input_thread_t *p_input );
56 static __inline__ void  input_SortPacket( input_thread_t *p_input,
57                                           ts_packet_t *ts_packet );
58 static __inline__ void  input_DemuxTS( input_thread_t *p_input,
59                                        ts_packet_t *ts_packet,
60                                        es_descriptor_t *es_descriptor );
61 static __inline__ void  input_DemuxPES( input_thread_t *p_input,
62                                         ts_packet_t *ts_packet,
63                                         es_descriptor_t *p_es_descriptor,
64                                         boolean_t b_unit_start, boolean_t b_packet_lost );
65 static __inline__ void  input_DemuxPSI( input_thread_t *p_input,
66                                         ts_packet_t *ts_packet,
67                                         es_descriptor_t *p_es_descriptor,
68                                         boolean_t b_unit_start, boolean_t b_packet_lost );
69
70 /*******************************************************************************
71  * input_CreateThread: creates a new input thread
72  *******************************************************************************
73  * This function creates a new input, and returns a pointer
74  * to its description. On error, it returns NULL.
75  * If pi_status is NULL, then the function will block until the thread is ready.
76  * If not, it will be updated using one of the THREAD_* constants.
77  *******************************************************************************/
78 input_thread_t *input_CreateThread ( int i_method, char *psz_source, int i_port, int i_vlan,
79                                      p_vout_thread_t p_vout, p_aout_thread_t p_aout, int *pi_status )
80 {
81     input_thread_t *    p_input;                          /* thread descriptor */
82     int                 i_status;                             /* thread status */    
83     int                 i_index;            /* index for tables initialization */    
84     
85     /* Allocate descriptor */
86     intf_DbgMsg("\n");
87     p_input = (input_thread_t *)malloc( sizeof(input_thread_t) );
88     if( p_input == NULL )
89     {
90         intf_ErrMsg("error: %s\n", strerror(ENOMEM));
91         return( NULL );
92     }
93
94     /* Initialize thread properties */
95     p_input->b_die              = 0;
96     p_input->b_error            = 0;
97     p_input->pi_status          = (pi_status != NULL) ? pi_status : &i_status;
98     *p_input->pi_status         = THREAD_CREATE;
99
100     /* Initialize input method description */
101     p_input->i_method           = i_method;
102     p_input->psz_source         = psz_source;
103     p_input->i_port             = i_port;
104     p_input->i_vlan             = i_vlan;
105     switch( i_method )
106     {
107     case INPUT_METHOD_TS_FILE:                                 /* file methods */
108         p_input->p_Open =   input_FileOpen;
109         p_input->p_Read =   input_FileRead;
110         p_input->p_Close =  input_FileClose;
111         break;
112     case INPUT_METHOD_TS_VLAN_BCAST:                    /* vlan network method */        
113         if( !p_main->b_vlans )
114         {
115             intf_ErrMsg("error: vlans are not activated\n");
116             free( p_input );
117             return( NULL );            
118         }        
119         /* ... pass through */
120     case INPUT_METHOD_TS_UCAST:                             /* network methods */
121     case INPUT_METHOD_TS_MCAST:
122     case INPUT_METHOD_TS_BCAST:
123         p_input->p_Open =   input_NetworkOpen;
124         p_input->p_Read =   input_NetworkRead;
125         p_input->p_Close =  input_NetworkClose;
126         break;
127     default:
128         intf_ErrMsg("error: unknow input method\n");
129         free( p_input );
130         return( NULL );  
131         break;            
132     }
133
134     /* Initialize stream description */
135     for( i_index = 0; i_index < INPUT_MAX_ES; i_index++ )
136     {
137         p_input->p_es[i_index].i_id = EMPTY_PID;
138     }
139     
140     /* Initialize default settings for spawned decoders */
141     p_input->p_aout                     = p_aout;
142     p_input->p_vout                     = p_vout; 
143
144 #ifdef STATS
145     /* Initialize statistics */
146     p_input->c_loops                    = 0;    
147     p_input->c_bytes                    = 0;
148     p_input->c_payload_bytes            = 0;
149     p_input->c_packets_read             = 0;
150     p_input->c_packets_trashed          = 0;
151 #endif
152
153     /* Initialize PSI and PCR decoders */
154     if( input_PsiInit( p_input ) )
155     {
156         free( p_input );
157         return( NULL );
158     }
159
160     if( input_PcrInit( p_input ) )
161     {
162         input_PsiEnd( p_input );
163         free( p_input );
164         return( NULL );
165     }
166
167     /* Initialize netlists */
168     if( input_NetlistInit( p_input ) )
169     {
170         input_PsiEnd( p_input );
171         input_PcrEnd( p_input );
172         free( p_input );
173         return( NULL );
174     }
175
176     intf_DbgMsg("configuration: method=%d, source=%s, port=%d, vlan=%d\n", 
177                 i_method, psz_source, i_port, i_vlan );    
178
179     /* Let the appropriate method open the socket. */
180     if( p_input->p_Open( p_input ) )
181     {
182         input_NetlistEnd( p_input );
183         input_PsiEnd( p_input );
184         input_PcrEnd( p_input );
185         free( p_input );
186         return( NULL );
187     }
188
189     /* Create thread and set locks. */
190     vlc_mutex_init( &p_input->netlist.lock );
191     vlc_mutex_init( &p_input->programs_lock );
192     vlc_mutex_init( &p_input->es_lock );
193     if( vlc_thread_create(&p_input->thread_id, "input", (void *) RunThread, (void *) p_input) )
194     {
195         intf_ErrMsg("error: %s\n", strerror(errno) );
196         p_input->p_Close( p_input );
197         input_NetlistEnd( p_input );;
198         input_PsiEnd( p_input );
199         input_PcrEnd( p_input );
200         free( p_input );
201         return( NULL );
202     }
203  
204     intf_Msg("Input initialized\n");
205  
206     /* If status is NULL, wait until the thread is created */
207     if( pi_status == NULL )
208     {
209         do
210         {            
211             msleep( THREAD_SLEEP );
212         }while( (i_status != THREAD_READY) && (i_status != THREAD_ERROR) 
213                 && (i_status != THREAD_FATAL) );
214         if( i_status != THREAD_READY )
215         {
216             return( NULL );            
217         }        
218     }
219     return( p_input );
220 }
221
222 /******************************************************************************
223  * input_DestroyThread: mark an input thread as zombie
224  ******************************************************************************
225  * This function should not return until the thread is effectively cancelled.
226  ******************************************************************************/
227 void input_DestroyThread( input_thread_t *p_input, int *pi_status )
228 {
229     int         i_status;                                    /* thread status */
230
231     /* Set status */
232     p_input->pi_status = (pi_status != NULL) ? pi_status : &i_status;
233     *p_input->pi_status = THREAD_DESTROY;    
234      
235     /* Request thread destruction */
236     p_input->b_die = 1;
237
238     /* If status is NULL, wait until thread has been destroyed */
239     if( pi_status == NULL )
240     {
241         do
242         {
243             msleep( THREAD_SLEEP );
244         }while( (i_status != THREAD_OVER) && (i_status != THREAD_ERROR) 
245                 && (i_status != THREAD_FATAL) );   
246     }
247 }
248
249 #if 0
250 /*******************************************************************************
251  * input_OpenAudioStream: open an audio stream
252  *******************************************************************************
253  * This function spawns an audio decoder and plugs it on the audio output
254  * thread.
255  *******************************************************************************/
256 int input_OpenAudioStream( input_thread_t *p_input, int i_id )
257 {
258     /* ?? */
259 }
260
261 /*******************************************************************************
262  * input_CloseAudioStream: close an audio stream
263  *******************************************************************************
264  * This function destroys an audio decoder.
265  *******************************************************************************/
266 void input_CloseAudioStream( input_thread_t *p_input, int i_id )
267 {
268     /* ?? */
269 }
270
271 /*******************************************************************************
272  * input_OpenVideoStream: open a video stream
273  *******************************************************************************
274  * This function spawns a video decoder and plugs it on a video output thread.
275  *******************************************************************************/
276 int input_OpenVideoStream( input_thread_t *p_input, 
277                            struct vout_thread_s *p_vout, struct video_cfg_s * p_cfg )
278 {
279     /* ?? */
280 }
281
282 /*******************************************************************************
283  * input_CloseVideoStream: close a video stream
284  *******************************************************************************
285  * This function destroys an video decoder.
286  *******************************************************************************/
287 void input_CloseVideoStream( input_thread_t *p_input, int i_id )
288 {
289     /* ?? */
290 }
291 #endif
292
293 /* following functions are local */
294
295 /*******************************************************************************
296  * InitThread: initialize input thread
297  *******************************************************************************
298  * This function is called from RunThread and performs the second step of the
299  * initialization. It returns 0 on success. Note that the thread's flag are not
300  * modified inside this function.
301  *******************************************************************************/
302 static int InitThread( input_thread_t *p_input )
303 {
304     /* Mark thread as running and return */
305     intf_DbgMsg("\n");
306     *p_input->pi_status =        THREAD_READY;    
307     intf_DbgMsg("thread ready\n");    
308     return( 0 );    
309 }
310
311 /*******************************************************************************
312  * RunThread: main thread loop
313  *******************************************************************************
314  * Thread in charge of processing the network packets and demultiplexing.
315  *******************************************************************************/
316 static void RunThread( input_thread_t *p_input )
317 {
318     /* 
319      * Initialize thread and free configuration 
320      */
321     p_input->b_error = InitThread( p_input );
322     if( p_input->b_error )
323     {
324         free( p_input );                                 /* destroy descriptor */
325         return;        
326     }
327
328     /*
329      * Main loop
330      */
331     intf_DbgMsg("\n");
332     while( !p_input->b_die && !p_input->b_error )
333     {
334         /* Scatter read the UDP packet from the network or the file. */
335         if( (input_ReadPacket( p_input )) == (-1) )
336         {
337             /* ??? Normally, a thread can't kill itself, but we don't have
338              * any method in case of an error condition ... */
339             p_input->b_error = 1;
340         }
341
342 #ifdef STATS
343         p_input->c_loops++;
344 #endif
345     }
346
347     /*
348      * Error loop
349      */
350     if( p_input->b_error )
351     {
352         ErrorThread( p_input );
353     }
354
355     /* End of thread */
356     EndThread( p_input );
357     intf_DbgMsg("thread end\n");
358 }
359
360
361 /******************************************************************************
362  * ErrorThread: RunThread() error loop
363  *******************************************************************************
364  * This function is called when an error occured during thread main's loop. 
365  ******************************************************************************/
366 static void ErrorThread( input_thread_t *p_input )
367 {
368     /* Wait until a `die' order */
369     intf_DbgMsg("\n");
370     while( !p_input->b_die )
371     {
372         /* Sleep a while */
373         msleep( VOUT_IDLE_SLEEP );
374     }
375 }
376
377 /*******************************************************************************
378  * EndThread: end the input thread
379  *******************************************************************************/
380 static void EndThread( input_thread_t * p_input )
381 {
382     int *       pi_status;                                    /* threas status */
383     int         i_es_loop;                                         /* es index */
384
385     /* Store status */
386     intf_DbgMsg("\n");
387     pi_status = p_input->pi_status;    
388     *pi_status = THREAD_END;  
389
390     /* Close input method */
391     p_input->p_Close( p_input );
392
393     /* Destroy all decoder threads */
394     for( i_es_loop = 0; 
395          (i_es_loop < INPUT_MAX_ES) && (p_input->pp_selected_es[i_es_loop] != NULL) ; 
396          i_es_loop++ )
397     {
398         switch( p_input->pp_selected_es[i_es_loop]->i_type )
399         {
400         case MPEG1_VIDEO_ES:
401         case MPEG2_VIDEO_ES:
402 #ifdef OLD_DECODER
403             vdec_DestroyThread( (vdec_thread_t*)(p_input->pp_selected_es[i_es_loop]->p_dec) /*, NULL */ );
404 #else
405             vpar_DestroyThread( (vpar_thread_t*)(p_input->pp_selected_es[i_es_loop]->p_dec) /*, NULL */ );
406 #endif
407             break;            
408         case MPEG1_AUDIO_ES:
409         case MPEG2_AUDIO_ES:
410             adec_DestroyThread( (adec_thread_t*)(p_input->pp_selected_es[i_es_loop]->p_dec) );
411             break;
412         case AC3_AUDIO_ES:
413             ac3dec_DestroyThread( (ac3dec_thread_t *)(p_input->pp_selected_es[i_es_loop]->p_dec) );
414             break;
415         case DVD_SPU_ES:
416             fprintf(stderr, "input.h : destroying spudec\n");
417             spudec_DestroyThread( (spudec_thread_t *)(p_input->pp_selected_es[i_es_loop]->p_dec) );
418             break;
419         case 0:
420             /* Special streams for the PSI decoder, PID 0 and 1 */
421             break;
422 #ifdef DEBUG
423         default:
424             intf_DbgMsg("error: unknown decoder type %d\n", p_input->pp_selected_es[i_es_loop]->i_type );
425             break;                
426 #endif
427         }
428     }
429
430     input_NetlistEnd( p_input );                              /* clean netlist */
431     input_PsiEnd( p_input );                          /* clean PSI information */
432     input_PcrEnd( p_input );                          /* clean PCR information */
433     free( p_input );                            /* free input_thread structure */
434
435     /* Update status */
436     *pi_status = THREAD_OVER;    
437 }
438
439 /*******************************************************************************
440  * input_ReadPacket: reads a packet from the network or the file
441  *******************************************************************************/
442 static __inline__ int input_ReadPacket( input_thread_t *p_input )
443 {
444     int                 i_base_index; /* index of the first free iovec */
445     int                 i_current_index;
446     int                 i_packet_size;
447 #ifdef INPUT_LIFO_TS_NETLIST
448     int                 i_meanwhile_released;
449     int                 i_currently_removed;
450 #endif
451     ts_packet_t *       p_ts_packet;
452
453     /* In this function, we only care about the TS netlist. PES netlist
454      * is for the demultiplexer. */
455 #ifdef INPUT_LIFO_TS_NETLIST
456     i_base_index = p_input->netlist.i_ts_index;
457
458     /* Verify that we still have packets in the TS netlist */
459     if( (INPUT_MAX_TS + INPUT_TS_READ_ONCE - 1 - p_input->netlist.i_ts_index) <= INPUT_TS_READ_ONCE )
460     {
461         intf_ErrMsg("input error: TS netlist is empty !\n");
462         return( -1 );
463     }
464
465 #else /* FIFO netlist */
466     i_base_index = p_input->netlist.i_ts_start;
467     if( p_input->netlist.i_ts_start + INPUT_TS_READ_ONCE -1 > INPUT_MAX_TS )
468     {
469         /* The netlist is splitted in 2 parts. We must gather them to consolidate
470            the FIFO (we make the loop easily in having the same iovec at the far
471            end and in the beginning of netlist_free).
472            That's why the netlist is (INPUT_MAX_TS +1) + (INPUT_TS_READ_ONCE -1)
473            large. */
474         memcpy( p_input->netlist.p_ts_free + INPUT_MAX_TS + 1,
475                 p_input->netlist.p_ts_free,
476                 (p_input->netlist.i_ts_start + INPUT_TS_READ_ONCE - 1 - INPUT_MAX_TS)
477                   * sizeof(struct iovec) );
478     }
479
480     /* Verify that we still have packets in the TS netlist */
481     if( ((p_input->netlist.i_ts_end -1 - p_input->netlist.i_ts_start) & INPUT_MAX_TS) <= INPUT_TS_READ_ONCE )
482     {
483         intf_ErrMsg("input error: TS netlist is empty !\n");
484         return( -1 );
485     }
486 #endif /* FIFO netlist */
487
488     /* Scatter read the buffer. */
489     i_packet_size = (*p_input->p_Read)( p_input,
490                            &p_input->netlist.p_ts_free[i_base_index],
491                            INPUT_TS_READ_ONCE );
492     if( i_packet_size == (-1) )
493     {
494 //      intf_DbgMsg("Read packet %d %p %d %d\n", i_base_index,
495 //                      &p_input->netlist.p_ts_free[i_base_index],
496 //                      p_input->netlist.i_ts_start,
497 //                      p_input->netlist.i_ts_end);
498         intf_ErrMsg("input error: readv() failed (%s)\n", strerror(errno));
499         return( -1 );
500     }
501
502     if( i_packet_size == 0 )
503     {
504         /* No packet has been received, so stop here. */
505         return( 0 );
506     }
507      
508     /* Demultiplex the TS packets (1..INPUT_TS_READ_ONCE) received. */
509     for( i_current_index = i_base_index;
510          (i_packet_size -= TS_PACKET_SIZE) >= 0;
511          i_current_index++ )
512     {
513         /* BTW, something REALLY bad could happen if we receive packets with
514            a wrong size. */
515         p_ts_packet = (ts_packet_t*)(p_input->netlist.p_ts_free[i_current_index].iov_base);
516         /* Don't cry :-), we are allowed to do that cast, because initially,
517            our buffer was malloc'ed with sizeof(ts_packet_t) */
518
519         /* Find out if we need this packet and demultiplex. */
520         input_SortPacket( p_input /* for current PIDs and netlist */,
521                           p_ts_packet);
522     }
523
524     if( i_packet_size > 0 )
525     {
526         intf_ErrMsg("input error: wrong size\n");
527         return( -1 );
528     }
529
530     /* Remove the TS packets we have just filled from the netlist */
531 #ifdef INPUT_LIFO_TS_NETLIST
532     /* We need to take a lock here while we're calculating index positions. */
533     vlc_mutex_lock( &p_input->netlist.lock );
534
535     i_meanwhile_released = i_base_index - p_input->netlist.i_ts_index;
536     if( i_meanwhile_released )
537     {
538         /* That's where it becomes funny :-). Since we didn't take locks for
539            efficiency reasons, other threads (including ourselves, with
540            input_DemuxPacket) might have released packets to the netlist.
541            So we have to copy these iovec where they should go.
542            
543            BTW, that explains why the TS netlist is
544            (INPUT_MAX_TS +1) + (TS_READ_ONCE -1) large. */
545
546         i_currently_removed = i_current_index - i_base_index;
547         if( i_meanwhile_released < i_currently_removed )
548         {
549             /* Copy all iovecs in that case */
550             memcpy( &p_input->netlist.p_ts_free[p_input->netlist.i_ts_index]
551                      + i_currently_removed,
552                     &p_input->netlist.p_ts_free[p_input->netlist.i_ts_index],
553                     i_meanwhile_released * sizeof(struct iovec) );
554         }
555         else
556         {
557             /* We have fewer places than items, so we only move
558                i_currently_removed of them. */
559             memcpy( &p_input->netlist.p_ts_free[i_base_index],
560                     &p_input->netlist.p_ts_free[p_input->netlist.i_ts_index],
561                     i_currently_removed * sizeof(struct iovec) );
562         }
563
564         /* Update i_netlist_index with the information gathered above. */
565         p_input->netlist.i_ts_index += i_currently_removed;
566     }
567     else
568     {
569         /* Nothing happened. */
570         p_input->netlist.i_ts_index = i_current_index;
571     }
572
573     vlc_mutex_unlock( &p_input->netlist.lock );
574
575 #else /* FIFO netlist */
576     /* & is modulo ; that's where we make the loop. */
577     p_input->netlist.i_ts_start = i_current_index & INPUT_MAX_TS;
578 #endif
579
580 #ifdef STATS
581     p_input->c_packets_read += i_current_index - i_base_index;
582     p_input->c_bytes += (i_current_index - i_base_index) * TS_PACKET_SIZE;
583 #endif
584     return( 0 );
585 }
586
587 /*******************************************************************************
588  * input_SortPacket: find out whether we need that packet
589  *******************************************************************************/
590 static __inline__ void input_SortPacket( input_thread_t *p_input,
591                                          ts_packet_t *p_ts_packet )
592 {
593     int             i_current_pid;
594     int             i_es_loop;
595
596     /* Verify that sync_byte, error_indicator and scrambling_control are
597        what we expected. */
598     if( !(p_ts_packet->buffer[0] == 0x47) || (p_ts_packet->buffer[1] & 0x80) ||
599         (p_ts_packet->buffer[3] & 0xc0) )
600     {
601         intf_DbgMsg("input debug: invalid TS header (%p)\n", p_ts_packet);
602     }
603     else
604     {
605         /* Get the PID of the packet. Note that ntohs is needed, for endianness
606            purposes (see man page). */
607         i_current_pid = U16_AT(&p_ts_packet->buffer[1]) & 0x1fff;
608
609 //      intf_DbgMsg("input debug: pid %d received (%p)\n",
610 //                    i_current_pid, p_ts_packet);
611
612         /* Lock current ES state. */
613         vlc_mutex_lock( &p_input->es_lock );
614         
615         /* Verify that we actually want this PID. */
616         for( i_es_loop = 0; i_es_loop < INPUT_MAX_SELECTED_ES; i_es_loop++ )
617         {
618             if( p_input->pp_selected_es[i_es_loop] != NULL)
619             {
620                 if( (*p_input->pp_selected_es[i_es_loop]).i_id
621                      == i_current_pid )
622                 {
623                     /* Don't need the lock anymore, since the value pointed
624                        out by p_input->pp_selected_es[i_es_loop] can only be
625                        modified from inside the input_thread (by the PSI
626                        decoder): interface thread is only allowed to modify
627                        the pp_selected_es table */
628                     vlc_mutex_unlock( &p_input->es_lock );
629
630                     /* We're interested. Pass it to the demultiplexer. */
631                     input_DemuxTS( p_input, p_ts_packet,
632                                    p_input->pp_selected_es[i_es_loop] );
633                     return;
634                 }
635             }
636             else
637             {
638                 /* pp_selected_es should not contain any hole. */
639                 break;
640             }
641         }
642         vlc_mutex_unlock( &p_input->es_lock );
643     }
644
645     /* We weren't interested in receiving this packet. Give it back to the
646        netlist. */
647 //    intf_DbgMsg("SortPacket: freeing unwanted TS %p (pid %d)\n", p_ts_packet,
648 //                     U16_AT(&p_ts_packet->buffer[1]) & 0x1fff);
649     input_NetlistFreeTS( p_input, p_ts_packet );
650 #ifdef STATS
651     p_input->c_packets_trashed++;
652 #endif
653 }
654
655 /*******************************************************************************
656  * input_DemuxTS: first step of demultiplexing: the TS header
657  *******************************************************************************
658  * Stream must also only contain PES and PSI, so PID must have been filtered
659  *******************************************************************************/
660 static __inline__ void input_DemuxTS( input_thread_t *p_input,
661                                       ts_packet_t *p_ts_packet,
662                                       es_descriptor_t *p_es_descriptor )
663 {
664     int         i_dummy;
665     boolean_t   b_adaption;                       /* Adaption field is present */
666     boolean_t   b_payload;                           /* Packet carries payload */
667     boolean_t   b_unit_start;            /* A PSI or a PES start in the packet */
668     boolean_t   b_trash = 0;                   /* Must the packet be trashed ? */
669     boolean_t   b_lost = 0;                      /* Was there a packet lost ? */
670
671     ASSERT(p_input);
672     ASSERT(p_ts_packet);
673     ASSERT(p_es_descriptor);
674
675 #define p (p_ts_packet->buffer)
676
677 //    intf_DbgMsg("input debug: TS-demultiplexing packet %p, pid %d, number %d\n",
678 //                p_ts_packet, U16_AT(&p[1]) & 0x1fff, p[3] & 0x0f);
679
680 #ifdef STATS
681     p_es_descriptor->c_packets++;
682     p_es_descriptor->c_bytes += TS_PACKET_SIZE;
683 #endif
684
685     /* Extract flags values from TS common header. */
686     b_unit_start = (p[1] & 0x40);
687     b_adaption = (p[3] & 0x20);
688     b_payload = (p[3] & 0x10);
689     
690     /* Extract adaption field informations if any */
691     if( !b_adaption )
692     {
693         /* We don't have any adaptation_field, so payload start immediately
694          after the 4 byte TS header */
695         p_ts_packet->i_payload_start = 4;
696     }
697     else
698     {
699         /* p[4] is adaptation_field_length minus one */
700         p_ts_packet->i_payload_start = 5 + p[4];
701
702         /* The adaption field can be limited to the adaptation_field_length byte,
703            so that there is nothing to do: skip this possibility */
704         if( p[4] )
705         {
706             /* If the packet has both adaptation_field and payload, adaptation_field
707                cannot be more than 182 bytes long; if there is only an
708                adaptation_field, it must fill the next 183 bytes. */
709             if( b_payload ? (p[4] > 182) : (p[4] != 183) )
710             {
711                 intf_DbgMsg("input debug: invalid TS adaptation field (%p)\n",
712                             p_ts_packet);
713 #ifdef STATS
714                 p_es_descriptor->c_invalid_packets++;
715 #endif
716                 b_trash = 1;
717             }
718
719             /* No we are sure that the byte containing flags is present: read it */
720             else
721             {
722                 /* discontinuity_indicator */
723                 if( p[5] & 0x80 )
724                 {
725                     intf_DbgMsg("discontinuity_indicator encountered by TS demux " \
726                                 "(position read: %d, saved: %d)\n", p[5] & 0x80,
727                                 p_es_descriptor->i_continuity_counter);
728
729                     /* If the PID carries the PCR, there will be a system time-base
730                        discontinuity. We let the PCR decoder handle that. */
731                     p_es_descriptor->b_discontinuity = 1;
732                     
733                     /* There also may be a continuity_counter discontinuity:
734                        resynchronise our counter with the one of the stream */
735                     p_es_descriptor->i_continuity_counter = (p[3] & 0x0f) - 1;
736                 }
737
738                 /* random_access_indicator */
739                 p_es_descriptor->b_random |= p[5] & 0x40;
740
741                 /* If this is a PCR_PID, and this TS packet contains a PCR,
742                    we pass it along to the PCR decoder. */
743                 if( (p_es_descriptor->b_pcr) && (p[5] & 0x10) )
744                 {
745                     /* There should be a PCR field in the packet, check if the
746                        adaption field is long enough to carry it */
747                     if( p[4] >= 7 )
748                     {
749                         /* Call the PCR decoder */
750                         input_PcrDecode( p_input, p_es_descriptor, &p[6] );
751                     }
752                 }
753             }
754         }
755     }
756
757     /* Check the continuity of the stream. */
758     i_dummy = ((p[3] & 0x0f) - p_es_descriptor->i_continuity_counter) & 0x0f;
759     if( i_dummy == 1 )
760     {
761         /* Everything is ok, just increase our counter */
762         p_es_descriptor->i_continuity_counter++;
763     }
764     else
765     {
766         if( !b_payload && i_dummy == 0 )
767         {
768             /* This is a packet without payload, this is allowed by the draft
769                As there is nothing interessant in this packet (except PCR that
770                have already been handled), we can trash the packet. */
771             intf_DbgMsg("Packet without payload received by TS demux\n");
772             b_trash = 1;
773         }
774         else if( i_dummy <= 0 )
775         {
776             /* Duplicate packet: mark it as being to be trashed. */
777             intf_DbgMsg("Duplicate packet received by TS demux\n");
778             b_trash = 1;
779         }
780         else if( p_es_descriptor->i_continuity_counter == 0xFF )
781         {
782             /* This means that the packet is the first one we receive for this
783                ES since the continuity counter ranges between 0 and 0x0F
784                excepts when it has been initialized by the input: Init the 
785                counter to the correct value. */
786             intf_DbgMsg("First packet for PID %d received by TS demux\n",
787                         p_es_descriptor->i_id);
788             p_es_descriptor->i_continuity_counter = (p[3] & 0x0f);
789         }
790         else
791         {
792             /* This can indicate that we missed a packet or that the
793                continuity_counter wrapped and we received a dup packet: as we
794                don't know, do as if we missed a packet to be sure to recover
795                from this situation */
796             intf_DbgMsg("Packet lost by TS demux: current %d, packet %d\n",
797                         p_es_descriptor->i_continuity_counter & 0x0f,
798                         p[3] & 0x0f);
799             b_lost = 1;
800             p_es_descriptor->i_continuity_counter = p[3] & 0x0f;
801         }
802     }
803
804     /* Trash the packet if it has no payload or if it is bad */
805     if( b_trash )
806     {
807         input_NetlistFreeTS( p_input, p_ts_packet );
808 #ifdef STATS
809         p_input->c_packets_trashed++;
810 #endif
811     }
812     else
813     {
814         if( p_es_descriptor->b_psi )
815         {
816             /* The payload contains PSI tables */
817             input_DemuxPSI( p_input, p_ts_packet, p_es_descriptor,
818                             b_unit_start, b_lost );
819         }
820         else
821         {
822             /* The payload carries a PES stream */ 
823             input_DemuxPES( p_input, p_ts_packet, p_es_descriptor,
824                             b_unit_start, b_lost );
825         }
826     }
827
828 #undef p
829 }
830
831
832
833
834 /*******************************************************************************
835  * input_DemuxPES: 
836  *******************************************************************************
837  * Gather a PES packet and analyzes its header.
838  *******************************************************************************/
839 static __inline__ void input_DemuxPES( input_thread_t *p_input,
840                                        ts_packet_t *p_ts_packet,
841                                        es_descriptor_t *p_es_descriptor,
842                                        boolean_t b_unit_start,
843                                        boolean_t b_packet_lost )
844 {
845     decoder_fifo_t *            p_fifo;
846     u8                          i_pes_header_size;
847     int                         i_dummy;
848     pes_packet_t*               p_last_pes;
849     ts_packet_t *               p_ts;
850     int                         i_ts_payload_size;
851     
852
853 #define p_pes (p_es_descriptor->p_pes_packet)
854
855     ASSERT(p_input);
856     ASSERT(p_ts_packet);
857     ASSERT(p_es_descriptor);
858
859 //    intf_DbgMsg("PES-demultiplexing %p (%p)\n", p_ts_packet, p_pes);
860
861     /* If we lost data, discard the PES packet we are trying to reassemble
862        if any and wait for the beginning of a new one in order to synchronise
863        again */
864     if( b_packet_lost && p_pes != NULL )
865     {
866         intf_DbgMsg("PES %p trashed because of packet lost\n", p_pes);
867         input_NetlistFreePES( p_input, p_pes );
868         p_pes = NULL;
869     }
870
871     /* If the TS packet contains the begining of a new PES packet, and if we
872        were reassembling a PES packet, then the PES should be complete now,
873        so parse its header and give it to the decoders */
874     if( b_unit_start && p_pes != NULL )
875     {
876 //        intf_DbgMsg("End of PES packet %p\n", p_pes);
877
878         /* Parse the header. The header has a variable length, but in order 
879            to improve the algorithm, we will read the 14 bytes we may be
880            interested in */
881         p_ts = p_pes->p_first_ts;
882         i_ts_payload_size = p_ts->i_payload_end - p_ts->i_payload_start;
883         i_dummy = 0;
884
885         if(i_ts_payload_size >= PES_HEADER_SIZE)
886         {
887             /* This part of the header entirely fits in the payload of   
888                the first TS packet */
889             p_pes->p_pes_header = &(p_ts->buffer[p_ts->i_payload_start]);
890         }
891         else
892         {
893             /* This part of the header does not fit in the current TS packet:
894                copy the part of the header we are interested in to the
895                p_pes_header_save buffer. The buffer is dynamicly allocated if
896                needed so it's time expensive but this situation almost never
897                occur. */
898             intf_DbgMsg("Code never tested encountered, WARNING ! (benny)\n");
899             if( !p_pes->p_pes_header_save )
900                 p_pes->p_pes_header_save = malloc(PES_HEADER_SIZE); 
901
902             do
903             {
904                 memcpy(p_pes->p_pes_header_save + i_dummy,
905                        &p_ts->buffer[p_ts->i_payload_start], i_ts_payload_size);
906                 i_dummy += i_ts_payload_size;
907             
908                 p_ts = p_ts->p_next_ts;
909                 if(!p_ts)
910                 {
911                   /* The payload of the PES packet is shorter than the 14 bytes
912                      we would read. This means that high packet lost occured
913                      so the PES won't be usefull for any decoder. Moreover,
914                      this should never happen so we can trash the packet and
915                      exit roughly without regrets */
916                   intf_DbgMsg("PES packet too short: trashed\n");
917                   input_NetlistFreePES( p_input, p_pes );
918                   p_pes = NULL;
919                   /* Stats ?? */
920                   return;
921                 }
922                 
923                 i_ts_payload_size = p_ts->i_payload_end - p_ts->i_payload_start;
924             }
925             while(i_ts_payload_size + i_dummy < PES_HEADER_SIZE);
926
927             /* This last TS packet is partly header, partly payload, so just
928                copy the header part */
929             memcpy(p_pes->p_pes_header_save + i_dummy,
930                    &p_ts->buffer[p_ts->i_payload_start],
931                    PES_HEADER_SIZE - i_dummy);
932
933             /* The header must be read in the buffer not in any TS packet */
934            p_pes->p_pes_header = p_pes->p_pes_header_save;
935         }
936         
937         /* Now we have the part of the PES header we were interested in:
938            parse it */
939
940         /* First read the 6 header bytes common to all PES packets:
941            use them to test the PES validity */
942         if( (p_pes->p_pes_header[0] || p_pes->p_pes_header[1] ||
943             (p_pes->p_pes_header[2] != 1)) ||
944                                      /* packet_start_code_prefix != 0x000001 */
945             ((i_dummy = U16_AT(p_pes->p_pes_header + 4)) &&
946              (i_dummy + 6 != p_pes->i_pes_size)) )
947                    /* PES_packet_length is set and != total received payload */
948         {
949           /* Trash the packet and set p_pes to NULL to be sure the next PES
950              packet will have its b_data_lost flag set */
951           intf_DbgMsg("Corrupted PES packet received: trashed\n");
952           input_NetlistFreePES( p_input, p_pes );
953           p_pes = NULL;
954           /* Stats ?? */
955         }
956         else
957         {
958             /* The PES packet is valid. Check its type to test if it may
959                carry additional informations in a header extension */
960             p_pes->i_stream_id =  p_pes->p_pes_header[3];
961
962             switch( p_pes->i_stream_id )
963             {
964             case 0xBE:  /* Padding */
965             case 0xBC:  /* Program stream map */
966             case 0xBF:  /* Private stream 2 */
967             case 0xB0:  /* ECM */
968             case 0xB1:  /* EMM */
969             case 0xFF:  /* Program stream directory */
970             case 0xF2:  /* DSMCC stream */
971             case 0xF8:  /* ITU-T H.222.1 type E stream */
972                 /* The payload begins immediatly after the 6 bytes header, so
973                    we have finished with the parsing */
974                 i_pes_header_size = 6;
975                 break;
976
977             default:
978                 /* The PES header contains at least 3 more bytes: parse them */
979                 p_pes->b_data_alignment = p_pes->p_pes_header[6] & 0x04;
980                 p_pes->b_has_pts = p_pes->p_pes_header[7] & 0x80;
981                 i_pes_header_size = p_pes->p_pes_header[8] + 9;
982
983                 /* Now parse the optional header extensions (in the limit of
984                    the 14 bytes */
985                 if( p_pes->b_has_pts )
986                 {
987                     pcr_descriptor_t * p_pcr;
988
989                     p_pcr = p_input->p_pcr;
990                     
991                     p_pes->i_pts = 
992                         ( ((mtime_t)(p_pes->p_pes_header[9] & 0x0E) << 29) |
993                           (((mtime_t)U16_AT(p_pes->p_pes_header + 10) << 14) - (1 << 14)) |
994                           ((mtime_t)U16_AT(p_pes->p_pes_header + 12) >> 1) ) * 300;
995                     p_pes->i_pts /= 27;
996                     
997                     if( p_pcr->i_synchro_state )
998                     {
999                         switch( p_pcr->i_synchro_state )
1000                         {
1001                             case SYNCHRO_NOT_STARTED:
1002                                 p_pes->b_has_pts = 0;
1003                                 break;
1004
1005                             case SYNCHRO_START:
1006                                 p_pes->i_pts += p_pcr->delta_pcr;
1007                                 p_pcr->delta_absolute = mdate() - p_pes->i_pts + INPUT_PTS_DELAY;
1008                                 p_pes->i_pts += p_pcr->delta_absolute;
1009                                 p_pcr->i_synchro_state = 0;
1010                                 break;
1011
1012                             case SYNCHRO_REINIT: /* We skip a PES */
1013                                 p_pes->b_has_pts = 0;
1014                                 p_pcr->i_synchro_state = SYNCHRO_START;
1015                                 break;
1016                         }
1017                     }
1018                     else
1019                     {
1020                         p_pes->i_pts += p_pcr->delta_pcr + p_pcr->delta_absolute;
1021                     }
1022                 }
1023                 break;
1024             }
1025
1026             /* Now we've parsed the header, we just have to indicate in some
1027                specific TS packets where the PES payload begins (renumber
1028                i_payload_start), so that the decoders can find the beginning
1029                of their data right out of the box. */
1030             p_ts = p_pes->p_first_ts;
1031             i_ts_payload_size = p_ts->i_payload_end - p_ts->i_payload_start;
1032             while( i_pes_header_size > i_ts_payload_size )
1033             {
1034                 /* These packets are entirely filled by the PES header. */
1035                 i_pes_header_size -= i_ts_payload_size;
1036                 p_ts->i_payload_start = p_ts->i_payload_end;
1037                 /* Go to the next TS packet: here we won't have to test it is
1038                    not NULL because we trash the PES packets when packet lost
1039                    occurs */
1040                 p_ts = p_ts->p_next_ts;
1041                 i_ts_payload_size = p_ts->i_payload_end - p_ts->i_payload_start;
1042             }
1043             /* This last packet is partly header, partly payload. */
1044             p_ts->i_payload_start += i_pes_header_size;
1045
1046             
1047             /* Now we can eventually put the PES packet in the decoder's
1048                PES fifo */
1049             switch( p_es_descriptor->i_type )
1050             {
1051                 case MPEG1_VIDEO_ES:
1052                 case MPEG2_VIDEO_ES:
1053 #ifdef OLD_DECODER
1054                     p_fifo = &(((vdec_thread_t*)(p_es_descriptor->p_dec))->fifo);
1055 #else
1056                     p_fifo = &(((vpar_thread_t*)(p_es_descriptor->p_dec))->fifo);
1057 #endif
1058                     break;
1059
1060                 case MPEG1_AUDIO_ES:
1061                 case MPEG2_AUDIO_ES:
1062                     p_fifo = &(((adec_thread_t*)(p_es_descriptor->p_dec))->fifo);
1063                     break;
1064
1065                 case AC3_AUDIO_ES:
1066                     /* we skip 4 bytes at the beginning of the AC3 payload */
1067                     p_ts->i_payload_start += 4;
1068                     p_fifo = &(((ac3dec_thread_t *)(p_es_descriptor->p_dec))->fifo);
1069                     break;
1070
1071                 case DVD_SPU_ES:
1072                     /* we skip 4 bytes at the beginning of the subpicture payload */
1073                     p_ts->i_payload_start += 4;
1074                     fprintf(stderr, "input.h : launching spudec\n");
1075                     p_fifo = &(((spudec_thread_t *)(p_es_descriptor->p_dec))->fifo);
1076                     break;
1077
1078                 default:
1079                     /* This should never happen */
1080                     intf_DbgMsg("Unknown stream type (%d, %d): PES trashed\n",
1081                         p_es_descriptor->i_id, p_es_descriptor->i_type);
1082                     p_fifo = NULL;
1083                     break;
1084             }
1085
1086             if( p_fifo != NULL )
1087             {
1088                 vlc_mutex_lock( &p_fifo->data_lock );
1089                 if( DECODER_FIFO_ISFULL( *p_fifo ) )
1090                 {
1091                     /* The FIFO is full !!! This should not happen. */
1092 #ifdef STATS
1093                     p_input->c_packets_trashed += p_pes->i_ts_packets;
1094                     p_es_descriptor->c_invalid_packets += p_pes->i_ts_packets;
1095 #endif
1096                     input_NetlistFreePES( p_input, p_pes );
1097                     intf_DbgMsg("PES trashed - fifo full ! (%d, %d)\n",
1098                                p_es_descriptor->i_id, p_es_descriptor->i_type);
1099                 }
1100                 else
1101                 {
1102 //                    intf_DbgMsg("Putting %p into fifo %p/%d\n",
1103 //                                p_pes, p_fifo, p_fifo->i_end);
1104                     p_fifo->buffer[p_fifo->i_end] = p_pes;
1105                     DECODER_FIFO_INCEND( *p_fifo );
1106
1107                     /* Warn the decoder that it's got work to do. */
1108                     vlc_cond_signal( &p_fifo->data_wait );
1109                 }
1110                 vlc_mutex_unlock( &p_fifo->data_lock );
1111             }
1112             else
1113             {
1114                 intf_DbgMsg("No fifo to receive PES %p: trash\n", p_pes);
1115 #ifdef STATS
1116                 p_input->c_packets_trashed += p_pes->i_ts_packets;
1117                 p_es_descriptor->c_invalid_packets += p_pes->i_ts_packets;
1118 #endif
1119                 input_NetlistFreePES( p_input, p_pes );
1120             }
1121         }
1122     }
1123
1124
1125     /* If we are at the beginning of a new PES packet, we must fetch a new
1126        PES buffer to begin with the reassembly of this PES packet. This is
1127        also here that we can synchronise with the stream if we we lost
1128        packets or if the decoder has just started */ 
1129     if( b_unit_start )
1130     {
1131         p_last_pes = p_pes;
1132
1133         /* Get a new one PES from the PES netlist. */
1134         if( (p_pes = input_NetlistGetPES( p_input )) == (NULL) )
1135         {
1136             /* PES netlist is empty ! */
1137             p_input->b_error = 1;
1138         }
1139         else
1140         {
1141 //           intf_DbgMsg("New PES packet %p (first TS: %p)\n", p_pes, p_ts_packet);
1142
1143             /* Init the PES fields so that the first TS packet could be correctly
1144                added to the PES packet (see below) */
1145             p_pes->p_first_ts = p_ts_packet;
1146             p_pes->p_last_ts = NULL;
1147
1148             /* If the last pes packet was null, this means that the synchronisation
1149                was lost and so warn the decoder that he will have to find a way to
1150                recover */
1151             if( !p_last_pes )
1152                 p_pes->b_data_loss = 1;
1153
1154             /* Read the b_random_access flag status and then reinit it */     
1155             p_pes->b_random_access = p_es_descriptor->b_random;
1156             p_es_descriptor->b_random = 0;
1157         }
1158     }
1159
1160
1161     /* If we are synchronised with the stream, and so if we are ready to
1162        receive correctly the data, add the TS packet to the current PES
1163        packet */
1164     if( p_pes != NULL )
1165     {
1166 //      intf_DbgMsg("Adding TS %p to PES %p\n", p_ts_packet, p_pes);
1167
1168         /* Size of the payload carried in the TS packet */
1169         i_ts_payload_size = p_ts_packet->i_payload_end -
1170                             p_ts_packet->i_payload_start;
1171
1172         /* Update the relations between the TS packets */
1173         p_ts_packet->p_prev_ts = p_pes->p_last_ts;
1174         p_ts_packet->p_next_ts = NULL;
1175         if( p_pes->i_ts_packets != 0 )
1176         {
1177             /* Regarder si il serait pas plus efficace de ne creer que les liens
1178                precedent->suivant pour le moment, et les liens suivant->precedent
1179                quand le paquet est termine */
1180             /* Otherwise it is the first TS packet. */
1181             p_pes->p_last_ts->p_next_ts = p_ts_packet;
1182         }
1183         /* Now add the TS to the PES packet */
1184         p_pes->p_last_ts = p_ts_packet;
1185         p_pes->i_ts_packets++;
1186         p_pes->i_pes_size += i_ts_payload_size;
1187
1188         /* Stats */
1189 #ifdef STATS
1190         i_dummy = p_ts_packet->i_payload_end - p_ts_packet->i_payload_start;
1191         p_es_descriptor->c_payload_bytes += i_dummy;
1192 #endif
1193     }
1194     else
1195     {
1196         /* Since we don't use the TS packet to build a PES packet, we don't
1197            need it anymore, so give it back to the netlist */
1198 //        intf_DbgMsg("Trashing TS %p: no PES being build\n", p_ts_packet);
1199         input_NetlistFreeTS( p_input, p_ts_packet );     
1200     }
1201     
1202 #undef p_pes
1203 }
1204
1205
1206
1207
1208 /*******************************************************************************
1209  * input_DemuxPSI:
1210  *******************************************************************************
1211  * Notice that current ES state has been locked by input_SortPacket. (No more true,
1212  * changed by benny - See if it's ok, and definitely change the code ???????? )
1213  *******************************************************************************/
1214 static __inline__ void input_DemuxPSI( input_thread_t *p_input,
1215                                        ts_packet_t *p_ts_packet,
1216                                        es_descriptor_t *p_es_descriptor,
1217                                        boolean_t b_unit_start, boolean_t b_packet_lost )
1218 {
1219     int i_data_offset;      /* Offset of the interesting data in the TS packet */
1220     u16 i_data_length;                                 /* Length of those data */
1221     //boolean_t b_first_section; /* Was there another section in the TS packet ? */
1222     
1223     ASSERT(p_input);
1224     ASSERT(p_ts_packet);
1225     ASSERT(p_es_descriptor);
1226
1227 #define p_psi (p_es_descriptor->p_psi_section)
1228
1229 //    intf_DbgMsg( "input debug: PSI demultiplexing %p (%p)\n", p_ts_packet, p_input);
1230
1231 //    intf_DbgMsg( "Packet: %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x %x (unit start: %d)\n", p_ts_packet->buffer[p_ts_packet->i_payload_start], p_ts_packet->buffer[p_ts_packet->i_payload_start+1], p_ts_packet->buffer[p_ts_packet->i_payload_start+2], p_ts_packet->buffer[p_ts_packet->i_payload_start+3], p_ts_packet->buffer[p_ts_packet->i_payload_start+4], p_ts_packet->buffer[p_ts_packet->i_payload_start+5], p_ts_packet->buffer[p_ts_packet->i_payload_start+6], p_ts_packet->buffer[p_ts_packet->i_payload_start+7], p_ts_packet->buffer[p_ts_packet->i_payload_start+8], p_ts_packet->buffer[p_ts_packet->i_payload_start+9], p_ts_packet->buffer[p_ts_packet->i_payload_start+10], p_ts_packet->buffer[p_ts_packet->i_payload_start+11], p_ts_packet->buffer[p_ts_packet->i_payload_start+12], p_ts_packet->buffer[p_ts_packet->i_payload_start+13], p_ts_packet->buffer[p_ts_packet->i_payload_start+14], p_ts_packet->buffer[p_ts_packet->i_payload_start+15], p_ts_packet->buffer[p_ts_packet->i_payload_start+16], p_ts_packet->buffer[p_ts_packet->i_payload_start+17], p_ts_packet->buffer[p_ts_packet->i_payload_start+18], p_ts_packet->buffer[p_ts_packet->i_payload_start+19], p_ts_packet->buffer[p_ts_packet->i_payload_start+20], b_unit_start);
1232
1233
1234     /* Try to find the beginning of the payload in the packet to initialise
1235        the do-while loop that follows -> Compute the i_data_offset variable:
1236        by default, the value is set so that we won't enter in the while loop.
1237        It will be set to a correct value if the data are not corrupted */
1238     i_data_offset = TS_PACKET_SIZE;
1239
1240     /* Has the reassembly of a section already began in a previous packet ? */
1241     if( p_psi->b_running_section )
1242     {
1243         /* Was data lost since the last TS packet ? */
1244         if( b_packet_lost )
1245         {
1246             /* Discard the packet and wait for the begining of a new one to resynch */
1247             p_psi->b_running_section = 0;
1248             p_psi->i_current_position = 0;
1249             intf_DbgMsg( "PSI section(s) discarded due to packet loss\n" );
1250         }
1251         else
1252         {
1253             /* The data that complete a previously began section are always at
1254                the beginning of the TS payload... */
1255             i_data_offset = p_ts_packet->i_payload_start;
1256             /* ...Unless there is a pointer field, that we have to bypass */
1257             if( b_unit_start )
1258                 i_data_offset++;
1259 //            intf_DbgMsg( "New part of the section received at offset %d\n", i_data_offset );
1260         }
1261     }
1262     /* We are looking for the beginning of a new section */
1263     else
1264     {
1265         if( b_unit_start )
1266         {
1267             /* Get the offset at which the data for that section can be found
1268                The offset is stored in the pointer_field since we are interested in
1269                the first section of the TS packet. Note that the +1 is to bypass
1270                the pointer field */
1271             i_data_offset = p_ts_packet->i_payload_start +
1272                             p_ts_packet->buffer[p_ts_packet->i_payload_start] + 1;
1273 //            intf_DbgMsg( "New section beginning at offset %d in TS packet\n", i_data_offset );
1274         }
1275         else
1276         {
1277             /* This may either mean that the TS is bad or that the packet contains
1278                the end of a section that had been discarded in a previous loop: 
1279                trash the TS packet since we cannot do anything with those data: */
1280             p_psi->b_running_section = 0;
1281             p_psi->i_current_position = 0;
1282             intf_DbgMsg( "PSI packet discarded due to lack of synchronisation\n" );
1283         }
1284     }
1285
1286     /* The section we will deal with during the first iteration of the following
1287        loop is the first one contained in the TS packet */
1288     //    b_first_section = 1;
1289
1290     /* Reassemble the pieces of sections contained in the TS packet and decode
1291        the sections that could have been completed.
1292        Stop when we reach the end of the packet or stuffing bytes */
1293     while( i_data_offset < TS_PACKET_SIZE && p_ts_packet->buffer[i_data_offset] != 0xFF )
1294     {
1295         /* If the current section is a new one, reinit the data fields of the p_psi
1296            struct to start its decoding */
1297         if( !p_psi->b_running_section )
1298         {
1299             /* Read the length of the new section */
1300             p_psi->i_length = (U16_AT(&p_ts_packet->buffer[i_data_offset+1]) & 0xFFF) + 3;
1301 //            intf_DbgMsg( "Section length %d\n", p_psi->i_length );
1302             if( p_psi->i_length > PSI_SECTION_SIZE )
1303             {
1304                 /* The TS packet is corrupted, stop here to avoid possible a seg fault */
1305                 intf_DbgMsg( "PSI Section size is too big, aborting its reception\n" );
1306                 break;
1307             }
1308
1309             /* Init the reassembly of that section */
1310             p_psi->b_running_section = 1;
1311             p_psi->i_current_position = 0;
1312         }
1313         
1314         /* Compute the length of data related to the section in this TS packet */
1315         if( p_psi->i_length - p_psi->i_current_position > TS_PACKET_SIZE - i_data_offset)
1316             i_data_length = TS_PACKET_SIZE - i_data_offset;
1317         else
1318           i_data_length = p_psi->i_length - p_psi->i_current_position;
1319
1320         /* Copy those data in the section buffer */
1321         memcpy( &p_psi->buffer[p_psi->i_current_position], &p_ts_packet->buffer[i_data_offset],
1322                 i_data_length );
1323
1324         /* Interesting data are now after the ones we copied, since no gap is
1325            allowed between 2 sections in a TS packets */
1326         i_data_offset += i_data_length;
1327
1328         /* Decode the packet if it is now complete */
1329         if (p_psi->i_length == p_psi->i_current_position + i_data_length)
1330         {
1331             /* Packet is complete, decode it */
1332 //            intf_DbgMsg( "SECTION COMPLETE: starting decoding of its data\n" );
1333             input_PsiDecode( p_input, p_psi );
1334
1335             /* Prepare the buffer to receive a new section */
1336             p_psi->i_current_position = 0;
1337             p_psi->b_running_section = 0;
1338         
1339             /* The new section won't be the first anymore */
1340             //b_first_section = 0;
1341         }
1342         else
1343         {
1344             /* Prepare the buffer to receive the next part of the section */
1345           p_psi->i_current_position += i_data_length;
1346 //          intf_DbgMsg( "Section not complete, waiting for the end\n" );
1347         }
1348     
1349 //        intf_DbgMsg( "Must loop ? Next data offset: %d, stuffing: %d\n",
1350 //                     i_data_offset, p_ts_packet->buffer[i_data_offset] );
1351     }
1352
1353     /* Relase the TS packet, we don't need it anymore */
1354     input_NetlistFreeTS( p_input, p_ts_packet );
1355
1356 #undef p_psi  
1357 }