Mercurial > mplayer.hg
annotate libmpcodecs/ad_qtaudio.c @ 12828:f9f8e49ff3d5
Cinepak and RoqA/V are now in ffmpeg
author | rtognimp |
---|---|
date | Thu, 15 Jul 2004 20:44:39 +0000 |
parents | 6a61d694f7d3 |
children | 462408dae3e7 |
rev | line source |
---|---|
8008 | 1 #include <stdio.h> |
2 #include <stdlib.h> | |
3 #include <unistd.h> | |
4 #include <inttypes.h> | |
5 | |
6 #include "config.h" | |
12193 | 7 #include "../mp_msg.h" |
8008 | 8 |
9502
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
9 #if defined(USE_QTX_CODECS) || defined(MACOSX) |
12356
6a61d694f7d3
minimal fix for alex's 1000000000000l compile errors. imo the fix in
rfelker
parents:
12193
diff
changeset
|
10 #include "wine/windef.h" |
8008 | 11 |
12 #include "ad_internal.h" | |
13 #include "bswap.h" | |
14 | |
9405
a4444e7ee56a
real cygwin support by Sascha Sommer <saschasommer@freenet.de>
alex
parents:
8632
diff
changeset
|
15 #ifdef WIN32_LOADER |
8451 | 16 #include "ldt_keeper.h" |
17 #endif | |
18 | |
9502
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
19 #ifdef MACOSX |
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
20 #include <QuickTime/QuickTimeComponents.h> |
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
21 #endif |
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
22 |
8008 | 23 static ad_info_t info = { |
24 "QuickTime Audio Decoder", | |
25 "qtaudio", | |
26 "A'rpi", | |
27 "Sascha Sommer", | |
28 "uses win32 quicktime DLLs" | |
29 }; | |
30 | |
31 LIBAD_EXTERN(qtaudio) | |
32 | |
9502
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
33 #ifdef USE_QTX_CODECS |
8008 | 34 typedef struct OpaqueSoundConverter* SoundConverter; |
35 typedef unsigned long OSType; | |
36 typedef unsigned long UnsignedFixed; | |
37 typedef uint8_t Byte; | |
38 typedef struct SoundComponentData { | |
39 long flags; | |
40 OSType format; | |
41 short numChannels; | |
42 short sampleSize; | |
43 UnsignedFixed sampleRate; | |
44 long sampleCount; | |
45 Byte * buffer; | |
46 long reserved; | |
47 }SoundComponentData; | |
48 | |
49 typedef int (__cdecl* LPFUNC1)(long flag); | |
50 typedef int (__cdecl* LPFUNC2)(const SoundComponentData *, const SoundComponentData *,SoundConverter *); | |
51 typedef int (__cdecl* LPFUNC3)(SoundConverter sc); | |
52 typedef int (__cdecl* LPFUNC4)(void); | |
53 typedef int (__cdecl* LPFUNC5)(SoundConverter sc, OSType selector,void * infoPtr); | |
54 typedef int (__cdecl* LPFUNC6)(SoundConverter sc, | |
55 unsigned long inputBytesTarget, | |
56 unsigned long *inputFrames, | |
57 unsigned long *inputBytes, | |
58 unsigned long *outputBytes ); | |
59 typedef int (__cdecl* LPFUNC7)(SoundConverter sc, | |
60 const void *inputPtr, | |
61 unsigned long inputFrames, | |
62 void *outputPtr, | |
63 unsigned long *outputFrames, | |
64 unsigned long *outputBytes ); | |
65 typedef int (__cdecl* LPFUNC8)(SoundConverter sc, | |
66 void *outputPtr, | |
67 unsigned long *outputFrames, | |
68 unsigned long *outputBytes); | |
69 typedef int (__cdecl* LPFUNC9)(SoundConverter sc) ; | |
70 | |
71 static HINSTANCE qtml_dll; | |
72 static LPFUNC1 InitializeQTML; | |
73 static LPFUNC2 SoundConverterOpen; | |
74 static LPFUNC3 SoundConverterClose; | |
75 static LPFUNC4 TerminateQTML; | |
76 static LPFUNC5 SoundConverterSetInfo; | |
77 static LPFUNC6 SoundConverterGetBufferSizes; | |
78 static LPFUNC7 SoundConverterConvertBuffer; | |
79 static LPFUNC8 SoundConverterEndConversion; | |
80 static LPFUNC9 SoundConverterBeginConversion; | |
81 | |
82 #define siDecompressionParams 2002876005 // siDecompressionParams = FOUR_CHAR_CODE('wave') | |
83 | |
84 HMODULE WINAPI LoadLibraryA(LPCSTR); | |
85 FARPROC WINAPI GetProcAddress(HMODULE,LPCSTR); | |
86 int WINAPI FreeLibrary(HMODULE); | |
87 | |
88 static int loader_init() | |
89 { | |
90 | |
9405
a4444e7ee56a
real cygwin support by Sascha Sommer <saschasommer@freenet.de>
alex
parents:
8632
diff
changeset
|
91 #ifdef WIN32_LOADER |
8270 | 92 Setup_LDT_Keeper(); |
93 #endif | |
8008 | 94 qtml_dll = LoadLibraryA("qtmlClient.dll"); |
95 if( qtml_dll == NULL ) | |
96 { | |
12193 | 97 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed loading dll\n" ); |
8008 | 98 return 1; |
99 } | |
100 #if 1 | |
101 InitializeQTML = (LPFUNC1)GetProcAddress(qtml_dll,"InitializeQTML"); | |
102 if ( InitializeQTML == NULL ) | |
103 { | |
12193 | 104 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed geting proc address InitializeQTML\n"); |
8008 | 105 return 1; |
106 } | |
107 SoundConverterOpen = (LPFUNC2)GetProcAddress(qtml_dll,"SoundConverterOpen"); | |
108 if ( SoundConverterOpen == NULL ) | |
109 { | |
12193 | 110 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed getting proc address SoundConverterOpen\n"); |
8008 | 111 return 1; |
112 } | |
113 SoundConverterClose = (LPFUNC3)GetProcAddress(qtml_dll,"SoundConverterClose"); | |
114 if ( SoundConverterClose == NULL ) | |
115 { | |
12193 | 116 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed getting proc address SoundConverterClose\n"); |
8008 | 117 return 1; |
118 } | |
119 TerminateQTML = (LPFUNC4)GetProcAddress(qtml_dll,"TerminateQTML"); | |
120 if ( TerminateQTML == NULL ) | |
121 { | |
12193 | 122 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed getting proc address TerminateQTML\n"); |
8008 | 123 return 1; |
124 } | |
125 SoundConverterSetInfo = (LPFUNC5)GetProcAddress(qtml_dll,"SoundConverterSetInfo"); | |
126 if ( SoundConverterSetInfo == NULL ) | |
127 { | |
12193 | 128 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed getting proc address SoundConverterSetInfo\n"); |
8008 | 129 return 1; |
130 } | |
131 SoundConverterGetBufferSizes = (LPFUNC6)GetProcAddress(qtml_dll,"SoundConverterGetBufferSizes"); | |
132 if ( SoundConverterGetBufferSizes == NULL ) | |
133 { | |
12193 | 134 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed getting proc address SoundConverterGetBufferSizes\n"); |
8008 | 135 return 1; |
136 } | |
137 SoundConverterConvertBuffer = (LPFUNC7)GetProcAddress(qtml_dll,"SoundConverterConvertBuffer"); | |
138 if ( SoundConverterConvertBuffer == NULL ) | |
139 { | |
12193 | 140 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed getting proc address SoundConverterConvertBuffer1\n"); |
8008 | 141 return 1; |
142 } | |
143 SoundConverterEndConversion = (LPFUNC8)GetProcAddress(qtml_dll,"SoundConverterEndConversion"); | |
144 if ( SoundConverterEndConversion == NULL ) | |
145 { | |
12193 | 146 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed getting proc address SoundConverterEndConversion\n"); |
8008 | 147 return 1; |
148 } | |
149 SoundConverterBeginConversion = (LPFUNC9)GetProcAddress(qtml_dll,"SoundConverterBeginConversion"); | |
150 if ( SoundConverterBeginConversion == NULL ) | |
151 { | |
12193 | 152 mp_msg(MSGT_DECAUDIO,MSGL_ERR,"failed getting proc address SoundConverterBeginConversion\n"); |
8008 | 153 return 1; |
154 } | |
155 #endif | |
12193 | 156 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"loader_init DONE???\n"); |
8008 | 157 return 0; |
158 } | |
9502
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
159 #endif /* USE_QTX_CODECS */ |
8008 | 160 |
161 static SoundConverter myConverter = NULL; | |
162 static SoundComponentData InputFormatInfo,OutputFormatInfo; | |
163 | |
164 static int InFrameSize; | |
165 static int OutFrameSize; | |
166 | |
167 static int preinit(sh_audio_t *sh){ | |
168 int error; | |
169 unsigned long FramesToGet=0; //how many frames the demuxer has to get | |
170 unsigned long InputBufferSize=0; //size of the input buffer | |
171 unsigned long OutputBufferSize=0; //size of the output buffer | |
172 unsigned long WantedBufferSize=0; //the size you want your buffers to be | |
173 | |
174 | |
9502
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
175 #ifdef MACOSX |
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
176 EnterMovies(); |
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
177 #else |
8008 | 178 if(loader_init()) return 0; // failed to load DLL |
179 | |
12193 | 180 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"loader_init DONE!\n"); |
8008 | 181 |
8159 | 182 error = InitializeQTML(6+16); |
12193 | 183 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"InitializeQTML:%i\n",error); |
8008 | 184 if(error) return 0; |
9694 | 185 #endif |
8008 | 186 |
9694 | 187 #if 1 |
8008 | 188 OutputFormatInfo.flags = InputFormatInfo.flags = 0; |
189 OutputFormatInfo.sampleCount = InputFormatInfo.sampleCount = 0; | |
190 OutputFormatInfo.buffer = InputFormatInfo.buffer = NULL; | |
191 OutputFormatInfo.reserved = InputFormatInfo.reserved = 0; | |
192 OutputFormatInfo.numChannels = InputFormatInfo.numChannels = sh->wf->nChannels; | |
8159 | 193 InputFormatInfo.sampleSize = sh->wf->wBitsPerSample; |
194 OutputFormatInfo.sampleSize = 16; | |
8008 | 195 OutputFormatInfo.sampleRate = InputFormatInfo.sampleRate = sh->wf->nSamplesPerSec; |
196 InputFormatInfo.format = bswap_32(sh->format); //1363430706;///*1768775988;//*/1902406962;//qdm2//1768775988;//FOUR_CHAR_CODE('ima4'); | |
197 OutputFormatInfo.format = 1313820229;// FOUR_CHAR_CODE('NONE'); | |
198 | |
199 error = SoundConverterOpen(&InputFormatInfo, &OutputFormatInfo, &myConverter); | |
12193 | 200 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"SoundConverterOpen:%i\n",error); |
8008 | 201 if(error) return 0; |
202 | |
203 if(sh->codecdata){ | |
204 error = SoundConverterSetInfo(myConverter,siDecompressionParams,sh->codecdata); | |
12193 | 205 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"SoundConverterSetInfo:%i\n",error); |
8632
9cecfb883d85
don't abort if setparams failed - fixes some Qclp files
arpi
parents:
8451
diff
changeset
|
206 // if(error) return 0; |
8008 | 207 } |
208 | |
209 WantedBufferSize=OutputFormatInfo.numChannels*OutputFormatInfo.sampleRate*2; | |
210 error = SoundConverterGetBufferSizes(myConverter, | |
211 WantedBufferSize,&FramesToGet,&InputBufferSize,&OutputBufferSize); | |
12193 | 212 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"SoundConverterGetBufferSizes:%i\n",error); |
213 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"WantedBufferSize = %li\n",WantedBufferSize); | |
214 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"InputBufferSize = %li\n",InputBufferSize); | |
215 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"OutputBufferSize = %li\n",OutputBufferSize); | |
216 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"FramesToGet = %li\n",FramesToGet); | |
8008 | 217 |
8389
e8c51ec86340
fixed in/out framesize rounding bug (causing low-rate QCLP hangup/looping)
arpi
parents:
8270
diff
changeset
|
218 InFrameSize=(InputBufferSize+FramesToGet-1)/FramesToGet; |
8008 | 219 OutFrameSize=OutputBufferSize/FramesToGet; |
220 | |
12193 | 221 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"FrameSize: %i -> %i\n",InFrameSize,OutFrameSize); |
8008 | 222 |
223 error = SoundConverterBeginConversion(myConverter); | |
12193 | 224 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"SoundConverterBeginConversion:%i\n",error); |
8008 | 225 if(error) return 0; |
226 | |
227 sh->audio_out_minsize=OutputBufferSize; | |
228 sh->audio_in_minsize=InputBufferSize; | |
229 | |
230 sh->channels=sh->wf->nChannels; | |
231 sh->samplerate=sh->wf->nSamplesPerSec; | |
8159 | 232 sh->samplesize=2; //(sh->wf->wBitsPerSample+7)/8; |
8008 | 233 |
234 sh->i_bps=sh->wf->nAvgBytesPerSec; | |
235 //InputBufferSize*WantedBufferSize/OutputBufferSize; | |
236 | |
237 #endif | |
8159 | 238 |
239 if(sh->format==0x3343414D){ | |
240 // MACE 3:1 | |
241 sh->ds->ss_div = 2*3; // 1 samples/packet | |
242 sh->ds->ss_mul = sh->channels*2*1; // 1 bytes/packet | |
243 } else | |
244 if(sh->format==0x3643414D){ | |
245 // MACE 6:1 | |
246 sh->ds->ss_div = 2*6; // 1 samples/packet | |
247 sh->ds->ss_mul = sh->channels*2*1; // 1 bytes/packet | |
248 } | |
249 | |
8008 | 250 return 1; // return values: 1=OK 0=ERROR |
251 } | |
252 | |
253 static int init(sh_audio_t *sh_audio){ | |
254 | |
255 return 1; // return values: 1=OK 0=ERROR | |
256 } | |
257 | |
258 static void uninit(sh_audio_t *sh){ | |
259 int error; | |
260 unsigned long ConvertedFrames=0; | |
261 unsigned long ConvertedBytes=0; | |
262 error=SoundConverterEndConversion(myConverter,NULL,&ConvertedFrames,&ConvertedBytes); | |
12193 | 263 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"SoundConverterEndConversion:%i\n",error); |
8008 | 264 error = SoundConverterClose(myConverter); |
12193 | 265 mp_msg(MSGT_DECAUDIO,MSGL_DBG2,"SoundConverterClose:%i\n",error); |
8389
e8c51ec86340
fixed in/out framesize rounding bug (causing low-rate QCLP hangup/looping)
arpi
parents:
8270
diff
changeset
|
266 // error = TerminateQTML(); |
e8c51ec86340
fixed in/out framesize rounding bug (causing low-rate QCLP hangup/looping)
arpi
parents:
8270
diff
changeset
|
267 // printf("TerminateQTML:%i\n",error); |
e8c51ec86340
fixed in/out framesize rounding bug (causing low-rate QCLP hangup/looping)
arpi
parents:
8270
diff
changeset
|
268 // FreeLibrary( qtml_dll ); |
e8c51ec86340
fixed in/out framesize rounding bug (causing low-rate QCLP hangup/looping)
arpi
parents:
8270
diff
changeset
|
269 // qtml_dll = NULL; |
e8c51ec86340
fixed in/out framesize rounding bug (causing low-rate QCLP hangup/looping)
arpi
parents:
8270
diff
changeset
|
270 // printf("qt dll loader uninit done\n"); |
9502
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
271 #ifdef MACOSX |
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
272 ExitMovies(); |
241bba8f60e8
MACOSX support patch, based on Dan Christiansens work
alex
parents:
9405
diff
changeset
|
273 #endif |
8008 | 274 } |
275 | |
276 static int decode_audio(sh_audio_t *sh,unsigned char *buf,int minlen,int maxlen){ | |
277 int error; | |
278 unsigned long FramesToGet=0; //how many frames the demuxer has to get | |
279 unsigned long InputBufferSize=0; //size of the input buffer | |
280 unsigned long ConvertedFrames=0; | |
281 unsigned long ConvertedBytes=0; | |
282 | |
283 FramesToGet=minlen/OutFrameSize; | |
284 if(FramesToGet*OutFrameSize<minlen && | |
285 (FramesToGet+1)*OutFrameSize<=maxlen) ++FramesToGet; | |
286 if(FramesToGet*InFrameSize>sh->a_in_buffer_size) | |
287 FramesToGet=sh->a_in_buffer_size/InFrameSize; | |
288 | |
289 InputBufferSize=FramesToGet*InFrameSize; | |
290 | |
8159 | 291 // printf("FramesToGet = %li (%li -> %li bytes)\n",FramesToGet, |
292 // InputBufferSize, FramesToGet*OutFrameSize); | |
8008 | 293 |
294 if(InputBufferSize>sh->a_in_buffer_len){ | |
295 int x=demux_read_data(sh->ds,&sh->a_in_buffer[sh->a_in_buffer_len], | |
296 InputBufferSize-sh->a_in_buffer_len); | |
297 if(x>0) sh->a_in_buffer_len+=x; | |
298 if(InputBufferSize>sh->a_in_buffer_len) | |
299 FramesToGet=sh->a_in_buffer_len/InFrameSize; // not enough data! | |
300 } | |
301 | |
8159 | 302 // printf("\nSoundConverterConvertBuffer(myConv=%p,inbuf=%p,frames=%d,outbuf=%p,&convframes=%p,&convbytes=%p)\n", |
303 // myConverter,sh->a_in_buffer,FramesToGet,buf,&ConvertedFrames,&ConvertedBytes); | |
8008 | 304 error = SoundConverterConvertBuffer(myConverter,sh->a_in_buffer, |
305 FramesToGet,buf,&ConvertedFrames,&ConvertedBytes); | |
8159 | 306 // printf("SoundConverterConvertBuffer:%i\n",error); |
307 // printf("ConvertedFrames = %li\n",ConvertedFrames); | |
308 // printf("ConvertedBytes = %li\n",ConvertedBytes); | |
8008 | 309 |
8389
e8c51ec86340
fixed in/out framesize rounding bug (causing low-rate QCLP hangup/looping)
arpi
parents:
8270
diff
changeset
|
310 // InputBufferSize=(ConvertedBytes/OutFrameSize)*InFrameSize; // FIXME!! |
e8c51ec86340
fixed in/out framesize rounding bug (causing low-rate QCLP hangup/looping)
arpi
parents:
8270
diff
changeset
|
311 InputBufferSize=FramesToGet*InFrameSize; |
8008 | 312 sh->a_in_buffer_len-=InputBufferSize; |
313 if(sh->a_in_buffer_len<0) sh->a_in_buffer_len=0; // should not happen... | |
314 else if(sh->a_in_buffer_len>0){ | |
315 memcpy(sh->a_in_buffer,&sh->a_in_buffer[InputBufferSize],sh->a_in_buffer_len); | |
316 } | |
317 | |
318 return ConvertedBytes; | |
319 } | |
320 | |
321 static int control(sh_audio_t *sh,int cmd,void* arg, ...){ | |
322 // various optional functions you MAY implement: | |
323 return CONTROL_UNKNOWN; | |
324 } | |
325 | |
326 #endif |