]> git.sesse.net Git - vlc/blob - src/audio_output/aout_common.h
* Mandatory step for video output IV and the audio output quality
[vlc] / src / audio_output / aout_common.h
1 /*****************************************************************************
2  * aout_common.h: audio output inner functions
3  *****************************************************************************
4  * Copyright (C) 1999, 2000, 2001 VideoLAN
5  * $Id: aout_common.h,v 1.4 2001/05/01 04:18:18 sam Exp $
6  *
7  * Authors: Michel Kaempf <maxx@via.ecp.fr>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  * 
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
22  *****************************************************************************/
23
24 /* Creating as many aout_Thread functions as configurations was one solution,
25  * examining the different cases in the Thread loop of an unique function was
26  * another. I chose the first solution. */
27 void aout_U8MonoThread        ( aout_thread_t * p_aout );
28 void aout_U8StereoThread      ( aout_thread_t * p_aout );
29 void aout_S8MonoThread        ( aout_thread_t * p_aout );
30 void aout_S8StereoThread      ( aout_thread_t * p_aout );
31 void aout_U16MonoThread       ( aout_thread_t * p_aout );
32 void aout_U16StereoThread     ( aout_thread_t * p_aout );
33 void aout_S16MonoThread       ( aout_thread_t * p_aout );
34 void aout_S16StereoThread     ( aout_thread_t * p_aout );
35 void aout_SpdifThread         ( aout_thread_t * p_aout );
36
37 #define UPDATE_INCREMENT( increment, integer ) \
38     if ( ((increment).l_remainder += (increment).l_euclidean_remainder) >= 0 )\
39     { \
40         (integer) += (increment).l_euclidean_integer + 1; \
41         (increment).l_remainder -= (increment).l_euclidean_denominator; \
42     } \
43     else \
44     { \
45         (integer) += (increment).l_euclidean_integer; \
46     }
47
48 /*****************************************************************************
49  * InitializeIncrement
50  *****************************************************************************/
51 static __inline__ void InitializeIncrement( aout_increment_t * p_increment,
52                                             long l_numerator,
53                                             long l_denominator )
54 {
55     p_increment->l_remainder = -l_denominator;
56
57     p_increment->l_euclidean_integer = 0;
58     while ( l_numerator >= l_denominator )
59     {
60         p_increment->l_euclidean_integer++;
61         l_numerator -= l_denominator;
62     }
63
64     p_increment->l_euclidean_remainder = l_numerator;
65
66     p_increment->l_euclidean_denominator = l_denominator;
67 }
68
69 /*****************************************************************************
70  * NextFrame
71  *****************************************************************************/
72 static __inline__ int NextFrame( aout_thread_t * p_aout, aout_fifo_t * p_fifo,
73                                  mtime_t aout_date )
74 {
75     long l_units, l_rate;
76
77     /* We take the lock */
78     vlc_mutex_lock( &p_fifo->data_lock );
79
80     /* Are we looking for a dated start frame ? */
81     if ( !p_fifo->b_start_frame )
82     {
83         while ( p_fifo->l_start_frame != p_fifo->l_end_frame )
84         {
85             if ( p_fifo->date[p_fifo->l_start_frame] != LAST_MDATE )
86             {
87                 p_fifo->b_start_frame = 1;
88                 p_fifo->l_next_frame = (p_fifo->l_start_frame + 1) & AOUT_FIFO_SIZE;
89                 p_fifo->l_unit = p_fifo->l_start_frame * (p_fifo->l_frame_size >> (p_fifo->b_stereo));
90                 break;
91             }
92             p_fifo->l_start_frame = (p_fifo->l_start_frame + 1) & AOUT_FIFO_SIZE;
93         }
94
95         if ( p_fifo->l_start_frame == p_fifo->l_end_frame )
96         {
97             vlc_mutex_unlock( &p_fifo->data_lock );
98             return( -1 );
99         }
100     }
101
102     /* We are looking for the next dated frame */
103     /* FIXME : is the output fifo full ?? */
104     while ( !p_fifo->b_next_frame )
105     {
106         while ( p_fifo->l_next_frame != p_fifo->l_end_frame )
107         {
108             if ( p_fifo->date[p_fifo->l_next_frame] != LAST_MDATE )
109             {
110                 p_fifo->b_next_frame = 1;
111                 break;
112             }
113             p_fifo->l_next_frame = (p_fifo->l_next_frame + 1) & AOUT_FIFO_SIZE;
114         }
115
116         while ( p_fifo->l_next_frame == p_fifo->l_end_frame )
117         {
118             vlc_cond_wait( &p_fifo->data_wait, &p_fifo->data_lock );
119             if ( p_fifo->b_die )
120             {
121                 vlc_mutex_unlock( &p_fifo->data_lock );
122                 return( -1 );
123             }
124         }
125     }
126
127     l_units = ((p_fifo->l_next_frame - p_fifo->l_start_frame) & AOUT_FIFO_SIZE) * (p_fifo->l_frame_size >> (p_fifo->b_stereo));
128
129     l_rate = p_fifo->l_rate + ((aout_date - p_fifo->date[p_fifo->l_start_frame]) / 256);
130     intf_DbgMsg( "aout debug: %lli (%li);", aout_date - p_fifo->date[p_fifo->l_start_frame], l_rate );
131
132     InitializeIncrement( &p_fifo->unit_increment, l_rate, p_aout->l_rate );
133
134     p_fifo->l_units = (((l_units - (p_fifo->l_unit -
135         (p_fifo->l_start_frame * (p_fifo->l_frame_size >> (p_fifo->b_stereo)))))
136         * p_aout->l_rate) / l_rate) + 1;
137
138     /* We release the lock before leaving */
139     vlc_mutex_unlock( &p_fifo->data_lock );
140     return( 0 );
141 }
142