1 /*****************************************************************************
2 * spu_decoder.c : spu decoder thread
3 *****************************************************************************
4 * Copyright (C) 2000 VideoLAN
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA.
21 *****************************************************************************/
23 /*****************************************************************************
25 *****************************************************************************/
28 #include <stdlib.h> /* malloc(), free() */
29 #include <sys/types.h> /* on BSD, uio.h needs types.h */
30 #include <sys/uio.h> /* for input.h */
31 #include <unistd.h> /* getpid() */
40 #include "debug.h" /* ASSERT */
42 #include "stream_control.h"
43 #include "input_ext-dec.h"
46 #include "video_output.h"
48 #include "spu_decoder.h"
53 static int InitThread ( spudec_thread_t *p_spudec );
54 static void RunThread ( spudec_thread_t *p_spudec );
55 static void ErrorThread ( spudec_thread_t *p_spudec );
56 static void EndThread ( spudec_thread_t *p_spudec );
58 #define GetWord( i ) \
59 i = GetByte( &p_spudec->bit_stream ) << 8; \
60 i += GetByte( &p_spudec->bit_stream ); \
63 /*****************************************************************************
64 * spudec_CreateThread: create a spu decoder thread
65 *****************************************************************************/
66 vlc_thread_t spudec_CreateThread( vdec_config_t * p_config )
68 spudec_thread_t * p_spudec;
70 intf_DbgMsg("spudec debug: creating spu decoder thread");
72 /* Allocate the memory needed to store the thread's structure */
73 if ( (p_spudec = (spudec_thread_t *)malloc( sizeof(spudec_thread_t) )) == NULL )
75 intf_ErrMsg("spudec error: not enough memory for spudec_CreateThread() to create the new thread");
80 * Initialize the thread properties
82 p_spudec->p_fifo->b_die = 0;
83 p_spudec->p_fifo->b_error = 0;
84 p_spudec->p_config = p_config;
85 p_spudec->p_fifo = p_config->decoder_config.p_decoder_fifo;
87 /* Get the video output informations */
88 p_spudec->p_vout = p_config->p_vout;
90 /* Spawn the spu decoder thread */
91 if ( vlc_thread_create(&p_spudec->thread_id, "spu decoder",
92 (vlc_thread_func_t)RunThread, (void *)p_spudec) )
94 intf_ErrMsg("spudec error: can't spawn spu decoder thread");
99 intf_DbgMsg("spudec debug: spu decoder thread (%p) created", p_spudec);
100 return( p_spudec->thread_id );
103 /* following functions are local */
105 /*****************************************************************************
106 * InitThread: initialize spu decoder thread
107 *****************************************************************************
108 * This function is called from RunThread and performs the second step of the
109 * initialization. It returns 0 on success. Note that the thread's flag are not
110 * modified inside this function.
111 *****************************************************************************/
112 static int InitThread( spudec_thread_t *p_spudec )
114 intf_DbgMsg("spudec debug: initializing spu decoder thread %p", p_spudec);
116 p_spudec->p_config->decoder_config.pf_init_bit_stream( &p_spudec->bit_stream,
117 p_spudec->p_config->decoder_config.p_decoder_fifo );
119 /* Mark thread as running and return */
120 intf_DbgMsg( "spudec debug: InitThread(%p) succeeded", p_spudec );
124 /*****************************************************************************
125 * RunThread: spu decoder thread
126 *****************************************************************************
127 * spu decoder thread. This function only returns when the thread is
129 *****************************************************************************/
130 static void RunThread( spudec_thread_t *p_spudec )
132 intf_DbgMsg("spudec debug: running spu decoder thread (%p) (pid == %i)",
136 * Initialize thread and free configuration
138 p_spudec->p_fifo->b_error = InitThread( p_spudec );
141 * Main loop - it is not executed if an error occured during
144 while( (!p_spudec->p_fifo->b_die) && (!p_spudec->p_fifo->b_error) )
151 boolean_t b_finished;
152 unsigned char * p_spu_data;
153 subpicture_t * p_spu = NULL;
155 while( !DECODER_FIFO_ISEMPTY(*p_spudec->p_fifo) )
157 /* wait for the next SPU ID.
158 * XXX: We trash 0xff bytes since they probably come from
159 * an incomplete previous packet */
162 i_packet_size = GetByte( &p_spudec->bit_stream );
164 while( i_packet_size == 0xff );
166 if( p_spudec->p_fifo->b_die )
169 /* the total size - should equal the sum of the
170 * PES packet size that form the SPU packet */
171 i_packet_size = ( i_packet_size << 8 )
172 + GetByte( &p_spudec->bit_stream );
175 /* get the useful PES size (real size - 10) */
176 i_pes_size = DECODER_FIFO_START(*p_spudec->p_fifo)->i_pes_size - 9;
179 /* the RLE stuff size */
180 GetWord( i_rle_size );
182 /* if the values we got aren't too strange, decode the data */
183 if( i_rle_size < i_packet_size )
185 /* allocate the subpicture.
186 * FIXME: we should check if the allocation failed */
187 p_spu = vout_CreateSubPicture( p_spudec->p_vout,
188 DVD_SUBPICTURE, i_rle_size );
189 p_spu_data = p_spu->p_data;
191 /* get display time */
192 p_spu->begin_date = p_spu->end_date
193 = DECODER_FIFO_START(*p_spudec->p_fifo)->i_pts;
195 /* getting the RLE part */
196 while( i_index++ < i_rle_size )
198 /* skip the leading byte of a PES */
199 /* FIXME: this part definitely looks strange */
200 if ( !((i_index + 3) % i_pes_size) )
204 *p_spu_data++ = GetByte( &p_spudec->bit_stream );
207 /* getting the control part */
218 /* next offset, no next offset if == i_index-5 */
220 b_finished = ( i_index - 5 >= i_word );
224 i_cmd = GetByte( &p_spudec->bit_stream );
229 case SPU_CMD_FORCE_DISPLAY:
230 /* 00 (force displaying) */
232 /* FIXME: here we have to calculate dates. It's
233 * around i_date * 12000 but I don't know
236 case SPU_CMD_START_DISPLAY:
237 /* 01 (start displaying) */
238 p_spu->begin_date += ( i_date * 12000 );
240 case SPU_CMD_STOP_DISPLAY:
241 /* 02 (stop displaying) */
242 p_spu->end_date += ( i_date * 12000 );
244 case SPU_CMD_SET_PALETTE:
245 /* 03xxxx (palette) */
248 case SPU_CMD_SET_ALPHACHANNEL:
249 /* 04xxxx (alpha channel) */
252 case SPU_CMD_SET_COORDINATES:
253 /* 05xxxyyyxxxyyy (coordinates) */
254 i_word = GetByte( &p_spudec->bit_stream );
255 p_spu->i_x = (i_word << 4)
256 | GetBits( &p_spudec->bit_stream, 4 );
258 i_word = GetBits( &p_spudec->bit_stream, 4 );
259 p_spu->i_width = p_spu->i_x - ( (i_word << 8)
260 | GetByte( &p_spudec->bit_stream ) ) + 1;
262 i_word = GetByte( &p_spudec->bit_stream );
263 p_spu->i_y = (i_word << 4)
264 | GetBits( &p_spudec->bit_stream, 4 );
266 i_word = GetBits( &p_spudec->bit_stream, 4 );
267 p_spu->i_height = p_spu->i_y - ( (i_word << 8)
268 | GetByte( &p_spudec->bit_stream ) ) + 1;
272 case SPU_CMD_SET_OFFSETS:
273 /* 06xxxxyyyy (byte offsets) */
275 p_spu->type.spu.i_offset[0] = i_word - 4;
277 p_spu->type.spu.i_offset[1] = i_word - 4;
283 /* ?? (unknown command) */
284 intf_ErrMsg( "spudec: unknown command 0x%.2x",
289 while( i_cmd != SPU_CMD_END );
291 while( !b_finished );
293 /* SPU is finished - we can tell the video output
295 vout_DisplaySubPicture( p_spudec->p_vout, p_spu );
299 /* Unexpected PES packet - trash it */
300 intf_ErrMsg( "spudec: trying to recover from bad packet" );
301 vlc_mutex_lock( &p_spudec->p_fifo->data_lock );
302 p_spudec->p_fifo->pf_delete_pes( p_spudec->p_fifo->p_packets_mgt,
303 DECODER_FIFO_START(*p_spudec->p_fifo) );
304 DECODER_FIFO_INCSTART( *p_spudec->p_fifo );
305 vlc_mutex_unlock( &p_spudec->p_fifo->data_lock );
314 if( p_spudec->p_fifo->b_error )
316 ErrorThread( p_spudec );
320 EndThread( p_spudec );
323 /*****************************************************************************
324 * ErrorThread: RunThread() error loop
325 *****************************************************************************
326 * This function is called when an error occured during thread main's loop. The
327 * thread can still receive feed, but must be ready to terminate as soon as
329 *****************************************************************************/
330 static void ErrorThread( spudec_thread_t *p_spudec )
332 /* We take the lock, because we are going to read/write the start/end
333 * indexes of the decoder fifo */
334 vlc_mutex_lock( &p_spudec->p_fifo->data_lock );
336 /* Wait until a `die' order is sent */
337 while( !p_spudec->p_fifo->b_die )
339 /* Trash all received PES packets */
340 while( !DECODER_FIFO_ISEMPTY(*p_spudec->p_fifo) )
342 p_spudec->p_fifo->pf_delete_pes( p_spudec->p_fifo->p_packets_mgt,
343 DECODER_FIFO_START(*p_spudec->p_fifo) );
344 DECODER_FIFO_INCSTART( *p_spudec->p_fifo );
347 /* Waiting for the input thread to put new PES packets in the fifo */
348 vlc_cond_wait( &p_spudec->p_fifo->data_wait, &p_spudec->p_fifo->data_lock );
351 /* We can release the lock before leaving */
352 vlc_mutex_unlock( &p_spudec->p_fifo->data_lock );
355 /*****************************************************************************
356 * EndThread: thread destruction
357 *****************************************************************************
358 * This function is called when the thread ends after a sucessful
360 *****************************************************************************/
361 static void EndThread( spudec_thread_t *p_spudec )
363 intf_DbgMsg( "spudec debug: destroying spu decoder thread %p", p_spudec );
364 free( p_spudec->p_config );
366 intf_DbgMsg( "spudec debug: spu decoder thread %p destroyed", p_spudec);