Mercurial > mplayer.hg
annotate stream/tvi_dshow.c @ 25060:96c96edb3d2f
Fix crash when pin connection fails.
author | voroshil |
---|---|
date | Sun, 18 Nov 2007 06:16:37 +0000 |
parents | 501f31700e77 |
children | 1f0eb7aa3206 |
rev | line source |
---|---|
24744 | 1 /* |
2 * TV support under Win32 | |
3 * | |
4 * (C) 2007 Vladimir Voroshilov <voroshil@gmail.com>. | |
5 * | |
6 * Based on tvi_dummy.c with help of tv.c, tvi_v4l2.c code . | |
7 * | |
8 * ------------------------------------------------------------------------------- | |
9 * | |
10 * This program is free software; you can redistribute it and/or modify | |
11 * it under the terms of the GNU General Public License as published by | |
12 * the Free Software Foundation; either version 2 of the License, or | |
13 * (at your option) any later version. | |
14 * | |
15 * This program is distributed in the hope that it will be useful, | |
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
18 * GNU General Public License for more details. | |
19 * | |
20 * You should have received a copy of the GNU General Public License | |
21 * along with this program; if not, write to the Free Software | |
22 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA | |
23 * | |
24 * | |
25 * ------------------------------------------------------------------------------- | |
26 * | |
27 * | |
28 * WARNING: This is alpha code! | |
29 * | |
30 * Abilities: | |
31 * * Watching TV under Win32 using WDM Capture driver and DirectShow | |
32 * * Grabbing synchronized audio/video with mencoder (synchronization is beeing done by DirectShow) | |
33 * * If device driver provides IAMStreamConfig interface, user can choose width/height with "-tv height=<h>:width=<w>" | |
34 * * Adjusting BRIGHTNESS,HUE,SATURATION,CONTRAST if supported by device | |
35 * * Selecting Tuner,S-Video,... as media source | |
36 * * User can select used video capture device, passing -tv device=<dev#> | |
37 * * User can select used audio input, passing -tv audioid=<input#> | |
38 * | |
39 * options which will not be implemented (probably sometime in future, if possible): | |
40 * * alsa | |
41 * * mjpeg | |
42 * * decimation=<1|2|4> | |
43 * * quality=<0\-100> | |
44 * * forceaudio | |
45 * * forcechan=<1\-2> | |
46 * * [volume|bass|treble|balance] | |
47 * | |
48 * Works with: | |
49 * - LifeView FlyTV Prime 34FM (SAA7134 based) with driver from Ivan Uskov | |
50 * Partially works with: | |
51 * - ATI 9200 VIVO based card | |
52 * - ATI AIW 7500 | |
53 * - nVidia Ti-4400 | |
54 * | |
55 * Known bugs: | |
56 * * stream goes with 24.93 FPS (NTSC), while reporting 25 FPS (PAL) ?! | |
57 * * direct set frequency call does not work ('Insufficient Buffer' error) | |
58 * * audio stream goes with about 1 sample/sec rate when capturing sound from audio card | |
59 * | |
60 * TODO: | |
61 * * check audio with small buffer on vivo !!! | |
62 * * norm for IAMVideoDecoder and for IAMTVtuner - differs !! | |
63 * * check how to change TVFormat on VIVO card without tuner | |
64 * * Flip image upside-down for RGB formats. | |
65 * * | |
66 * * remove debug sleep() | |
67 * * Add some notes to methods' parameters | |
68 * * refactor console messages | |
69 * * check using header files and keep only needed | |
70 * * add additional comments to methods' bodies | |
71 * | |
72 */ | |
73 | |
74 | |
75 /// \ingroup tvi_dshow | |
76 | |
77 #include "config.h" | |
78 | |
79 #include <stdio.h> | |
80 #include "libmpcodecs/img_format.h" | |
81 #include "libaf/af_format.h" | |
82 #include "help_mp.h" | |
83 #include "osdep/timer.h" | |
84 | |
85 | |
86 #include "tv.h" | |
87 #include "mp_msg.h" | |
88 #include "frequencies.h" | |
89 | |
90 | |
91 #include "tvi_dshow.h" | |
92 | |
93 static tvi_handle_t *tvi_init_dshow(tv_param_t* tv_param); | |
94 | |
95 /* | |
96 *--------------------------------------------------------------------------------------- | |
97 * | |
98 * Data structures | |
99 * | |
100 *--------------------------------------------------------------------------------------- | |
101 */ | |
102 /** | |
103 information about this file | |
104 */ | |
105 tvi_info_t tvi_info_dshow = { | |
106 tvi_init_dshow, | |
107 "DirectShow TV", | |
108 "dshow", | |
109 "Vladimir Voroshilov", | |
110 "Very experimental!! Use with caution" | |
111 }; | |
112 | |
113 | |
114 /** | |
115 ringbuffer related info | |
116 */ | |
117 typedef struct { | |
118 CRITICAL_SECTION *pMutex; ///< pointer to critical section (mutex) | |
119 char **ringbuffer; ///< ringbuffer array | |
120 double*dpts; ///< samples' timestamps | |
121 | |
122 int buffersize; ///< size of buffer in blocks | |
123 int blocksize; ///< size of individual block | |
124 int head; ///< index of first valid sample | |
125 int tail; ///< index of last valid sample | |
126 int count; ///< count of valid samples in ringbuffer | |
127 double tStart; ///< pts of first sample (first sample should have pts 0) | |
128 } grabber_ringbuffer_t; | |
129 | |
130 /** | |
131 CSampleGrabberCD definition | |
132 */ | |
133 typedef struct CSampleGrabberCB { | |
134 ISampleGrabberCBVtbl *lpVtbl; | |
135 int refcount; | |
136 GUID interfaces[2]; | |
137 grabber_ringbuffer_t *pbuf; | |
138 } CSampleGrabberCB; | |
139 | |
140 typedef struct { | |
141 int dev_index; ///< capture device index in device list (defaul: 0, first available device) | |
142 int adev_index; ///< audio capture device index in device list (default: -1, not used) | |
143 int immediate_mode; ///< immediate mode (no sound capture) | |
144 int state; ///< state: 1-filter graph running, 0-filter graph stopped | |
145 int direct_setfreq_call; ///< 0-find nearest channels from system channel list(workaround),1-direct call to set frequency | |
146 int direct_getfreq_call; ///< 0-find frequncy from frequency table (workaround),1-direct call to get frequency | |
147 | |
148 int fcc; ///< used video format code (FourCC) | |
149 int width; ///< picture width (default: auto) | |
150 int height; ///< picture height (default: auto) | |
151 | |
152 int channels; ///< number of audio channels (default: auto) | |
153 int samplerate; ///< audio samplerate (default: auto) | |
154 | |
155 long *freq_table; ///< frequency table (in Hz) | |
156 int freq_table_len; ///< length of freq table | |
157 int first_channel; ///< channel number of first entry in freq table | |
158 int input; ///< used input | |
159 | |
160 // video related stuff | |
161 int nVideoFormatUsed; ///< index of used video format | |
162 AM_MEDIA_TYPE **arpmtVideo; ///< available video formats | |
163 VIDEO_STREAM_CONFIG_CAPS **arVideoCaps; ///< capabilities of output video | |
164 AM_MEDIA_TYPE *pmtVideo; ///< video stream properties | |
165 grabber_ringbuffer_t *v_buf; | |
166 IBaseFilter *pVideoFilter; ///< interface for capture device | |
167 IAMStreamConfig *pVideoStreamConfig; ///< for configuring video stream | |
168 | |
169 // audio related stuff | |
170 int nAudioFormatUsed; ///< index of used audio format | |
171 AM_MEDIA_TYPE **arpmtAudio; ///< available audio formats | |
172 AUDIO_STREAM_CONFIG_CAPS **arAudioCaps; ///< capabilities of output audio | |
173 AM_MEDIA_TYPE *pmtAudio; ///< audio stream properties. | |
174 grabber_ringbuffer_t *a_buf; | |
175 IBaseFilter *pAudioFilter; ///< interface for audio capture device (if adevice passed) | |
176 IAMStreamConfig *pAudioStreamConfig; ///< for configuring audio stream | |
177 | |
178 AM_MEDIA_TYPE *pmtVBI; ///< available audio formats | |
179 grabber_ringbuffer_t *vbi_buf; | |
180 | |
181 IAMTVTuner *pTVTuner; ///< interface for tuner device | |
182 IGraphBuilder *pGraph; ///< filter graph | |
183 ICaptureGraphBuilder2 *pBuilder; ///< graph builder | |
184 IMediaControl *pMediaControl; ///< interface for controlling graph (start, stop,...) | |
185 IAMVideoProcAmp *pVideoProcAmp; ///< for adjusting hue,saturation,etc | |
186 IAMCrossbar *pCrossbar; ///< for selecting input (Tuner,Composite,S-Video,...) | |
187 DWORD dwRegister; ///< allow graphedit to connect to our graph | |
188 CSampleGrabberCB* pCSGCB; ///< callback object | |
189 void *priv_vbi; ///< private VBI data structure | |
190 tt_stream_props tsp; ///< data for VBI initialization | |
191 | |
192 tv_param_t* tv_param; ///< TV parameters | |
193 } priv_t; | |
194 | |
195 #include "tvi_def.h" | |
196 | |
197 /** | |
198 country table entry structure (for loading freq table stored in kstvtuner.ax | |
199 | |
200 \note | |
201 structure have to be 2-byte aligned and have 10-byte length!! | |
202 */ | |
203 typedef struct __attribute__((__packed__)) { | |
204 WORD CountryCode; ///< Country code | |
205 WORD CableFreqTable; ///< index of resource with frequencies for cable channels | |
206 WORD BroadcastFreqTable; ///< index of resource with frequencies for broadcast channels | |
207 DWORD VideoStandard; ///< used video standard | |
208 } TRCCountryList; | |
209 /** | |
210 information about image formats | |
211 */ | |
212 typedef struct { | |
213 uint32_t fmt; ///< FourCC | |
214 const GUID *subtype; ///< DirectShow's subtype | |
215 int nBits; ///< number of bits | |
216 int nCompression; ///< complression | |
217 int tail; ///< number of additional bytes followed VIDEOINFOHEADER structure | |
218 } img_fmt; | |
219 | |
220 /* | |
221 *--------------------------------------------------------------------------------------- | |
222 * | |
223 * Methods forward declaration | |
224 * | |
225 *--------------------------------------------------------------------------------------- | |
226 */ | |
227 static HRESULT init_ringbuffer(grabber_ringbuffer_t * rb, int buffersize, | |
228 int blocksize); | |
229 static HRESULT show_filter_info(IBaseFilter * pFilter); | |
230 #if 0 | |
231 //defined in current MinGW release | |
232 HRESULT STDCALL GetRunningObjectTable(DWORD, IRunningObjectTable **); | |
233 HRESULT STDCALL CreateItemMoniker(LPCOLESTR, LPCOLESTR, IMoniker **); | |
234 #endif | |
235 static CSampleGrabberCB *CSampleGrabberCB_Create(grabber_ringbuffer_t * | |
236 pbuf); | |
237 static int set_crossbar_input(priv_t * priv, int input); | |
238 static int subtype2imgfmt(const GUID * subtype); | |
239 | |
240 /* | |
241 *--------------------------------------------------------------------------------------- | |
242 * | |
243 * Global constants and variables | |
244 * | |
245 *--------------------------------------------------------------------------------------- | |
246 */ | |
247 /** | |
248 lookup tables for physical connector types | |
249 */ | |
250 static const struct { | |
251 long type; | |
252 char *name; | |
253 } tv_physcon_types[]={ | |
254 {PhysConn_Video_Tuner, "Tuner" }, | |
255 {PhysConn_Video_Composite, "Composite" }, | |
256 {PhysConn_Video_SVideo, "S-Video" }, | |
257 {PhysConn_Video_RGB, "RGB" }, | |
258 {PhysConn_Video_YRYBY, "YRYBY" }, | |
259 {PhysConn_Video_SerialDigital, "SerialDigital" }, | |
260 {PhysConn_Video_ParallelDigital, "ParallelDigital"}, | |
261 {PhysConn_Video_VideoDecoder, "VideoDecoder" }, | |
262 {PhysConn_Video_VideoEncoder, "VideoEncoder" }, | |
263 {PhysConn_Video_SCART, "SCART" }, | |
264 {PhysConn_Video_Black, "Blaack" }, | |
265 {PhysConn_Audio_Tuner, "Tuner" }, | |
266 {PhysConn_Audio_Line, "Line" }, | |
267 {PhysConn_Audio_Mic, "Mic" }, | |
268 {PhysConn_Audio_AESDigital, "AESDiital" }, | |
269 {PhysConn_Audio_SPDIFDigital, "SPDIFDigital" }, | |
270 {PhysConn_Audio_AudioDecoder, "AudioDecoder" }, | |
271 {PhysConn_Audio_SCSI, "SCSI" }, | |
272 {PhysConn_Video_SCSI, "SCSI" }, | |
273 {PhysConn_Audio_AUX, "AUX" }, | |
274 {PhysConn_Video_AUX, "AUX" }, | |
275 {PhysConn_Audio_1394, "1394" }, | |
276 {PhysConn_Video_1394, "1394" }, | |
277 {PhysConn_Audio_USB, "USB" }, | |
278 {PhysConn_Video_USB, "USB" }, | |
279 {-1, NULL } | |
280 }; | |
281 | |
282 static const struct { | |
283 char *chanlist_name; | |
284 int country_code; | |
285 } tv_chanlist2country[]={ | |
286 {"us-bcast", 1}, | |
287 {"russia", 7}, | |
288 {"argentina", 54}, | |
289 {"japan-bcast", 81}, | |
290 {"china-bcast", 86}, | |
291 {"southafrica", 27}, | |
292 {"australia", 61}, | |
293 {"ireland", 353}, | |
294 {"france", 33}, | |
295 {"italy", 39}, | |
296 {"newzealand", 64}, | |
297 //directshow table uses eastern europe freq table for russia | |
298 {"europe-east", 7}, | |
299 //directshow table uses western europe freq table for germany | |
300 {"europe-west", 49}, | |
301 /* cable channels */ | |
302 {"us-cable", 1}, | |
303 {"us-cable-hrc", 1}, | |
304 {"japan-cable", 81}, | |
305 //default is USA | |
306 {NULL, 1} | |
307 }; | |
308 | |
309 /** | |
310 array, contains information about various supported (i hope) image formats | |
311 */ | |
312 static const img_fmt img_fmt_list[] = { | |
25054 | 313 {IMGFMT_YUY2, &MEDIASUBTYPE_YUY2, 16, IMGFMT_YUY2, 0}, |
314 {IMGFMT_YV12, &MEDIASUBTYPE_YV12, 12, IMGFMT_YV12, 0}, | |
315 {IMGFMT_IYUV, &MEDIASUBTYPE_IYUV, 12, IMGFMT_IYUV, 0}, | |
316 {IMGFMT_I420, &MEDIASUBTYPE_I420, 12, IMGFMT_I420, 0}, | |
317 {IMGFMT_UYVY, &MEDIASUBTYPE_UYVY, 16, IMGFMT_UYVY, 0}, | |
318 {IMGFMT_YVYU, &MEDIASUBTYPE_YVYU, 16, IMGFMT_YVYU, 0}, | |
319 {IMGFMT_YVU9, &MEDIASUBTYPE_YVU9, 9, IMGFMT_YVU9, 0}, | |
320 {IMGFMT_BGR32, &MEDIASUBTYPE_RGB32, 32, 0, 0}, | |
321 {IMGFMT_BGR24, &MEDIASUBTYPE_RGB24, 24, 0, 0}, | |
322 {IMGFMT_BGR16, &MEDIASUBTYPE_RGB565, 16, 3, 12}, | |
323 {IMGFMT_BGR15, &MEDIASUBTYPE_RGB555, 16, 3, 12}, | |
324 {0, &GUID_NULL, 0, 0, 0} | |
24744 | 325 }; |
326 | |
327 #define TV_NORMS_COUNT 19 | |
328 static const struct { | |
329 long index; | |
330 char *name; | |
331 } tv_norms[TV_NORMS_COUNT] = { | |
332 { | |
333 AnalogVideo_NTSC_M, "ntsc-m"}, { | |
334 AnalogVideo_NTSC_M_J, "ntsc-mj"}, { | |
335 AnalogVideo_NTSC_433, "ntsc-433"}, { | |
336 AnalogVideo_PAL_B, "pal-b"}, { | |
337 AnalogVideo_PAL_D, "pal-d"}, { | |
338 AnalogVideo_PAL_G, "pal-g"}, { | |
339 AnalogVideo_PAL_H, "pal-h"}, { | |
340 AnalogVideo_PAL_I, "pal-i"}, { | |
341 AnalogVideo_PAL_M, "pal-m"}, { | |
342 AnalogVideo_PAL_N, "pal-n"}, { | |
343 AnalogVideo_PAL_60, "pal-60"}, { | |
344 AnalogVideo_SECAM_B, "secam-b"}, { | |
345 AnalogVideo_SECAM_D, "secam-d"}, { | |
346 AnalogVideo_SECAM_G, "secam-g"}, { | |
347 AnalogVideo_SECAM_H, "secam-h"}, { | |
348 AnalogVideo_SECAM_K, "secam-k"}, { | |
349 AnalogVideo_SECAM_K1, "secam-k1"}, { | |
350 AnalogVideo_SECAM_L, "secam-l"}, { | |
351 AnalogVideo_SECAM_L1, "secam-l1"} | |
352 }; | |
353 static long tv_available_norms[TV_NORMS_COUNT]; | |
354 static int tv_available_norms_count = 0; | |
355 | |
356 | |
357 static long *tv_available_inputs; | |
358 static int tv_available_inputs_count = 0; | |
359 | |
360 /* | |
361 *--------------------------------------------------------------------------------------- | |
362 * | |
363 * Various GUID definitions | |
364 * | |
365 *--------------------------------------------------------------------------------------- | |
366 */ | |
367 /// CLSID definitions (used for CoCreateInstance call) | |
368 DEFINE_GUID(CLSID_SampleGrabber, 0xC1F400A0, 0x3F08, 0x11d3, 0x9F, 0x0B, | |
369 0x00, 0x60, 0x08, 0x03, 0x9E, 0x37); | |
370 DEFINE_GUID(CLSID_NullRenderer, 0xC1F400A4, 0x3F08, 0x11d3, 0x9F, 0x0B, | |
371 0x00, 0x60, 0x08, 0x03, 0x9E, 0x37); | |
372 DEFINE_GUID(CLSID_SystemDeviceEnum, 0x62BE5D10, 0x60EB, 0x11d0, 0xBD, 0x3B, | |
373 0x00, 0xA0, 0xC9, 0x11, 0xCE, 0x86); | |
374 DEFINE_GUID(CLSID_CaptureGraphBuilder2, 0xBF87B6E1, 0x8C27, 0x11d0, 0xB3, | |
375 0xF0, 0x00, 0xAA, 0x00, 0x37, 0x61, 0xC5); | |
376 DEFINE_GUID(CLSID_VideoInputDeviceCategory, 0x860BB310, 0x5D01, 0x11d0, | |
377 0xBD, 0x3B, 0x00, 0xA0, 0xC9, 0x11, 0xCE, 0x86); | |
378 DEFINE_GUID(CLSID_AudioInputDeviceCategory, 0x33d9a762, 0x90c8, 0x11d0, | |
379 0xbd, 0x43, 0x00, 0xa0, 0xc9, 0x11, 0xce, 0x86); | |
380 DEFINE_GUID(CLSID_FilterGraph, 0xe436ebb3, 0x524f, 0x11ce, 0x9f, 0x53, | |
381 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70); | |
382 DEFINE_GUID(CLSID_SystemClock, 0xe436ebb1, 0x524f, 0x11ce, 0x9f, 0x53, | |
383 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70); | |
384 #ifdef NOT_USED | |
385 DEFINE_GUID(CLSID_CaptureGraphBuilder, 0xBF87B6E0, 0x8C27, 0x11d0, 0xB3, | |
386 0xF0, 0x00, 0xAA, 0x00, 0x37, 0x61, 0xC5); | |
387 DEFINE_GUID(CLSID_VideoPortManager, 0x6f26a6cd, 0x967b, 0x47fd, 0x87, 0x4a, | |
388 0x7a, 0xed, 0x2c, 0x9d, 0x25, 0xa2); | |
389 DEFINE_GUID(IID_IPin, 0x56a86891, 0x0ad4, 0x11ce, 0xb0, 0x3a, 0x00, 0x20, | |
390 0xaf, 0x0b, 0xa7, 0x70); | |
391 DEFINE_GUID(IID_ICaptureGraphBuilder, 0xbf87b6e0, 0x8c27, 0x11d0, 0xb3, | |
392 0xf0, 0x00, 0xaa, 0x00, 0x37, 0x61, 0xc5); | |
393 DEFINE_GUID(IID_IFilterGraph, 0x56a8689f, 0x0ad4, 0x11ce, 0xb0, 0x3a, 0x00, | |
394 0x20, 0xaf, 0x0b, 0xa7, 0x70); | |
395 DEFINE_GUID(PIN_CATEGORY_PREVIEW, 0xfb6c4282, 0x0353, 0x11d1, 0x90, 0x5f, | |
396 0x00, 0x00, 0xc0, 0xcc, 0x16, 0xba); | |
397 #endif | |
398 | |
399 /// IID definitions (used for QueryInterface call) | |
400 DEFINE_GUID(IID_IReferenceClock, 0x56a86897, 0x0ad4, 0x11ce, 0xb0, 0x3a, | |
401 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70); | |
402 DEFINE_GUID(IID_IAMBufferNegotiation, 0x56ED71A0, 0xAF5F, 0x11D0, 0xB3, 0xF0, | |
403 0x00, 0xAA, 0x00, 0x37, 0x61, 0xC5); | |
404 DEFINE_GUID(IID_IKsPropertySet, 0x31efac30, 0x515c, 0x11d0, 0xa9, 0xaa, | |
405 0x00, 0xaa, 0x00, 0x61, 0xbe, 0x93); | |
406 DEFINE_GUID(IID_ISampleGrabber, 0x6B652FFF, 0x11FE, 0x4fce, 0x92, 0xAD, | |
407 0x02, 0x66, 0xB5, 0xD7, 0xC7, 0x8F); | |
408 DEFINE_GUID(IID_ISampleGrabberCB, 0x0579154A, 0x2B53, 0x4994, 0xB0, 0xD0, | |
409 0xE7, 0x73, 0x14, 0x8E, 0xFF, 0x85); | |
410 DEFINE_GUID(IID_ICaptureGraphBuilder2, 0x93e5a4e0, 0x2d50, 0x11d2, 0xab, | |
411 0xfa, 0x00, 0xa0, 0xc9, 0xc6, 0xe3, 0x8d); | |
412 DEFINE_GUID(IID_ICreateDevEnum, 0x29840822, 0x5b84, 0x11d0, 0xbd, 0x3b, | |
413 0x00, 0xa0, 0xc9, 0x11, 0xce, 0x86); | |
414 DEFINE_GUID(IID_IGraphBuilder, 0x56a868a9, 0x0ad4, 0x11ce, 0xb0, 0x3a, | |
415 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70); | |
416 DEFINE_GUID(IID_IAMVideoProcAmp, 0xC6E13360, 0x30AC, 0x11d0, 0xA1, 0x8C, | |
417 0x00, 0xA0, 0xC9, 0x11, 0x89, 0x56); | |
418 DEFINE_GUID(IID_IVideoWindow, 0x56a868b4, 0x0ad4, 0x11ce, 0xb0, 0x3a, 0x00, | |
419 0x20, 0xaf, 0x0b, 0xa7, 0x70); | |
420 DEFINE_GUID(IID_IMediaControl, 0x56a868b1, 0x0ad4, 0x11ce, 0xb0, 0x3a, | |
421 0x00, 0x20, 0xaf, 0x0b, 0xa7, 0x70); | |
422 DEFINE_GUID(IID_IAMTVTuner, 0x211A8766, 0x03AC, 0x11d1, 0x8D, 0x13, 0x00, | |
423 0xAA, 0x00, 0xBD, 0x83, 0x39); | |
424 DEFINE_GUID(IID_IAMCrossbar, 0xc6e13380, 0x30ac, 0x11d0, 0xa1, 0x8c, 0x00, | |
425 0xa0, 0xc9, 0x11, 0x89, 0x56); | |
426 DEFINE_GUID(IID_IAMStreamConfig, 0xc6e13340, 0x30ac, 0x11d0, 0xa1, 0x8c, | |
427 0x00, 0xa0, 0xc9, 0x11, 0x89, 0x56); | |
428 DEFINE_GUID(IID_IAMAudioInputMixer, 0x54C39221, 0x8380, 0x11d0, 0xB3, 0xF0, | |
429 0x00, 0xAA, 0x00, 0x37, 0x61, 0xC5); | |
430 DEFINE_GUID(IID_IAMTVAudio, 0x83EC1C30, 0x23D1, 0x11d1, 0x99, 0xE6, 0x00, | |
431 0xA0, 0xC9, 0x56, 0x02, 0x66); | |
432 DEFINE_GUID(IID_IAMAnalogVideoDecoder, 0xC6E13350, 0x30AC, 0x11d0, 0xA1, | |
433 0x8C, 0x00, 0xA0, 0xC9, 0x11, 0x89, 0x56); | |
434 DEFINE_GUID(IID_IPropertyBag, 0x55272a00, 0x42cb, 0x11ce, 0x81, 0x35, 0x00, | |
435 0xaa, 0x00, 0x4b, 0xb8, 0x51); | |
436 DEFINE_GUID(PIN_CATEGORY_CAPTURE, 0xfb6c4281, 0x0353, 0x11d1, 0x90, 0x5f, | |
437 0x00, 0x00, 0xc0, 0xcc, 0x16, 0xba); | |
438 DEFINE_GUID(PIN_CATEGORY_VIDEOPORT, 0xfb6c4285, 0x0353, 0x11d1, 0x90, 0x5f, | |
439 0x00, 0x00, 0xc0, 0xcc, 0x16, 0xba); | |
440 DEFINE_GUID(PIN_CATEGORY_PREVIEW, 0xfb6c4282, 0x0353, 0x11d1, 0x90, 0x5f, | |
441 0x00, 0x00, 0xc0, 0xcc, 0x16, 0xba); | |
442 DEFINE_GUID(PIN_CATEGORY_VBI, 0xfb6c4284, 0x0353, 0x11d1, 0x90, 0x5f, | |
443 0x00, 0x00, 0xc0, 0xcc, 0x16, 0xba); | |
444 DEFINE_GUID(PROPSETID_TUNER, 0x6a2e0605, 0x28e4, 0x11d0, 0xa1, 0x8c, 0x00, | |
445 0xa0, 0xc9, 0x11, 0x89, 0x56); | |
446 DEFINE_GUID(MEDIATYPE_VBI, 0xf72a76e1, 0xeb0a, 0x11d0, 0xac, 0xe4, 0x00, | |
447 0x00, 0xc0, 0xcc, 0x16, 0xba); | |
448 | |
449 #define INSTANCEDATA_OF_PROPERTY_PTR(x) (((KSPROPERTY*)(x)) + 1) | |
450 #define INSTANCEDATA_OF_PROPERTY_SIZE(x) (sizeof((x)) - sizeof(KSPROPERTY)) | |
451 | |
452 #define DEVICE_NAME_MAX_LEN 2000 | |
453 | |
454 /*--------------------------------------------------------------------------------------- | |
455 * Methods, called only from this file | |
456 *---------------------------------------------------------------------------------------*/ | |
457 | |
458 void set_buffer_preference(int nDiv,WAVEFORMATEX* pWF,IPin* pOutPin,IPin* pInPin){ | |
459 ALLOCATOR_PROPERTIES prop; | |
460 IAMBufferNegotiation* pBN; | |
461 HRESULT hr; | |
462 | |
463 prop.cbAlign = -1; | |
464 prop.cbBuffer = pWF->nAvgBytesPerSec/nDiv; | |
465 if (!prop.cbBuffer) | |
466 prop.cbBuffer = 1; | |
467 prop.cbBuffer += pWF->nBlockAlign - 1; | |
468 prop.cbBuffer -= prop.cbBuffer % pWF->nBlockAlign; | |
469 prop.cbPrefix = -1; | |
470 prop.cBuffers = -1; | |
471 | |
472 hr=OLE_QUERYINTERFACE(pOutPin,IID_IAMBufferNegotiation,pBN); | |
473 if(FAILED(hr)) | |
474 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: pOutPin->QueryInterface(IID_IAMBufferNegotiation) Error: 0x%x\n",(unsigned int)hr); | |
475 else{ | |
476 hr=OLE_CALL_ARGS(pBN,SuggestAllocatorProperties,&prop); | |
477 if(FAILED(hr)) | |
478 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow:pOutPin->SuggestAllocatorProperties Error:0x%x\n",(unsigned int)hr); | |
479 OLE_RELEASE_SAFE(pBN); | |
480 } | |
481 hr=OLE_QUERYINTERFACE(pInPin,IID_IAMBufferNegotiation,pBN); | |
482 if(FAILED(hr)) | |
483 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: pInPin->QueryInterface(IID_IAMBufferNegotiation) Error: 0x%x",(unsigned int)hr); | |
484 else{ | |
485 hr=OLE_CALL_ARGS(pBN,SuggestAllocatorProperties,&prop); | |
486 if(FAILED(hr)) | |
487 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: pInPit->SuggestAllocatorProperties Error:0x%x\n",(unsigned int)hr); | |
488 OLE_RELEASE_SAFE(pBN); | |
489 } | |
490 } | |
491 /* | |
492 *--------------------------------------------------------------------------------------- | |
493 * | |
494 * CSampleGrabberCD class. Used for receiving samples from DirectShow. | |
495 * | |
496 *--------------------------------------------------------------------------------------- | |
497 */ | |
498 /// CSampleGrabberCD destructor | |
499 static void CSampleGrabberCB_Destroy(CSampleGrabberCB * This) | |
500 { | |
501 free(This->lpVtbl); | |
502 free(This); | |
503 } | |
504 | |
505 /// CSampleGrabberCD IUnknown interface methods implementation | |
506 static long STDCALL CSampleGrabberCB_QueryInterface(ISampleGrabberCB * | |
507 This, | |
508 const GUID * riid, | |
509 void **ppvObject) | |
510 { | |
511 CSampleGrabberCB *me = (CSampleGrabberCB *) This; | |
512 GUID *r; | |
513 unsigned int i = 0; | |
514 Debug printf("CSampleGrabberCB_QueryInterface(%p) called\n", This); | |
515 if (!ppvObject) | |
516 return E_POINTER; | |
517 for (r = me->interfaces; | |
518 i < sizeof(me->interfaces) / sizeof(me->interfaces[0]); r++, i++) | |
519 if (!memcmp(r, riid, sizeof(*r))) { | |
520 OLE_CALL(This, AddRef); | |
521 *ppvObject = This; | |
522 return 0; | |
523 } | |
524 Debug printf("Query failed! (GUID: 0x%x)\n", *(unsigned int *) riid); | |
525 return E_NOINTERFACE; | |
526 } | |
527 | |
528 static long STDCALL CSampleGrabberCB_AddRef(ISampleGrabberCB * This) | |
529 { | |
530 CSampleGrabberCB *me = (CSampleGrabberCB *) This; | |
531 Debug printf("CSampleGrabberCB_AddRef(%p) called (ref:%d)\n", This, | |
532 me->refcount); | |
533 return ++(me->refcount); | |
534 } | |
535 | |
536 static long STDCALL CSampleGrabberCB_Release(ISampleGrabberCB * This) | |
537 { | |
538 CSampleGrabberCB *me = (CSampleGrabberCB *) This; | |
539 Debug printf("CSampleGrabberCB_Release(%p) called (new ref:%d)\n", | |
540 This, me->refcount - 1); | |
541 if (--(me->refcount) == 0) | |
542 CSampleGrabberCB_Destroy(me); | |
543 return 0; | |
544 } | |
545 | |
546 | |
547 HRESULT STDCALL CSampleGrabberCB_BufferCB(ISampleGrabberCB * This, | |
548 double SampleTime, | |
549 BYTE * pBuffer, long lBufferLen) | |
550 { | |
551 CSampleGrabberCB *this = (CSampleGrabberCB *) This; | |
552 grabber_ringbuffer_t *rb = this->pbuf; | |
553 | |
554 if (!lBufferLen) | |
555 return E_FAIL; | |
556 | |
557 if (!rb->ringbuffer) { | |
558 rb->buffersize /= lBufferLen; | |
559 if (init_ringbuffer(rb, rb->buffersize, lBufferLen) != S_OK) | |
560 return E_FAIL; | |
561 } | |
562 mp_msg(MSGT_TV, MSGL_DBG4, | |
563 "tvi_dshow: BufferCB(%p): len=%ld ts=%f\n", This, lBufferLen, SampleTime); | |
564 EnterCriticalSection(rb->pMutex); | |
565 if (rb->count >= rb->buffersize) { | |
566 rb->head = (rb->head + 1) % rb->buffersize; | |
567 rb->count--; | |
568 } | |
569 | |
570 memcpy(rb->ringbuffer[rb->tail], pBuffer, | |
571 lBufferLen < rb->blocksize ? lBufferLen : rb->blocksize); | |
572 rb->dpts[rb->tail] = SampleTime; | |
573 rb->tail = (rb->tail + 1) % rb->buffersize; | |
574 rb->count++; | |
575 LeaveCriticalSection(rb->pMutex); | |
576 | |
577 return S_OK; | |
578 } | |
579 | |
580 /// wrapper. directshow does the same when BufferCB callback is requested | |
581 HRESULT STDCALL CSampleGrabberCB_SampleCB(ISampleGrabberCB * This, | |
582 double SampleTime, | |
583 LPMEDIASAMPLE pSample) | |
584 { | |
585 char* buf; | |
586 long len; | |
587 long long tStart,tEnd; | |
588 HRESULT hr; | |
589 grabber_ringbuffer_t *rb = ((CSampleGrabberCB*)This)->pbuf; | |
590 | |
591 len=OLE_CALL(pSample,GetSize); | |
592 tStart=tEnd=0; | |
593 hr=OLE_CALL_ARGS(pSample,GetTime,&tStart,&tEnd); | |
594 if(FAILED(hr)){ | |
595 return hr; | |
596 } | |
597 mp_msg(MSGT_TV, MSGL_DBG4,"tvi_dshow: SampleCB(%p): %d/%d %f\n", This,rb->count,rb->buffersize,1e-7*tStart); | |
598 hr=OLE_CALL_ARGS(pSample,GetPointer,(void*)&buf); | |
599 if(FAILED(hr)){ | |
600 return hr; | |
601 } | |
602 hr=CSampleGrabberCB_BufferCB(This,1e-7*tStart,buf,len); | |
603 return hr; | |
604 | |
605 } | |
606 | |
607 /// main grabbing routine | |
608 static CSampleGrabberCB *CSampleGrabberCB_Create(grabber_ringbuffer_t * | |
609 pbuf) | |
610 { | |
611 CSampleGrabberCB *This = malloc(sizeof(CSampleGrabberCB)); | |
612 if (!This) | |
613 return NULL; | |
614 | |
615 This->lpVtbl = malloc(sizeof(ISampleGrabberVtbl)); | |
616 if (!This->lpVtbl) { | |
617 CSampleGrabberCB_Destroy(This); | |
618 return NULL; | |
619 } | |
620 This->refcount = 1; | |
621 This->lpVtbl->QueryInterface = CSampleGrabberCB_QueryInterface; | |
622 This->lpVtbl->AddRef = CSampleGrabberCB_AddRef; | |
623 This->lpVtbl->Release = CSampleGrabberCB_Release; | |
624 This->lpVtbl->SampleCB = CSampleGrabberCB_SampleCB; | |
625 This->lpVtbl->BufferCB = CSampleGrabberCB_BufferCB; | |
626 | |
627 This->interfaces[0] = IID_IUnknown; | |
628 This->interfaces[1] = IID_ISampleGrabberCB; | |
629 | |
630 This->pbuf = pbuf; | |
631 | |
632 return This; | |
633 } | |
634 | |
635 /* | |
636 *--------------------------------------------------------------------------------------- | |
637 * | |
638 * ROT related methods (register, unregister) | |
639 * | |
640 *--------------------------------------------------------------------------------------- | |
641 */ | |
642 /** | |
643 Registering graph in ROT. User will be able to connect to graph from GraphEdit. | |
644 */ | |
645 static HRESULT AddToRot(IUnknown * pUnkGraph, DWORD * pdwRegister) | |
646 { | |
647 IMoniker *pMoniker; | |
648 IRunningObjectTable *pROT; | |
649 WCHAR wsz[256]; | |
650 HRESULT hr; | |
651 | |
652 if (FAILED(GetRunningObjectTable(0, &pROT))) { | |
653 return E_FAIL; | |
654 } | |
655 wsprintfW(wsz, L"FilterGraph %08x pid %08x", (DWORD_PTR) pUnkGraph, | |
656 GetCurrentProcessId()); | |
657 hr = CreateItemMoniker(L"!", wsz, &pMoniker); | |
658 if (SUCCEEDED(hr)) { | |
659 hr = OLE_CALL_ARGS(pROT, Register, ROTFLAGS_REGISTRATIONKEEPSALIVE, | |
660 pUnkGraph, pMoniker, pdwRegister); | |
661 OLE_RELEASE_SAFE(pMoniker); | |
662 } | |
663 OLE_RELEASE_SAFE(pROT); | |
664 return hr; | |
665 } | |
666 | |
667 /// Unregistering graph in ROT | |
668 static void RemoveFromRot(DWORD dwRegister) | |
669 { | |
670 IRunningObjectTable *pROT; | |
671 if (SUCCEEDED(GetRunningObjectTable(0, &pROT))) { | |
672 OLE_CALL_ARGS(pROT, Revoke, dwRegister); | |
673 OLE_RELEASE_SAFE(pROT); | |
674 } | |
675 } | |
676 | |
677 /* | |
678 *--------------------------------------------------------------------------------------- | |
679 * | |
680 * ringbuffer related methods (init, destroy) | |
681 * | |
682 *--------------------------------------------------------------------------------------- | |
683 */ | |
684 /** | |
685 * \brief ringbuffer destroying routine | |
686 * | |
687 * \param rb pointer to empty (just allocated) ringbuffer structure | |
688 * | |
689 * \note routine does not frees memory, allocated for grabber_rinbuffer_s structure | |
690 */ | |
691 static void destroy_ringbuffer(grabber_ringbuffer_t * rb) | |
692 { | |
693 int i; | |
694 | |
695 if (!rb) | |
696 return; | |
697 | |
698 if (rb->ringbuffer) { | |
699 for (i = 0; i < rb->buffersize; i++) | |
700 if (rb->ringbuffer[i]) | |
701 free(rb->ringbuffer[i]); | |
702 free(rb->ringbuffer); | |
703 rb->ringbuffer = NULL; | |
704 } | |
705 if (rb->dpts) { | |
706 free(rb->dpts); | |
707 rb->dpts = NULL; | |
708 } | |
709 if (rb->pMutex) { | |
710 DeleteCriticalSection(rb->pMutex); | |
711 free(rb->pMutex); | |
712 rb->pMutex = NULL; | |
713 } | |
714 | |
715 rb->blocksize = 0; | |
716 rb->buffersize = 0; | |
717 rb->head = 0; | |
718 rb->tail = 0; | |
719 rb->count = 0; | |
720 } | |
721 | |
722 /** | |
723 * \brief ringbuffer initialization | |
724 * | |
725 * \param rb pointer to empty (just allocated) ringbuffer structure | |
726 * \param buffersize size of buffer in blocks | |
727 * \param blocksize size of buffer's block | |
728 * | |
729 * \return S_OK if success | |
730 * \return E_OUTOFMEMORY not enough memory | |
731 * | |
732 * \note routine does not allocates memory for grabber_rinbuffer_s structure | |
733 */ | |
734 static HRESULT init_ringbuffer(grabber_ringbuffer_t * rb, int buffersize, | |
735 int blocksize) | |
736 { | |
737 int i; | |
738 | |
739 if (!rb) | |
740 return E_OUTOFMEMORY; | |
741 | |
742 rb->buffersize = buffersize < 2 ? 2 : buffersize; | |
743 rb->blocksize = blocksize; | |
744 | |
745 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Capture buffer: %d blocks of %d bytes.\n", | |
746 rb->buffersize, rb->blocksize); | |
747 | |
748 rb->ringbuffer = (char **) malloc(rb->buffersize * sizeof(char *)); | |
749 if (!rb) | |
750 return E_POINTER; | |
751 memset(rb->ringbuffer, 0, rb->buffersize * sizeof(char *)); | |
752 | |
753 for (i = 0; i < rb->buffersize; i++) { | |
754 rb->ringbuffer[i] = (char *) malloc(rb->blocksize * sizeof(char)); | |
755 if (!rb->ringbuffer[i]) { | |
756 destroy_ringbuffer(rb); | |
757 return E_OUTOFMEMORY; | |
758 } | |
759 } | |
760 rb->dpts = (double*) malloc(rb->buffersize * sizeof(double)); | |
761 if (!rb->dpts) { | |
762 destroy_ringbuffer(rb); | |
763 return E_OUTOFMEMORY; | |
764 } | |
765 rb->head = 0; | |
766 rb->tail = 0; | |
767 rb->count = 0; | |
768 rb->tStart = -1; | |
769 rb->pMutex = (CRITICAL_SECTION *) malloc(sizeof(CRITICAL_SECTION)); | |
770 if (!rb->pMutex) { | |
771 destroy_ringbuffer(rb); | |
772 return E_OUTOFMEMORY; | |
773 } | |
774 InitializeCriticalSection(rb->pMutex); | |
775 return S_OK; | |
776 } | |
777 | |
778 /* | |
779 *--------------------------------------------------------------------------------------- | |
780 * | |
781 * Tuner related methods (frequency, capabilities, etc | |
782 * | |
783 *--------------------------------------------------------------------------------------- | |
784 */ | |
785 /** | |
786 * \brief returns string with name for givend PsysCon_* constant | |
787 * | |
788 * \param lPhysicalType constant from PhysicalConnectorType enumeration | |
789 * | |
790 * \return pointer to string with apropriate name | |
791 * | |
792 * \note | |
793 * Caller should not free returned pointer | |
794 */ | |
795 static char *physcon2str(const long lPhysicalType) | |
796 { | |
797 int i; | |
798 for(i=0; tv_physcon_types[i].name; i++) | |
799 if(tv_physcon_types[i].type==lPhysicalType) | |
800 return tv_physcon_types[i].name; | |
801 return "Unknown"; | |
802 }; | |
803 | |
804 /** | |
805 * \brief converts MPlayer's chanlist to system country code. | |
806 * | |
807 * \param chanlist MPlayer's chanlist name | |
808 * | |
809 * \return system country code | |
810 * | |
811 * \remarks | |
812 * After call to IAMTVTuner::put_CountryCode with returned value tuner switches to frequency table used in specified | |
813 * country (which is usually larger then MPlayer's one, so workaround will work fine). | |
814 * | |
815 * \todo | |
816 * Resolve trouble with cable channels (DirectShow's tuners must be switched between broadcast and cable channels modes. | |
817 */ | |
818 static int chanlist2country(char *chanlist) | |
819 { | |
820 int i; | |
821 for(i=0; tv_chanlist2country[i].chanlist_name; i++) | |
822 if (!strcmp(chanlist, tv_chanlist2country[i].chanlist_name)) | |
823 break; | |
824 return tv_chanlist2country[i].country_code; | |
825 } | |
826 | |
827 /** | |
828 * \brief loads specified resource from module and return pointer to it | |
829 * | |
830 * \param hDLL valid module desriptor | |
831 * \param index index of resource. resource with name "#<index>" will be loaded | |
832 * | |
833 * \return pointer to loader resource or NULL if error occured | |
834 */ | |
835 static void *GetRC(HMODULE hDLL, int index) | |
836 { | |
837 char szRCDATA[10]; | |
838 char szName[10]; | |
839 HRSRC hRes; | |
840 HGLOBAL hTable; | |
841 | |
842 snprintf(szRCDATA, 10, "#%d", (int)RT_RCDATA); | |
843 snprintf(szName, 10, "#%d", index); | |
844 | |
845 hRes = FindResource(hDLL, szName, szRCDATA); | |
846 if (!hRes) { | |
847 return NULL; | |
848 } | |
849 hTable = LoadResource(hDLL, hRes); | |
850 if (!hTable) { | |
851 return NULL; | |
852 } | |
853 return LockResource(hTable); | |
854 } | |
855 | |
856 /** | |
857 * \brief loads frequency table for given country from kstvtune.ax | |
858 * | |
859 * \param[in] nCountry - country code | |
860 * \param[in] nInputType (TunerInputCable or TunerInputAntenna) | |
861 * \param[out] pplFreqTable - address of variable that receives pointer to array, containing frequencies | |
862 * \param[out] pnLen length of array | |
863 * \param[out] pnFirst - channel number of first entry in array (nChannelMax) | |
864 * | |
865 * \return S_OK if success | |
866 * \return E_POINTER pplFreqTable==NULL || plFirst==NULL || pnLen==NULL | |
867 * \return E_FAIL error occured during load | |
868 * | |
869 * \remarks | |
870 * - array must be freed by caller | |
871 * - MSDN says that it is not neccessery to unlock or free resource. It will be done after unloading DLL | |
872 */ | |
873 static HRESULT load_freq_table(int nCountry, int nInputType, | |
874 long **pplFreqTable, int *pnLen, | |
875 int *pnFirst) | |
876 { | |
877 HMODULE hDLL; | |
878 long *plFreqTable; | |
879 TRCCountryList *pCountryList; | |
880 int i, index; | |
881 | |
882 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: load_freq_table called %d (%d)\n",nCountry,nInputType); | |
883 /* ASSERT(sizeof(TRCCountryList)==10); // need properly aligned structure */ | |
884 | |
885 if (!pplFreqTable || !pnFirst || !pnLen) | |
886 return E_POINTER; | |
887 if (!nCountry) | |
888 return E_FAIL; | |
889 | |
890 hDLL = LoadLibrary("kstvtune.ax"); | |
891 if (!hDLL) { | |
892 return E_FAIL; | |
893 } | |
894 pCountryList = GetRC(hDLL, 9999); | |
895 if (!pCountryList) { | |
896 FreeLibrary(hDLL); | |
897 return E_FAIL; | |
898 } | |
899 for (i = 0; pCountryList[i].CountryCode != 0; i++) | |
900 if (pCountryList[i].CountryCode == nCountry) | |
901 break; | |
902 if (pCountryList[i].CountryCode == 0) { | |
903 FreeLibrary(hDLL); | |
904 return E_FAIL; | |
905 } | |
906 if (nInputType == TunerInputCable) | |
907 index = pCountryList[i].CableFreqTable; | |
908 else | |
909 index = pCountryList[i].BroadcastFreqTable; | |
910 | |
911 plFreqTable = GetRC(hDLL, index); //First element is number of first channel, second - number of last channel | |
912 if (!plFreqTable) { | |
913 FreeLibrary(hDLL); | |
914 return E_FAIL; | |
915 } | |
916 *pnFirst = plFreqTable[0]; | |
917 *pnLen = (int) (plFreqTable[1] - plFreqTable[0] + 1); | |
918 *pplFreqTable = (long *) malloc((*pnLen) * sizeof(long)); | |
919 if (!*pplFreqTable) { | |
920 FreeLibrary(hDLL); | |
921 return E_FAIL; | |
922 } | |
923 for (i = 0; i < *pnLen; i++) { | |
924 (*pplFreqTable)[i] = plFreqTable[i + 2]; | |
925 } | |
926 FreeLibrary(hDLL); | |
927 return S_OK; | |
928 } | |
929 | |
930 /** | |
931 * \brief tunes to given frequency through IKsPropertySet call | |
932 * | |
933 * \param pTVTuner IAMTVTuner interface of capture device | |
934 * \param lFreq frequency to tune (in Hz) | |
935 * | |
936 * \return S_OK success | |
937 * \return apropriate error code otherwise | |
938 * | |
939 * \note | |
940 * Due to either bug in driver or error in following code calll to IKsProperty::Set | |
941 * in this methods always fail with error 0x8007007a. | |
942 * | |
943 * \todo test code on other machines and an error | |
944 */ | |
945 static HRESULT set_frequency_direct(IAMTVTuner * pTVTuner, long lFreq) | |
946 { | |
947 HRESULT hr; | |
948 DWORD dwSupported = 0; | |
949 DWORD cbBytes = 0; | |
950 KSPROPERTY_TUNER_MODE_CAPS_S mode_caps; | |
951 KSPROPERTY_TUNER_FREQUENCY_S frequency; | |
952 IKsPropertySet *pKSProp; | |
953 | |
954 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: set_frequency_direct called\n"); | |
955 | |
956 memset(&mode_caps, 0, sizeof(mode_caps)); | |
957 memset(&frequency, 0, sizeof(frequency)); | |
958 | |
959 hr = OLE_QUERYINTERFACE(pTVTuner, IID_IKsPropertySet, pKSProp); | |
960 if (FAILED(hr)) | |
961 return hr; //no IKsPropertySet interface | |
962 | |
963 mode_caps.Mode = AMTUNER_MODE_TV; | |
964 hr = OLE_CALL_ARGS(pKSProp, QuerySupported, &PROPSETID_TUNER, | |
965 KSPROPERTY_TUNER_MODE_CAPS, &dwSupported); | |
966 if (FAILED(hr)) { | |
967 OLE_RELEASE_SAFE(pKSProp); | |
968 return hr; | |
969 } | |
970 | |
971 if (!dwSupported & KSPROPERTY_SUPPORT_GET) { | |
972 OLE_RELEASE_SAFE(pKSProp); | |
973 return E_FAIL; //PROPSETID_TINER not supported | |
974 } | |
975 | |
976 hr = OLE_CALL_ARGS(pKSProp, Get, &PROPSETID_TUNER, | |
977 KSPROPERTY_TUNER_MODE_CAPS, | |
978 INSTANCEDATA_OF_PROPERTY_PTR(&mode_caps), | |
979 INSTANCEDATA_OF_PROPERTY_SIZE(mode_caps), | |
980 &mode_caps, sizeof(mode_caps), &cbBytes); | |
981 | |
982 frequency.Frequency = lFreq; | |
983 | |
984 if (mode_caps.Strategy == KS_TUNER_STRATEGY_DRIVER_TUNES) | |
985 frequency.TuningFlags = KS_TUNER_TUNING_FINE; | |
986 else | |
987 frequency.TuningFlags = KS_TUNER_TUNING_EXACT; | |
988 | |
989 if (lFreq < mode_caps.MinFrequency || lFreq > mode_caps.MaxFrequency) { | |
990 OLE_RELEASE_SAFE(pKSProp); | |
991 return E_FAIL; | |
992 } | |
993 | |
994 hr = OLE_CALL_ARGS(pKSProp, Set, &PROPSETID_TUNER, | |
995 KSPROPERTY_TUNER_FREQUENCY, | |
996 INSTANCEDATA_OF_PROPERTY_PTR(&frequency), | |
997 INSTANCEDATA_OF_PROPERTY_SIZE(frequency), | |
998 &frequency, sizeof(frequency)); | |
999 if (FAILED(hr)) { | |
1000 OLE_RELEASE_SAFE(pKSProp); | |
1001 return hr; | |
1002 } | |
1003 | |
1004 OLE_RELEASE_SAFE(pKSProp); | |
1005 | |
1006 return S_OK; | |
1007 } | |
1008 | |
1009 /** | |
1010 * \brief find channel with nearest frequency and set it | |
1011 * | |
1012 * \param priv driver's private data | |
1013 * \param lFreq frequency in Hz | |
1014 * | |
1015 * \return S_OK if success | |
1016 * \return E_FAIL if error occured | |
1017 */ | |
1018 static HRESULT set_nearest_freq(priv_t * priv, long lFreq) | |
1019 { | |
1020 HRESULT hr; | |
1021 int i; | |
1022 long lFreqDiff=-1; | |
1023 int nChannel; | |
1024 TunerInputType tunerInput; | |
1025 long lInput; | |
1026 | |
1027 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: set_nearest_freq called\n"); | |
1028 if(priv->freq_table_len == -1 && !priv->freq_table) { | |
1029 | |
1030 hr = OLE_CALL_ARGS(priv->pTVTuner, get_ConnectInput, &lInput); | |
1031 if(FAILED(hr)){ //Falling back to 0 | |
1032 lInput=0; | |
1033 } | |
1034 | |
1035 hr = OLE_CALL_ARGS(priv->pTVTuner, get_InputType, lInput, &tunerInput); | |
1036 | |
1037 if (load_freq_table(chanlist2country(priv->tv_param->chanlist), tunerInput, &(priv->freq_table), &(priv->freq_table_len), &(priv->first_channel)) != S_OK) {//FIXME | |
1038 priv->freq_table_len=0; | |
1039 priv->freq_table=NULL; | |
1040 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_UnableExtractFreqTable); | |
1041 return E_FAIL; | |
1042 }; | |
1043 mp_msg(MSGT_TV, MSGL_V, MSGTR_TVI_DS_FreqTableLoaded, tunerInput == TunerInputAntenna ? "broadcast" : "cable", | |
1044 chanlist2country(priv->tv_param->chanlist), priv->freq_table_len); | |
1045 } | |
1046 | |
1047 if (priv->freq_table_len <= 0) | |
1048 return E_FAIL; | |
1049 | |
1050 //FIXME: rewrite search algo | |
1051 nChannel = -1; | |
1052 for (i = 0; i < priv->freq_table_len; i++) { | |
1053 if (nChannel == -1 || labs(lFreq - priv->freq_table[i]) < lFreqDiff) { | |
1054 nChannel = priv->first_channel + i; | |
1055 lFreqDiff = labs(lFreq - priv->freq_table[i]); | |
1056 } | |
1057 } | |
1058 if (nChannel == -1) { | |
1059 mp_msg(MSGT_TV,MSGL_ERR, MSGTR_TVI_DS_UnableFindNearestChannel); | |
1060 return E_FAIL; | |
1061 } | |
1062 hr = OLE_CALL_ARGS(priv->pTVTuner, put_Channel, nChannel, | |
1063 AMTUNER_SUBCHAN_DEFAULT, AMTUNER_SUBCHAN_DEFAULT); | |
1064 if (FAILED(hr)) { | |
1065 mp_msg(MSGT_TV,MSGL_ERR,MSGTR_TVI_DS_UnableToSetChannel, (unsigned int)hr); | |
1066 return E_FAIL; | |
1067 } | |
1068 return S_OK; | |
1069 } | |
1070 | |
1071 /** | |
1072 * \brief setting frequency. decides whether use direct call/workaround | |
1073 * | |
1074 * \param priv driver's private data | |
1075 * \param lFreq frequency in Hz | |
1076 * | |
1077 * \return TVI_CONTROL_TRUE if success | |
1078 * \return TVI_CONTROL_FALSE if error occured | |
1079 * | |
1080 * \todo check for freq boundary | |
1081 */ | |
1082 static int set_frequency(priv_t * priv, long lFreq) | |
1083 { | |
1084 HRESULT hr; | |
1085 | |
1086 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: set_frequency called\n"); | |
1087 if (!priv->pTVTuner) | |
1088 return TVI_CONTROL_FALSE; | |
1089 if (priv->direct_setfreq_call) { //using direct call to set frequency | |
1090 hr = set_frequency_direct(priv->pTVTuner, lFreq); | |
1091 if (FAILED(hr)) { | |
1092 mp_msg(MSGT_TV, MSGL_V, MSGTR_TVI_DS_DirectSetFreqFailed); | |
1093 priv->direct_setfreq_call = 0; | |
1094 } | |
1095 } | |
1096 if (!priv->direct_setfreq_call) { | |
1097 hr = set_nearest_freq(priv, lFreq); | |
1098 } | |
1099 if (FAILED(hr)) | |
1100 return TVI_CONTROL_FALSE; | |
1101 #ifdef DEPRECATED | |
1102 priv->pGrabber->ClearBuffer(priv->pGrabber); | |
1103 #endif | |
1104 return TVI_CONTROL_TRUE; | |
1105 } | |
1106 | |
1107 /** | |
1108 * \brief return current frequency from tuner (in Hz) | |
1109 * | |
1110 * \param pTVTuner IAMTVTuner interface of tuner | |
1111 * \param plFreq address of variable that receives current frequency | |
1112 * | |
1113 * \return S_OK success | |
1114 * \return E_POINTER pTVTuner==NULL || plFreq==NULL | |
1115 * \return apropriate error code otherwise | |
1116 */ | |
1117 static HRESULT get_frequency_direct(IAMTVTuner * pTVTuner, long *plFreq) | |
1118 { | |
1119 HRESULT hr; | |
1120 KSPROPERTY_TUNER_STATUS_S TunerStatus; | |
1121 DWORD cbBytes; | |
1122 IKsPropertySet *pKSProp; | |
1123 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: get_frequency_direct called\n"); | |
1124 | |
1125 if (!plFreq) | |
1126 return E_POINTER; | |
1127 | |
1128 hr = OLE_QUERYINTERFACE(pTVTuner, IID_IKsPropertySet, pKSProp); | |
1129 if (FAILED(hr)) { | |
1130 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Get freq QueryInterface failed\n"); | |
1131 return hr; | |
1132 } | |
1133 | |
1134 hr = OLE_CALL_ARGS(pKSProp, Get, &PROPSETID_TUNER, | |
1135 KSPROPERTY_TUNER_STATUS, | |
1136 INSTANCEDATA_OF_PROPERTY_PTR(&TunerStatus), | |
1137 INSTANCEDATA_OF_PROPERTY_SIZE(TunerStatus), | |
1138 &TunerStatus, sizeof(TunerStatus), &cbBytes); | |
1139 if (FAILED(hr)) { | |
1140 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Get freq Get failure\n"); | |
1141 return hr; | |
1142 } | |
1143 *plFreq = TunerStatus.CurrentFrequency; | |
1144 return S_OK; | |
1145 } | |
1146 | |
1147 /** | |
1148 * \brief gets current frequency | |
1149 * | |
1150 * \param priv driver's private data structure | |
1151 * \param plFreq - pointer to long int to store frequency to (in Hz) | |
1152 * | |
1153 * \return TVI_CONTROL_TRUE if success, TVI_CONTROL_FALSE otherwise | |
1154 */ | |
1155 static int get_frequency(priv_t * priv, long *plFreq) | |
1156 { | |
1157 HRESULT hr; | |
1158 | |
1159 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: get_frequency called\n"); | |
1160 | |
1161 if (!plFreq || !priv->pTVTuner) | |
1162 return TVI_CONTROL_FALSE; | |
1163 | |
1164 if (priv->direct_getfreq_call) { //using direct call to get frequency | |
1165 hr = get_frequency_direct(priv->pTVTuner, plFreq); | |
1166 if (FAILED(hr)) { | |
1167 mp_msg(MSGT_TV, MSGL_INFO, MSGTR_TVI_DS_DirectGetFreqFailed); | |
1168 priv->direct_getfreq_call = 0; | |
1169 } | |
1170 } | |
1171 if (!priv->direct_getfreq_call) { | |
1172 hr=OLE_CALL_ARGS(priv->pTVTuner, get_VideoFrequency, plFreq); | |
1173 if (FAILED(hr)) | |
1174 return TVI_CONTROL_FALSE; | |
1175 | |
1176 } | |
1177 return TVI_CONTROL_TRUE; | |
1178 } | |
1179 | |
1180 /** | |
1181 * \brief get tuner capabilities | |
1182 * | |
1183 * \param priv driver's private data | |
1184 */ | |
1185 static void get_capabilities(priv_t * priv) | |
1186 { | |
1187 long lAvailableFormats; | |
1188 HRESULT hr; | |
1189 int i; | |
1190 long lInputPins, lOutputPins, lRelated, lPhysicalType; | |
1191 IEnumPins *pEnum; | |
1192 char tmp[200]; | |
1193 IPin *pPin = 0; | |
1194 PIN_DIRECTION ThisPinDir; | |
1195 PIN_INFO pi; | |
1196 IAMAudioInputMixer *pIAMixer; | |
1197 | |
1198 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: get_capabilities called\n"); | |
1199 if (priv->pTVTuner) { | |
1200 | |
1201 mp_msg(MSGT_TV, MSGL_V, MSGTR_TVI_DS_SupportedNorms); | |
1202 hr = OLE_CALL_ARGS(priv->pTVTuner, get_AvailableTVFormats, | |
1203 &lAvailableFormats); | |
1204 if (FAILED(hr)) | |
1205 tv_available_norms_count = 0; | |
1206 else { | |
1207 for (i = 0; i < TV_NORMS_COUNT; i++) { | |
1208 if (lAvailableFormats & tv_norms[i].index) { | |
1209 tv_available_norms[tv_available_norms_count] = i; | |
1210 mp_msg(MSGT_TV, MSGL_V, " %d=%s;", | |
1211 tv_available_norms_count + 1, tv_norms[i].name); | |
1212 tv_available_norms_count++; | |
1213 } | |
1214 } | |
1215 } | |
1216 mp_msg(MSGT_TV, MSGL_INFO, "\n"); | |
1217 } | |
1218 if (priv->pCrossbar) { | |
1219 OLE_CALL_ARGS(priv->pCrossbar, get_PinCounts, &lOutputPins, | |
1220 &lInputPins); | |
1221 | |
1222 tv_available_inputs = (long *) malloc(sizeof(long) * lInputPins); | |
1223 tv_available_inputs_count = 0; | |
1224 | |
1225 mp_msg(MSGT_TV, MSGL_V, MSGTR_TVI_DS_AvailableVideoInputs); | |
1226 for (i = 0; i < lInputPins; i++) { | |
1227 OLE_CALL_ARGS(priv->pCrossbar, get_CrossbarPinInfo, 1, i, | |
1228 &lRelated, &lPhysicalType); | |
1229 | |
1230 if (lPhysicalType < 0x1000) { | |
1231 tv_available_inputs[tv_available_inputs_count++] = i; | |
1232 mp_msg(MSGT_TV, MSGL_V, " %d=%s;", | |
1233 tv_available_inputs_count - 1, | |
1234 physcon2str(lPhysicalType)); | |
1235 } | |
1236 } | |
1237 mp_msg(MSGT_TV, MSGL_INFO, "\n"); | |
1238 | |
1239 set_crossbar_input(priv, 0); | |
1240 } | |
1241 | |
1242 if (priv->adev_index != -1) { | |
1243 hr = OLE_CALL_ARGS(priv->pAudioFilter, EnumPins, &pEnum); | |
1244 if (FAILED(hr)) | |
1245 return; | |
1246 mp_msg(MSGT_TV, MSGL_V, MSGTR_TVI_DS_AvailableAudioInputs); | |
1247 i = 0; | |
1248 while (OLE_CALL_ARGS(pEnum, Next, 1, &pPin, NULL) == S_OK) { | |
1249 memset(&pi, 0, sizeof(pi)); | |
1250 memset(tmp, 0, 200); | |
1251 OLE_CALL_ARGS(pPin, QueryDirection, &ThisPinDir); | |
1252 if (ThisPinDir == PINDIR_INPUT) { | |
1253 OLE_CALL_ARGS(pPin, QueryPinInfo, &pi); | |
1254 wtoa(pi.achName, tmp, 200); | |
1255 OLE_RELEASE_SAFE(pi.pFilter); | |
1256 mp_msg(MSGT_TV, MSGL_V, " %d=%s", i, tmp); | |
1257 mp_msg(MSGT_TV, MSGL_DBG3, " (%p)", pPin); | |
1258 hr = OLE_QUERYINTERFACE(pPin, IID_IAMAudioInputMixer,pIAMixer); | |
1259 if (SUCCEEDED(hr)) { | |
1260 if (i == priv->tv_param->audio_id) { | |
1261 OLE_CALL_ARGS(pIAMixer, put_Enable, TRUE); | |
1262 if(priv->tv_param->volume>0) | |
1263 OLE_CALL_ARGS(pIAMixer, put_MixLevel, 0.01 * priv->tv_param->volume); | |
1264 #if 0 | |
1265 else | |
1266 OLE_CALL_ARGS(pIAMixer, put_MixLevel, 1.0); | |
1267 #endif | |
1268 mp_msg(MSGT_TV, MSGL_V, MSGTR_TVI_DS_InputSelected); | |
1269 } else { | |
1270 OLE_CALL_ARGS(pIAMixer, put_Enable, FALSE); | |
1271 #if 0 | |
1272 OLE_CALL_ARGS(pIAMixer, put_MixLevel, 0.0); | |
1273 #endif | |
1274 } | |
1275 OLE_RELEASE_SAFE(pIAMixer); | |
1276 } | |
1277 mp_msg(MSGT_TV, MSGL_V, ";"); | |
1278 OLE_RELEASE_SAFE(pPin); | |
1279 i++; | |
1280 } | |
1281 } | |
1282 mp_msg(MSGT_TV, MSGL_INFO, "\n"); | |
1283 OLE_RELEASE_SAFE(pEnum); | |
1284 } | |
1285 } | |
1286 | |
1287 /* | |
1288 *--------------------------------------------------------------------------------------- | |
1289 * | |
1290 * Filter related methods | |
1291 * | |
1292 *--------------------------------------------------------------------------------------- | |
1293 */ | |
1294 /** | |
1295 * \brief building in graph audio/video capture chain | |
1296 * | |
1297 * \param priv driver's private data | |
1298 * \param pCaptureFilter pointer to capture device's IBaseFilter interface | |
1299 * \param pbuf ringbuffer data structure | |
1300 * \param pmt media type for chain (AM_MEDIA_TYPE) | |
1301 * | |
1302 * \note routine does not frees memory, allocated for grabber_rinbuffer_s structure | |
1303 */ | |
1304 static HRESULT build_sub_graph(priv_t * priv, IBaseFilter * pCaptureFilter, | |
1305 grabber_ringbuffer_t * pbuf, | |
1306 AM_MEDIA_TYPE * pmt, const GUID* ppin_category) | |
1307 { | |
1308 HRESULT hr; | |
1309 | |
25029
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1310 AM_MEDIA_TYPE conn_mt; //Media type of established connection |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1311 |
24744 | 1312 IPin *pSGIn; |
1313 IPin *pSGOut; | |
1314 IPin *pNRIn=NULL; | |
1315 IPin *pCapturePin; | |
1316 | |
1317 IBaseFilter *pNR = NULL; | |
1318 IBaseFilter *pSGF = NULL; | |
1319 | |
1320 ISampleGrabber *pSG = NULL; | |
1321 | |
1322 hr=S_OK; | |
25060 | 1323 CopyMediaType(&conn_mt, pmt); |
24744 | 1324 do{ |
1325 hr = OLE_CALL_ARGS(priv->pBuilder, FindPin, | |
1326 (IUnknown *) pCaptureFilter, | |
1327 PINDIR_OUTPUT, ppin_category, | |
1328 &(pmt->majortype), FALSE, 0, &pCapturePin); | |
1329 if(FAILED(hr)){ | |
1330 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: FindPin(pCapturePin) call failed. Error:0x%x\n", (unsigned int)hr); | |
1331 break; | |
1332 } | |
1333 /* Addinf SampleGrabber filter for video stream */ | |
1334 hr = CoCreateInstance((GUID *) & CLSID_SampleGrabber, NULL,CLSCTX_INPROC_SERVER, &IID_IBaseFilter,(void *) &pSGF); | |
1335 if(FAILED(hr)){ | |
1336 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: CoCreateInstance(SampleGrabber) call failed. Error:0x%x\n", (unsigned int)hr); | |
1337 break; | |
1338 } | |
1339 hr = OLE_CALL_ARGS(priv->pGraph, AddFilter, pSGF, L"Sample Grabber"); | |
1340 if(FAILED(hr)){ | |
1341 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: AddFilter(SampleGrabber) call failed. Error:0x%x\n", (unsigned int)hr); | |
1342 break; | |
1343 } | |
1344 hr = OLE_CALL_ARGS(priv->pBuilder, FindPin, (IUnknown *) pSGF,PINDIR_INPUT, NULL, NULL, FALSE, 0, &pSGIn); | |
1345 if(FAILED(hr)){ | |
1346 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: FindPin(pSGIn) call failed. Error:0x%x\n", (unsigned int)hr); | |
1347 break; | |
1348 } | |
1349 hr = OLE_CALL_ARGS(priv->pBuilder, FindPin, (IUnknown *) pSGF,PINDIR_OUTPUT, NULL, NULL, FALSE, 0, &pSGOut); | |
1350 if(FAILED(hr)){ | |
1351 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: FindPin(pSGOut) call failed. Error:0x%x\n", (unsigned int)hr); | |
1352 break; | |
1353 } | |
1354 | |
1355 /* creating ringbuffer for video samples */ | |
1356 priv->pCSGCB = CSampleGrabberCB_Create(pbuf); | |
1357 if(!priv->pCSGCB){ | |
1358 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: CSampleGrabberCB_Create(pbuf) call failed. Error:0x%x\n", (unsigned int)E_OUTOFMEMORY); | |
1359 break; | |
1360 } | |
1361 | |
1362 /* initializing SampleGrabber filter */ | |
1363 hr = OLE_QUERYINTERFACE(pSGF, IID_ISampleGrabber, pSG); | |
1364 if(FAILED(hr)){ | |
1365 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: QueryInterface(IID_ISampleGrabber) call failed. Error:0x%x\n", (unsigned int)hr); | |
1366 break; | |
1367 } | |
1368 hr = OLE_CALL_ARGS(pSG, SetMediaType, pmt); //set desired mediatype | |
1369 if(FAILED(hr)){ | |
1370 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: SetMediaType(pSG) call failed. Error:0x%x\n", (unsigned int)hr); | |
1371 break; | |
1372 } | |
1373 // hr = OLE_CALL_ARGS(pSG, SetCallback, (ISampleGrabberCB *) pCSGCB, 1); //we want to receive copy of sample's data | |
1374 hr = OLE_CALL_ARGS(pSG, SetCallback, (ISampleGrabberCB *) priv->pCSGCB, 0); //we want to receive sample | |
1375 | |
1376 if(FAILED(hr)){ | |
1377 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: SetCallback(pSG) call failed. Error:0x%x\n", (unsigned int)hr); | |
1378 break; | |
1379 } | |
1380 hr = OLE_CALL_ARGS(pSG, SetOneShot, FALSE); //... for all frames | |
1381 if(FAILED(hr)){ | |
1382 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: SetOneShot(pSG) call failed. Error:0x%x\n", (unsigned int)hr); | |
1383 break; | |
1384 } | |
1385 hr = OLE_CALL_ARGS(pSG, SetBufferSamples, FALSE); //... do not buffer samples in sample grabber | |
1386 if(FAILED(hr)){ | |
1387 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: SetBufferSamples(pSG) call failed. Error:0x%x\n", (unsigned int)hr); | |
1388 break; | |
1389 } | |
1390 OLE_RELEASE_SAFE(pSG); | |
1391 | |
1392 if(priv->tv_param->normalize_audio_chunks && !memcmp(&(pmt->majortype),&(MEDIATYPE_Audio),16)){ | |
1393 set_buffer_preference(20,(WAVEFORMATEX*)(pmt->pbFormat),pCapturePin,pSGIn); | |
25048 | 1394 } |
24744 | 1395 |
1396 /* connecting filters together: VideoCapture --> SampleGrabber */ | |
1397 hr = OLE_CALL_ARGS(priv->pGraph, Connect, pCapturePin, pSGIn); | |
1398 if(FAILED(hr)){ | |
1399 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: Unable to create pCapturePin<->pSGIn connection. Error:0x%x\n", (unsigned int)hr); | |
1400 break; | |
1401 } | |
25029
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1402 hr = OLE_CALL_ARGS(pCapturePin, ConnectionMediaType, &conn_mt); |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1403 if(FAILED(hr)) |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1404 { |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1405 mp_msg(MSGT_TV, MSGL_WARN, MSGTR_TVI_DS_GetActualMediatypeFailed, (unsigned int)hr); |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1406 } |
24744 | 1407 |
1408 if(priv->tv_param->hidden_video_renderer){ | |
1409 IEnumFilters* pEnum; | |
1410 IBaseFilter* pFilter; | |
1411 | |
1412 hr=OLE_CALL_ARGS(priv->pBuilder,RenderStream,NULL,NULL,(IUnknown*)pCapturePin,NULL,NULL); | |
1413 | |
1414 OLE_CALL_ARGS(priv->pGraph, EnumFilters, &pEnum); | |
1415 while (OLE_CALL_ARGS(pEnum, Next, 1, &pFilter, NULL) == S_OK) { | |
1416 LPVIDEOWINDOW pVideoWindow; | |
1417 hr = OLE_QUERYINTERFACE(pFilter, IID_IVideoWindow, pVideoWindow); | |
1418 if (SUCCEEDED(hr)) | |
1419 { | |
1420 OLE_CALL_ARGS(pVideoWindow,put_Visible,/* OAFALSE*/ 0); | |
1421 OLE_CALL_ARGS(pVideoWindow,put_AutoShow,/* OAFALSE*/ 0); | |
1422 OLE_RELEASE_SAFE(pVideoWindow); | |
1423 } | |
1424 OLE_RELEASE_SAFE(pFilter); | |
1425 } | |
1426 OLE_RELEASE_SAFE(pEnum); | |
1427 }else | |
1428 { | |
25052
7b2b17b57cf7
Disable terminating directshow chains with NullRenderer filter,
voroshil
parents:
25051
diff
changeset
|
1429 #if 0 |
7b2b17b57cf7
Disable terminating directshow chains with NullRenderer filter,
voroshil
parents:
25051
diff
changeset
|
1430 /* |
7b2b17b57cf7
Disable terminating directshow chains with NullRenderer filter,
voroshil
parents:
25051
diff
changeset
|
1431 Code below is disabled, because terminating chain with NullRenderer leads to jerky video. |
7b2b17b57cf7
Disable terminating directshow chains with NullRenderer filter,
voroshil
parents:
25051
diff
changeset
|
1432 Perhaps, this happens because NullRenderer filter discards each received |
7b2b17b57cf7
Disable terminating directshow chains with NullRenderer filter,
voroshil
parents:
25051
diff
changeset
|
1433 frame while discarded frames causes live source filter to dramatically reduce frame rate. |
7b2b17b57cf7
Disable terminating directshow chains with NullRenderer filter,
voroshil
parents:
25051
diff
changeset
|
1434 */ |
24744 | 1435 /* adding sink for video stream */ |
1436 hr = CoCreateInstance((GUID *) & CLSID_NullRenderer, NULL,CLSCTX_INPROC_SERVER, &IID_IBaseFilter,(void *) &pNR); | |
1437 if(FAILED(hr)){ | |
1438 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: CoCreateInstance(NullRenderer) call failed. Error:0x%x\n", (unsigned int)hr); | |
1439 break; | |
1440 } | |
1441 hr = OLE_CALL_ARGS(priv->pGraph, AddFilter, pNR, L"Null Renderer"); | |
1442 if(FAILED(hr)){ | |
1443 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: AddFilter(NullRenderer) call failed. Error:0x%x\n", (unsigned int)hr); | |
1444 break; | |
1445 } | |
1446 hr = OLE_CALL_ARGS(priv->pBuilder, FindPin, (IUnknown *) pNR,PINDIR_INPUT, NULL, NULL, FALSE, 0, &pNRIn); | |
1447 if(FAILED(hr)){ | |
1448 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: FindPin(pNRIn) call failed. Error:0x%x\n", (unsigned int)hr); | |
1449 break; | |
1450 } | |
1451 /* | |
1452 Prevent ending VBI chain with NullRenderer filter, because this causes VBI pin disconnection | |
1453 */ | |
1454 if(memcmp(&(pmt->majortype),&MEDIATYPE_VBI,16)){ | |
1455 /* connecting filters together: SampleGrabber --> NullRenderer */ | |
1456 hr = OLE_CALL_ARGS(priv->pGraph, Connect, pSGOut, pNRIn); | |
1457 if(FAILED(hr)){ | |
1458 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: Unable to create pSGOut<->pNRIn connection. Error:0x%x\n", (unsigned int)hr); | |
1459 break; | |
1460 } | |
1461 } | |
25052
7b2b17b57cf7
Disable terminating directshow chains with NullRenderer filter,
voroshil
parents:
25051
diff
changeset
|
1462 #endif |
24744 | 1463 } |
1464 | |
1465 hr = S_OK; | |
1466 } while(0); | |
25029
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1467 |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1468 FreeMediaType(pmt); |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1469 CopyMediaType(pmt, &conn_mt); |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1470 FreeMediaType(&conn_mt); |
c9f20e41bc13
Make sure that mplayer will receive actual media type
voroshil
parents:
25028
diff
changeset
|
1471 |
24744 | 1472 OLE_RELEASE_SAFE(pSGF); |
1473 OLE_RELEASE_SAFE(pSGIn); | |
1474 OLE_RELEASE_SAFE(pSGOut); | |
1475 OLE_RELEASE_SAFE(pNR); | |
1476 OLE_RELEASE_SAFE(pNRIn); | |
1477 OLE_RELEASE_SAFE(pCapturePin); | |
1478 | |
1479 return hr; | |
1480 } | |
1481 | |
1482 /** | |
1483 * \brief configures crossbar for grabbing video stream from given input | |
1484 * | |
1485 * \param priv driver's private data | |
1486 * \param input index of available video input to get data from | |
1487 * | |
1488 * \return TVI_CONTROL_TRUE success | |
1489 * \return TVI_CONTROL_FALSE error | |
1490 */ | |
1491 static int set_crossbar_input(priv_t * priv, int input) | |
1492 { | |
1493 HRESULT hr; | |
1494 int i, nVideoDecoder, nAudioDecoder; | |
1495 long lInput, lInputRelated, lRelated, lPhysicalType, lOutputPins, | |
1496 lInputPins; | |
1497 | |
1498 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: Configuring crossbar\n"); | |
1499 if (!priv->pCrossbar || input < 0 | |
1500 || input >= tv_available_inputs_count) | |
1501 return TVI_CONTROL_FALSE; | |
1502 | |
1503 OLE_CALL_ARGS(priv->pCrossbar, get_PinCounts, &lOutputPins, &lInputPins); | |
1504 | |
1505 lInput = tv_available_inputs[input]; | |
1506 | |
1507 if (lInput < 0 || lInput >= lInputPins) | |
1508 return TVI_CONTROL_FALSE; | |
1509 | |
1510 OLE_CALL_ARGS(priv->pCrossbar, get_CrossbarPinInfo, 1 /* input */ , lInput, | |
1511 &lInputRelated, &lPhysicalType); | |
1512 | |
1513 nVideoDecoder = nAudioDecoder = -1; | |
1514 for (i = 0; i < lOutputPins; i++) { | |
1515 OLE_CALL_ARGS(priv->pCrossbar, get_CrossbarPinInfo, 0 /*output */ , i, | |
1516 &lRelated, &lPhysicalType); | |
1517 if (lPhysicalType == PhysConn_Video_VideoDecoder) | |
1518 nVideoDecoder = i; | |
1519 if (lPhysicalType == PhysConn_Audio_AudioDecoder) | |
1520 nAudioDecoder = i; | |
1521 } | |
1522 if (nVideoDecoder >= 0) { | |
1523 //connecting given input with video decoder | |
1524 hr = OLE_CALL_ARGS(priv->pCrossbar, Route, nVideoDecoder, lInput); | |
1525 if (hr != S_OK) { | |
1526 mp_msg(MSGT_TV,MSGL_ERR,MSGTR_TVI_DS_UnableConnectInputVideoDecoder, (unsigned int)hr); | |
1527 return TVI_CONTROL_FALSE; | |
1528 } | |
1529 } | |
1530 if (nAudioDecoder >= 0 && lInputRelated >= 0) { | |
1531 hr = OLE_CALL_ARGS(priv->pCrossbar, Route, nAudioDecoder, | |
1532 lInputRelated); | |
1533 if (hr != S_OK) { | |
1534 mp_msg(MSGT_TV,MSGL_ERR,MSGTR_TVI_DS_UnableConnectInputAudioDecoder, (unsigned int)hr); | |
1535 return TVI_CONTROL_FALSE; | |
1536 } | |
1537 } | |
1538 return TVI_CONTROL_TRUE; | |
1539 } | |
1540 | |
1541 /** | |
1542 * \brief adjusts video control (hue,saturation,contrast,brightess) | |
1543 * | |
1544 * \param priv driver's private data | |
1545 * \param control which control to adjust | |
1546 * \param value new value for control (0-100) | |
1547 * | |
1548 * \return TVI_CONTROL_TRUE success | |
1549 * \return TVI_CONTROL_FALSE error | |
1550 */ | |
1551 static int set_control(priv_t * priv, int control, int value) | |
1552 { | |
1553 long lMin, lMax, lStepping, lDefault, lFlags, lValue; | |
1554 HRESULT hr; | |
1555 | |
1556 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: set_control called\n"); | |
1557 if (value < -100 || value > 100 || !priv->pVideoProcAmp) | |
1558 return TVI_CONTROL_FALSE; | |
1559 | |
1560 hr = OLE_CALL_ARGS(priv->pVideoProcAmp, GetRange, control, | |
1561 &lMin, &lMax, &lStepping, &lDefault, &lFlags); | |
1562 if (FAILED(hr) || lFlags != VideoProcAmp_Flags_Manual) | |
1563 return TVI_CONTROL_FALSE; | |
1564 | |
1565 lValue = lMin + (value + 100) * (lMax - lMin) / 200; | |
1566 /* | |
1567 Workaround for ATI AIW 7500. The driver reports: max=255, stepping=256 | |
1568 */ | |
1569 if (lStepping > lMax) { | |
1570 mp_msg(MSGT_TV, MSGL_DBG3, | |
1571 "tvi_dshow: Stepping (%ld) is bigger than max value (%ld) for control %d. Assuming 1\n", | |
1572 lStepping, lMax,control); | |
1573 lStepping = 1; | |
1574 } | |
1575 lValue -= lValue % lStepping; | |
1576 hr = OLE_CALL_ARGS(priv->pVideoProcAmp, Set, control, lValue, | |
1577 VideoProcAmp_Flags_Manual); | |
1578 if (FAILED(hr)) | |
1579 return TVI_CONTROL_FALSE; | |
1580 | |
1581 return TVI_CONTROL_TRUE; | |
1582 } | |
1583 | |
1584 /** | |
1585 * \brief get current value of video control (hue,saturation,contrast,brightess) | |
1586 * | |
1587 * \param priv driver's private data | |
1588 * \param control which control to adjust | |
1589 * \param pvalue address of variable thar receives current value | |
1590 * | |
1591 * \return TVI_CONTROL_TRUE success | |
1592 * \return TVI_CONTROL_FALSE error | |
1593 */ | |
1594 static int get_control(priv_t * priv, int control, int *pvalue) | |
1595 { | |
1596 long lMin, lMax, lStepping, lDefault, lFlags, lValue; | |
1597 HRESULT hr; | |
1598 | |
1599 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: get_control called\n"); | |
1600 if (!pvalue || !priv->pVideoProcAmp) | |
1601 return TVI_CONTROL_FALSE; | |
1602 | |
1603 hr = OLE_CALL_ARGS(priv->pVideoProcAmp, GetRange, control, | |
1604 &lMin, &lMax, &lStepping, &lDefault, &lFlags); | |
1605 if (FAILED(hr)) | |
1606 return TVI_CONTROL_FALSE; | |
1607 if (lMin == lMax) { | |
1608 *pvalue = lMin; | |
1609 return TVI_CONTROL_TRUE; | |
1610 } | |
1611 | |
1612 hr = OLE_CALL_ARGS(priv->pVideoProcAmp, Get, control, &lValue, &lFlags); | |
1613 if (FAILED(hr)) | |
1614 return TVI_CONTROL_FALSE; | |
1615 | |
1616 *pvalue = 200 * (lValue - lMin) / (lMax - lMin) - 100; | |
1617 | |
1618 return TVI_CONTROL_TRUE; | |
1619 } | |
1620 | |
1621 /** | |
25053
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1622 * \brief create AM_MEDIA_TYPE structure, corresponding to given FourCC code and width/height/fps |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1623 * \param fcc FourCC code for video format |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1624 * \param width picture width |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1625 * \param height pciture height |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1626 * \param fps frames per second (required for bitrate calculation) |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1627 * |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1628 * \return pointer to AM_MEDIA_TYPE structure if success, NULL - otherwise |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1629 */ |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1630 static AM_MEDIA_TYPE* create_video_format(int fcc, int width, int height, int fps) |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1631 { |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1632 int i; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1633 AM_MEDIA_TYPE mt; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1634 VIDEOINFOHEADER vHdr; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1635 |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1636 /* Check given fcc in lookup table*/ |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1637 for(i=0; img_fmt_list[i].fmt && img_fmt_list[i].fmt!=fcc; i++) /* NOTHING */; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1638 if(!img_fmt_list[i].fmt) |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1639 return NULL; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1640 |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1641 memset(&mt, 0, sizeof(AM_MEDIA_TYPE)); |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1642 memset(&vHdr, 0, sizeof(VIDEOINFOHEADER)); |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1643 |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1644 vHdr.bmiHeader.biSize = sizeof(vHdr.bmiHeader); |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1645 vHdr.bmiHeader.biWidth = width; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1646 vHdr.bmiHeader.biHeight = height; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1647 //FIXME: is biPlanes required too? |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1648 //vHdr.bmiHeader.biPlanes = img_fmt_list[i].nPlanes; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1649 vHdr.bmiHeader.biBitCount = img_fmt_list[i].nBits; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1650 vHdr.bmiHeader.biCompression = img_fmt_list[i].nCompression; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1651 vHdr.bmiHeader.biSizeImage = width * height * img_fmt_list[i].nBits / 8; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1652 vHdr.dwBitRate = vHdr.bmiHeader.biSizeImage * 8 * fps; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1653 |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1654 mt.pbFormat = (char*)&vHdr; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1655 mt.cbFormat = sizeof(vHdr); |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1656 |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1657 mt.majortype = MEDIATYPE_Video; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1658 mt.subtype = *img_fmt_list[i].subtype; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1659 mt.formattype = FORMAT_VideoInfo; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1660 |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1661 mt.bFixedSizeSamples = 1; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1662 mt.bTemporalCompression = 0; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1663 mt.lSampleSize = vHdr.bmiHeader.biSizeImage; |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1664 |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1665 return CreateMediaType(&mt); |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1666 } |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1667 |
086c2accaaa2
Service routine for constructing AM_MEDIA_TYPE structure from
voroshil
parents:
25052
diff
changeset
|
1668 /** |
24744 | 1669 * \brief extracts fcc,width,height from AM_MEDIA_TYPE |
1670 * | |
1671 * \param pmt pointer to AM_MEDIA_TYPE to extract data from | |
1672 * \param pfcc address of variable that receives FourCC | |
1673 * \param pwidth address of variable that receives width | |
1674 * \param pheight address of variable that recevies height | |
1675 * | |
1676 * \return 1 if data extracted successfully, 0 - otherwise | |
1677 */ | |
1678 static int extract_video_format(AM_MEDIA_TYPE * pmt, int *pfcc, | |
1679 int *pwidth, int *pheight) | |
1680 { | |
1681 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: extract_video_format called\n"); | |
1682 if (!pmt) | |
1683 return 0; | |
1684 if (!pmt->pbFormat) | |
1685 return 0; | |
1686 if (memcmp(&(pmt->formattype), &FORMAT_VideoInfo, 16) != 0) | |
1687 return 0; | |
1688 if (pfcc) | |
1689 *pfcc = subtype2imgfmt(&(pmt->subtype)); | |
1690 if (pwidth) | |
1691 *pwidth = ((VIDEOINFOHEADER *) pmt->pbFormat)->bmiHeader.biWidth; | |
1692 if (pheight) | |
1693 *pheight = ((VIDEOINFOHEADER *) pmt->pbFormat)->bmiHeader.biHeight; | |
1694 return 1; | |
1695 } | |
1696 | |
1697 /** | |
1698 * \brief extracts samplerate,bits,channels from AM_MEDIA_TYPE | |
1699 * | |
1700 * \param pmt pointer to AM_MEDIA_TYPE to extract data from | |
1701 * \param pfcc address of variable that receives samplerate | |
1702 * \param pwidth address of variable that receives number of bits per sample | |
1703 * \param pheight address of variable that recevies number of channels | |
1704 * | |
1705 * \return 1 if data extracted successfully, 0 - otherwise | |
1706 */ | |
1707 static int extract_audio_format(AM_MEDIA_TYPE * pmt, int *psamplerate, | |
1708 int *pbits, int *pchannels) | |
1709 { | |
1710 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: extract_audio_format called\n"); | |
1711 if (!pmt) | |
1712 return 0; | |
1713 if (!pmt->pbFormat) | |
1714 return 0; | |
1715 if (memcmp(&(pmt->formattype), &FORMAT_WaveFormatEx, 16) != 0) | |
1716 return 0; | |
1717 if (psamplerate) | |
1718 *psamplerate = ((WAVEFORMATEX *) pmt->pbFormat)->nSamplesPerSec; | |
1719 if (pbits) | |
1720 *pbits = ((WAVEFORMATEX *) pmt->pbFormat)->wBitsPerSample; | |
1721 if (pchannels) | |
1722 *pchannels = ((WAVEFORMATEX *) pmt->pbFormat)->nChannels; | |
1723 return 1; | |
1724 } | |
1725 | |
1726 /** | |
1727 * \brief checks if AM_MEDIA_TYPE compatible with given samplerate,bits,channels | |
1728 * | |
1729 * \param pmt pointer to AM_MEDIA_TYPE for check | |
1730 * \param samplerate audio samplerate | |
1731 * \param bits bits per sample | |
1732 * \param channels number of audio channels | |
1733 * | |
1734 * \return 1 if AM_MEDIA_TYPE compatible | |
1735 * \return 0 if not | |
1736 */ | |
1737 static int check_audio_format(AM_MEDIA_TYPE * pmt, int samplerate, | |
1738 int bits, int channels) | |
1739 { | |
1740 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: check_audio_format called\n"); | |
1741 if (!pmt) | |
1742 return 0; | |
1743 if (memcmp(&(pmt->majortype), &MEDIATYPE_Audio, 16) != 0) | |
1744 return 0; | |
1745 if (memcmp(&(pmt->subtype), &MEDIASUBTYPE_PCM, 16) != 0) | |
1746 return 0; | |
1747 if (memcmp(&(pmt->formattype), &FORMAT_WaveFormatEx, 16) != 0) | |
1748 return 0; | |
1749 if (!pmt->pbFormat) | |
1750 return 0; | |
1751 if (((WAVEFORMATEX *) pmt->pbFormat)->nSamplesPerSec != samplerate) | |
1752 return 0; | |
1753 if (((WAVEFORMATEX *) pmt->pbFormat)->wBitsPerSample != bits) | |
1754 return 0; | |
1755 if (channels > 0 | |
1756 && ((WAVEFORMATEX *) pmt->pbFormat)->nChannels != channels) | |
1757 return 0; | |
1758 | |
1759 return 1; | |
1760 } | |
1761 | |
1762 /** | |
1763 * \brief checks if AM_MEDIA_TYPE compatible with given fcc,width,height | |
1764 * | |
1765 * \param pmt pointer to AM_MEDIA_TYPE for check | |
1766 * \param fcc FourCC (compression) | |
1767 * \param width width of picture | |
1768 * \param height height of picture | |
1769 * | |
1770 * \return 1 if AM_MEDIA_TYPE compatible | |
1771 & \return 0 if not | |
1772 * | |
1773 * \note | |
1774 * width and height are currently not used | |
1775 * | |
1776 * \todo | |
1777 * add width/height check | |
1778 */ | |
1779 static int check_video_format(AM_MEDIA_TYPE * pmt, int fcc, int width, | |
1780 int height) | |
1781 { | |
1782 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: check_video_format called\n"); | |
1783 if (!pmt) | |
1784 return 0; | |
1785 if (memcmp(&(pmt->majortype), &MEDIATYPE_Video, 16) != 0) | |
1786 return 0; | |
1787 if (subtype2imgfmt(&(pmt->subtype)) != fcc) | |
1788 return 0; | |
1789 return 1; | |
1790 } | |
1791 | |
1792 /** | |
1793 * \brief converts DirectShow subtype to MPlayer's IMGFMT | |
1794 * | |
1795 * \param subtype DirectShow subtype for video format | |
1796 * | |
1797 * \return MPlayer's IMGFMT or 0 if error occured | |
1798 */ | |
1799 static int subtype2imgfmt(const GUID * subtype) | |
1800 { | |
1801 int i; | |
1802 for (i = 0; img_fmt_list[i].fmt; i++) { | |
1803 if (memcmp(subtype, img_fmt_list[i].subtype, 16) == 0) | |
1804 return img_fmt_list[i].fmt; | |
1805 } | |
1806 return 0; | |
1807 } | |
1808 | |
1809 /** | |
1810 * \brief prints filter name and it pins | |
1811 * | |
1812 * \param pFilter - IBaseFilter to get data from | |
1813 * | |
1814 * \return S_OK if success, error code otherwise | |
1815 */ | |
1816 static HRESULT show_filter_info(IBaseFilter * pFilter) | |
1817 { | |
1818 char tmp[200]; | |
1819 FILTER_INFO fi; | |
1820 LPENUMPINS pEnum = 0; | |
1821 IPin *pPin = 0; | |
1822 PIN_DIRECTION ThisPinDir; | |
1823 PIN_INFO pi; | |
1824 HRESULT hr; | |
1825 int i; | |
1826 | |
1827 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: show_filter_info called\n"); | |
1828 memset(&fi, 0, sizeof(fi)); | |
1829 memset(tmp, 0, 200); | |
1830 | |
1831 OLE_CALL_ARGS(pFilter, QueryFilterInfo, &fi); | |
1832 OLE_RELEASE_SAFE(fi.pGraph); | |
1833 wtoa(fi.achName, tmp, 200); | |
1834 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: BaseFilter (%p): Name=%s, Graph=%p output pins:", | |
1835 pFilter, tmp, fi.pGraph); | |
1836 hr = OLE_CALL_ARGS(pFilter, EnumPins, &pEnum); | |
1837 if (FAILED(hr)) | |
1838 return hr; | |
1839 i = 0; | |
1840 while (OLE_CALL_ARGS(pEnum, Next, 1, &pPin, NULL) == S_OK) { | |
1841 memset(&pi, 0, sizeof(pi)); | |
1842 memset(tmp, 0, 200); | |
1843 OLE_CALL_ARGS(pPin, QueryDirection, &ThisPinDir); | |
1844 if (ThisPinDir == PINDIR_OUTPUT) { | |
1845 OLE_CALL_ARGS(pPin, QueryPinInfo, &pi); | |
1846 wtoa(pi.achName, tmp, 200); | |
1847 OLE_RELEASE_SAFE(pi.pFilter); | |
1848 mp_msg(MSGT_TV, MSGL_DBG2, " %d=%s", i, tmp); | |
1849 mp_msg(MSGT_TV, MSGL_DBG3, " (%p)", pPin); | |
1850 mp_msg(MSGT_TV, MSGL_DBG2, ";"); | |
1851 OLE_RELEASE_SAFE(pPin); | |
1852 i++; | |
1853 } | |
1854 } | |
1855 mp_msg(MSGT_TV, MSGL_DBG2, "\n"); | |
1856 OLE_RELEASE_SAFE(pEnum); | |
1857 return S_OK; | |
1858 } | |
1859 | |
1860 /** | |
1861 * \brief gets device's frendly in ANSI encoding | |
1862 * | |
1863 * \param pM IMoniker interface, got in enumeration process | |
1864 * \param category device category | |
1865 * | |
1866 * \return TVI_CONTROL_TRUE if operation succeded, TVI_CONTROL_FALSE - otherwise | |
1867 */ | |
1868 static int get_device_name(IMoniker * pM, char *pBuf, int nLen) | |
1869 { | |
1870 HRESULT hr; | |
1871 VARIANT var; | |
1872 IPropertyBag *pPropBag; | |
1873 hr = OLE_CALL_ARGS(pM, BindToStorage, 0, 0, &IID_IPropertyBag,(void *) &pPropBag); | |
1874 if (FAILED(hr)) { | |
1875 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Call to BindToStorage failed\n"); | |
1876 return TVI_CONTROL_FALSE; | |
1877 } | |
1878 var.vt = VT_BSTR; | |
1879 hr = OLE_CALL_ARGS(pPropBag, Read, L"Description", (LPVARIANT) & var, | |
1880 NULL); | |
1881 if (FAILED(hr)) { | |
1882 hr = OLE_CALL_ARGS(pPropBag, Read, L"FriendlyName", (LPVARIANT) & var, | |
1883 NULL); | |
1884 } | |
1885 OLE_RELEASE_SAFE(pPropBag); | |
1886 if (SUCCEEDED(hr)) { | |
1887 wtoa(var.bstrVal, pBuf, nLen); | |
1888 return TVI_CONTROL_TRUE; | |
1889 } | |
1890 return TVI_CONTROL_FALSE; | |
1891 } | |
1892 | |
1893 /** | |
1894 * \brief find capture device at given index | |
1895 * | |
1896 * \param index device index to search for (-1 mean only print available) | |
1897 * \param category device category | |
1898 * | |
1899 * \return IBaseFilter interface for capture device with given index | |
1900 * | |
1901 * Sample values for category: | |
1902 * CLSID_VideoInputDeviceCategory - Video Capture Sources | |
1903 * CLSID_AudioInputDeviceCategory - Audio Capture Sources | |
1904 * See DirectShow SDK documentation for other possible values | |
1905 */ | |
1906 static IBaseFilter *find_capture_device(int index, REFCLSID category) | |
1907 { | |
1908 IBaseFilter *pFilter = NULL; | |
1909 ICreateDevEnum *pDevEnum = NULL; | |
1910 IEnumMoniker *pClassEnum = NULL; | |
1911 IMoniker *pM; | |
1912 HRESULT hr; | |
1913 ULONG cFetched; | |
1914 int i; | |
1915 char tmp[DEVICE_NAME_MAX_LEN + 1]; | |
1916 hr = CoCreateInstance((GUID *) & CLSID_SystemDeviceEnum, NULL, | |
1917 CLSCTX_INPROC_SERVER, &IID_ICreateDevEnum, | |
1918 (void *) &pDevEnum); | |
1919 if (FAILED(hr)) { | |
1920 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Unable to create device enumerator\n"); | |
1921 return NULL; | |
1922 } | |
1923 | |
1924 hr = OLE_CALL_ARGS(pDevEnum, CreateClassEnumerator, category, &pClassEnum, 0); | |
1925 OLE_RELEASE_SAFE(pDevEnum); | |
1926 if (FAILED(hr)) { | |
1927 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Unable to create class enumerator\n"); | |
1928 return NULL; | |
1929 } | |
1930 if (hr == S_FALSE) { | |
1931 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: No capture devices found\n"); | |
1932 return NULL; | |
1933 } | |
1934 | |
1935 OLE_CALL(pClassEnum,Reset); | |
1936 for (i = 0; OLE_CALL_ARGS(pClassEnum, Next, 1, &pM, &cFetched) == S_OK; i++) { | |
1937 if(get_device_name(pM, tmp, DEVICE_NAME_MAX_LEN)!=TVI_CONTROL_TRUE) | |
1938 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_UnableGetDeviceName, i); | |
1939 else | |
1940 mp_msg(MSGT_TV, MSGL_V, MSGTR_TVI_DS_DeviceName, i, tmp); | |
1941 if (index != -1 && i == index) { | |
1942 mp_msg(MSGT_TV, MSGL_INFO, MSGTR_TVI_DS_UsingDevice, index, tmp); | |
1943 hr = OLE_CALL_ARGS(pM, BindToObject, 0, 0, &IID_IBaseFilter,(void *) &pFilter); | |
1944 if (FAILED(hr)) | |
1945 pFilter = NULL; | |
1946 } | |
1947 OLE_RELEASE_SAFE(pM); | |
1948 } | |
1949 if (index != -1 && !pFilter) { | |
1950 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_DeviceNotFound, | |
1951 index); | |
1952 } | |
1953 OLE_RELEASE_SAFE(pClassEnum); | |
1954 | |
1955 return pFilter; | |
1956 } | |
1957 | |
1958 /** | |
1959 * \brief get array of available formats through call to IAMStreamConfig::GetStreamCaps | |
1960 * | |
1961 * \param[in] pVideoStreamConfig IAMStreamConfig of used capture device's output pin | |
1962 * \param[in] pMediaType MEDIATYPE_Video or MEDIATYPE_Audio | |
1963 * \param[out] parpmtMedia address of variable that receives pointer to array of AM_MEDIA_TYPE structures | |
1964 * \param[out] parCaps address of variable thar receives pointer to array of VIDEOSTREAM_CONFIG_CAPS or AUDIO_STREAM_CONFIG_CAPS | |
1965 * | |
1966 * \return S_OK success | |
1967 * \return E_POINTER one of parameters is NULL | |
1968 * \return E_FAIL required size of buffer is unknown for given media type | |
1969 * \return E_OUTOFMEMORY not enough memory | |
1970 * \return other error code from called methods | |
1971 * | |
1972 * \remarks | |
1973 * last item or returned arrays will be NULL | |
1974 */ | |
1975 static HRESULT get_available_formats_stream(IAMStreamConfig * | |
1976 pStreamConfig, | |
1977 const GUID * pMediaType, | |
1978 AM_MEDIA_TYPE *** parpmtMedia, | |
1979 void ***parCaps) | |
1980 { | |
1981 AM_MEDIA_TYPE **arpmt; | |
1982 void **pBuf=NULL; | |
1983 | |
1984 HRESULT hr; | |
1985 int i, count, size; | |
1986 int done; | |
1987 | |
1988 mp_msg(MSGT_TV, MSGL_DBG4, | |
1989 "tvi_dshow: get_available_formats_stream called\n"); | |
1990 | |
1991 if (!pStreamConfig || !pMediaType || !parpmtMedia || !parCaps) | |
1992 return E_POINTER; | |
1993 | |
1994 hr=OLE_CALL_ARGS(pStreamConfig, GetNumberOfCapabilities, &count, &size); | |
1995 if (FAILED(hr)) { | |
1996 mp_msg(MSGT_TV, MSGL_DBG4, | |
1997 "tvi_dshow: Call to GetNumberOfCapabilities failed (get_available_formats_stream)\n"); | |
1998 return hr; | |
1999 } | |
2000 if (memcmp(pMediaType, &MEDIATYPE_Video, 16) == 0){ | |
2001 if (size != sizeof(VIDEO_STREAM_CONFIG_CAPS)) { | |
2002 mp_msg(MSGT_TV, MSGL_DBG4, | |
2003 "tvi_dshow: Wrong video structure size for GetNumberOfCapabilities (get_available_formats_stream)\n"); | |
2004 return E_FAIL; | |
2005 } else if (memcmp(pMediaType, &MEDIATYPE_Audio, 16) == 0){ | |
2006 if (size != sizeof(AUDIO_STREAM_CONFIG_CAPS)) { | |
2007 mp_msg(MSGT_TV, MSGL_DBG4, | |
2008 "tvi_dshow: Wrong audio structure size for GetNumberOfCapabilities (get_available_formats_stream)\n"); | |
2009 return E_FAIL; | |
2010 } else { | |
2011 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_UnsupportedMediaType,"get_available_formats_stream"); | |
2012 return E_FAIL; | |
2013 } | |
2014 } | |
2015 } | |
2016 done = 0; | |
2017 | |
2018 arpmt = (AM_MEDIA_TYPE **) malloc((count + 1) * sizeof(AM_MEDIA_TYPE *)); | |
2019 if (arpmt) { | |
2020 memset(arpmt, 0, (count + 1) * sizeof(AM_MEDIA_TYPE *)); | |
2021 | |
2022 pBuf = (void **) malloc((count + 1) * sizeof(void *)); | |
2023 if (pBuf) { | |
2024 memset(pBuf, 0, (count + 1) * sizeof(void *)); | |
2025 | |
2026 for (i = 0; i < count; i++) { | |
2027 pBuf[i] = malloc(size); | |
2028 | |
2029 if (!pBuf[i]) | |
2030 break; | |
2031 | |
2032 hr = OLE_CALL_ARGS(pStreamConfig, GetStreamCaps, i, | |
2033 &(arpmt[i]), pBuf[i]); | |
2034 if (FAILED(hr)) | |
2035 break; | |
2036 } | |
2037 if (i == count) { | |
2038 *parpmtMedia = arpmt; | |
2039 *parCaps = pBuf; | |
2040 done = 1; | |
2041 } | |
2042 } | |
2043 } | |
2044 if (!done) { | |
2045 for (i = 0; i < count; i++) { | |
2046 if (pBuf && pBuf[i]) | |
2047 free(pBuf[i]); | |
2048 if (arpmt && arpmt[i]) | |
2049 DeleteMediaType(arpmt[i]); | |
2050 } | |
2051 if (pBuf) | |
2052 free(pBuf); | |
2053 if (arpmt) | |
2054 free(arpmt); | |
2055 if (hr != S_OK) { | |
2056 mp_msg(MSGT_TV, MSGL_DBG4, "tvi_dshow: Call to GetStreamCaps failed (get_available_formats_stream)\n"); | |
2057 return hr; | |
2058 } else | |
2059 return E_OUTOFMEMORY; | |
2060 } | |
2061 return S_OK; | |
2062 } | |
2063 | |
2064 /** | |
2065 * \brief returns allocates an array and store available media formats for given pin type to it | |
2066 * | |
2067 * \param pBuilder ICaptureGraphBuilder2 interface of graph builder | |
2068 * \param pFilter IBaseFilter interface of capture device | |
2069 * \param pMediaType media type for search (MEDIATYPE_Video or MEDIATYPE_Audio) | |
2070 * \param[out] parpmtMediaType address of variable that receives pointer to array | |
2071 * | |
2072 * \return S_OK success | |
2073 * \return E_POINTER one of given pointers is null | |
2074 * \return apropriate error code otherwise | |
2075 */ | |
2076 static HRESULT get_available_formats_pin(ICaptureGraphBuilder2 * pBuilder, | |
2077 IBaseFilter * pFilter, | |
2078 const GUID * pMediaType, | |
2079 AM_MEDIA_TYPE *** parpmtMedia, | |
2080 void ***parCaps) | |
2081 { | |
2082 IEnumMediaTypes *pEnum; | |
2083 IPin *pPin; | |
2084 int i, count, size; | |
2085 ULONG cFetched; | |
2086 AM_MEDIA_TYPE *pmt; | |
2087 HRESULT hr; | |
2088 void **pBuf; | |
2089 AM_MEDIA_TYPE **arpmt; //This will be real array | |
2090 int isvideo; | |
2091 VIDEO_STREAM_CONFIG_CAPS *pVideoCaps; | |
2092 AUDIO_STREAM_CONFIG_CAPS *pAudioCaps; | |
2093 int p1, p2, p3; | |
2094 | |
2095 mp_msg(MSGT_TV, MSGL_DBG4, | |
2096 "tvi_dshow: get_available_formats_pin called\n"); | |
2097 if (!pBuilder || !pFilter || !pMediaType || !parpmtMedia) | |
2098 return E_POINTER; | |
2099 | |
2100 hr = OLE_CALL_ARGS(pBuilder, FindPin, (IUnknown *) pFilter, | |
2101 PINDIR_OUTPUT, NULL, pMediaType, FALSE, 0, &pPin); | |
2102 if (FAILED(hr)) { | |
2103 mp_msg(MSGT_TV, MSGL_DBG4, | |
2104 "tvi_dshow: Call to FindPin failed (get_available_formats_pin)\n"); | |
2105 return hr; | |
2106 } | |
2107 if (memcmp(pMediaType, &MEDIATYPE_Video, 16) == 0) { | |
2108 isvideo = 1; | |
2109 size = sizeof(VIDEO_STREAM_CONFIG_CAPS); | |
2110 } else if (memcmp(pMediaType, &MEDIATYPE_Audio, 16) == 0) { | |
2111 isvideo = 0; | |
2112 size = sizeof(AUDIO_STREAM_CONFIG_CAPS); | |
2113 } else { | |
2114 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_UnsupportedMediaType,"get_available_formats_pin"); | |
2115 return E_FAIL; | |
2116 } | |
2117 | |
2118 hr = OLE_CALL_ARGS(pPin, EnumMediaTypes, &pEnum); | |
2119 OLE_RELEASE_SAFE(pPin); | |
2120 if (FAILED(hr)) { | |
2121 mp_msg(MSGT_TV, MSGL_DBG4, | |
2122 "tvi_dshow: Call to EnumMediaTypes failed (get_available_formats_pin)\n"); | |
2123 return hr; | |
2124 } | |
2125 for (i = 0; OLE_CALL_ARGS(pEnum, Next, 1, &pmt, &cFetched) == S_OK; i++) { | |
2126 if (!pmt) | |
2127 break; | |
2128 } | |
2129 OLE_CALL(pEnum,Reset); | |
2130 | |
2131 count = i; | |
2132 arpmt = | |
2133 (AM_MEDIA_TYPE **) malloc((count + 1) * sizeof(AM_MEDIA_TYPE *)); | |
2134 if (!arpmt) | |
2135 return E_OUTOFMEMORY; | |
2136 memset(arpmt, 0, (count + 1) * sizeof(AM_MEDIA_TYPE *)); | |
2137 | |
2138 for (i = 0; | |
2139 i < count | |
2140 && OLE_CALL_ARGS(pEnum, Next, 1, &(arpmt[i]), &cFetched) == S_OK; | |
2141 i++); | |
2142 | |
2143 OLE_RELEASE_SAFE(pEnum); | |
2144 | |
2145 | |
2146 pBuf = (void **) malloc((count + 1) * sizeof(void *)); | |
2147 if (!pBuf) { | |
2148 for (i = 0; i < count; i++) | |
2149 if (arpmt[i]) | |
2150 DeleteMediaType(arpmt[i]); | |
2151 free(arpmt); | |
2152 return E_OUTOFMEMORY; | |
2153 } | |
2154 memset(pBuf, 0, (count + 1) * sizeof(void *)); | |
2155 | |
2156 for (i = 0; i < count; i++) { | |
2157 pBuf[i] = malloc(size); | |
2158 if (!pBuf[i]) | |
2159 break; | |
2160 memset(pBuf[i], 0, size); | |
2161 | |
2162 if (isvideo) { | |
2163 pVideoCaps = (VIDEO_STREAM_CONFIG_CAPS *) pBuf[i]; | |
2164 extract_video_format(arpmt[i], NULL, &p1, &p2); | |
2165 pVideoCaps->MaxOutputSize.cx = pVideoCaps->MinOutputSize.cx = | |
2166 p1; | |
2167 pVideoCaps->MaxOutputSize.cy = pVideoCaps->MinOutputSize.cy = | |
2168 p2; | |
2169 } else { | |
2170 pAudioCaps = (AUDIO_STREAM_CONFIG_CAPS *) pBuf[i]; | |
2171 extract_audio_format(arpmt[i], &p1, &p2, &p3); | |
2172 pAudioCaps->MaximumSampleFrequency = | |
2173 pAudioCaps->MinimumSampleFrequency = p1; | |
2174 pAudioCaps->MaximumBitsPerSample = | |
2175 pAudioCaps->MinimumBitsPerSample = p2; | |
2176 pAudioCaps->MaximumChannels = pAudioCaps->MinimumChannels = p3; | |
2177 } | |
2178 | |
2179 } | |
2180 if (i != count) { | |
2181 for (i = 0; i < count; i++) { | |
2182 if (arpmt[i]) | |
2183 DeleteMediaType(arpmt[i]); | |
2184 if (pBuf[i]) | |
2185 free(pBuf[i]); | |
2186 } | |
2187 free(arpmt); | |
2188 free(pBuf); | |
2189 return E_OUTOFMEMORY; | |
2190 } | |
2191 *parpmtMedia = arpmt; | |
2192 *parCaps = pBuf; | |
2193 | |
2194 return S_OK; | |
2195 } | |
2196 | |
2197 /* | |
2198 *--------------------------------------------------------------------------------------- | |
2199 * | |
2200 * Public methods | |
2201 * | |
2202 *--------------------------------------------------------------------------------------- | |
2203 */ | |
2204 /** | |
2205 * \brief fills given buffer with audio data (usually one block) | |
2206 * | |
2207 * \param priv driver's private data structure | |
2208 * \param buffer buffer to store data to | |
2209 * \param len buffer's size in bytes (usually one block size) | |
2210 * | |
2211 * \return audio pts if audio present, 1 - otherwise | |
2212 */ | |
2213 static double grab_audio_frame(priv_t * priv, char *buffer, int len) | |
2214 { | |
2215 int bytes = 0; | |
2216 int i; | |
2217 double pts; | |
2218 grabber_ringbuffer_t *rb = priv->a_buf; | |
2219 grabber_ringbuffer_t *vrb = priv->v_buf; | |
2220 | |
2221 if (!rb || !rb->ringbuffer) | |
2222 return 1; | |
2223 | |
2224 if(vrb && vrb->tStart<0){ | |
2225 memset(buffer,0,len); | |
2226 return 0; | |
2227 } | |
2228 if(vrb && rb->tStart<0) | |
2229 rb->tStart=vrb->tStart; | |
2230 | |
2231 if (len < rb->blocksize) | |
2232 bytes = len; | |
2233 else | |
2234 bytes = rb->blocksize; | |
2235 | |
2236 mp_msg(MSGT_TV, MSGL_DBG3,"tvi_dshow: FillBuffer (audio) called. %d blocks in buffer, %d bytes requested\n", | |
2237 rb->count, len); | |
2238 if(!rb->count){ | |
2239 mp_msg(MSGT_TV,MSGL_DBG4,"tvi_dshow: waiting for frame\n"); | |
2240 for(i=0;i<1000 && !rb->count;i++) usec_sleep(1000); | |
2241 if(!rb->count){ | |
2242 mp_msg(MSGT_TV,MSGL_DBG4,"tvi_dshow: waiting timeout\n"); | |
2243 return 0; | |
2244 } | |
2245 mp_msg(MSGT_TV,MSGL_DBG4,"tvi_dshow: got frame!\n"); | |
2246 } | |
2247 | |
2248 EnterCriticalSection(rb->pMutex); | |
2249 pts=rb->dpts[rb->head]-rb->tStart; | |
2250 memcpy(buffer, rb->ringbuffer[rb->head], bytes); | |
2251 rb->head = (rb->head + 1) % rb->buffersize; | |
2252 rb->count--; | |
2253 LeaveCriticalSection(rb->pMutex); | |
2254 return pts; | |
2255 } | |
2256 | |
2257 /** | |
2258 * \brief returns audio frame size | |
2259 * | |
2260 * \param priv driver's private data structure | |
2261 * | |
2262 * \return audio block size if audio enabled and 1 - otherwise | |
2263 */ | |
2264 static int get_audio_framesize(priv_t * priv) | |
2265 { | |
2266 if (!priv->a_buf) | |
2267 return 1; //no audio | |
2268 mp_msg(MSGT_TV,MSGL_DBG3,"get_audio_framesize: %d\n",priv->a_buf->blocksize); | |
2269 return priv->a_buf->blocksize; | |
2270 } | |
2271 | |
2272 #ifdef HAVE_TV_TELETEXT | |
2273 static int vbi_get_props(priv_t* priv,tt_stream_props* ptsp) | |
2274 { | |
2275 if(!priv || !ptsp) | |
2276 return TVI_CONTROL_FALSE; | |
2277 | |
2278 //STUBS!!! | |
2279 ptsp->interlaced=0; | |
2280 ptsp->offset=256; | |
2281 | |
2282 ptsp->sampling_rate=27e6; | |
2283 ptsp->samples_per_line=720; | |
2284 | |
2285 ptsp->count[0]=16; | |
2286 ptsp->count[1]=16; | |
2287 //END STUBS!!! | |
2288 ptsp->bufsize = ptsp->samples_per_line * (ptsp->count[0] + ptsp->count[1]); | |
2289 | |
2290 mp_msg(MSGT_TV,MSGL_V,"vbi_get_props: sampling_rate=%d,offset:%d,samples_per_line: %d\n interlaced:%s, count=[%d,%d]\n", | |
2291 ptsp->sampling_rate, | |
2292 ptsp->offset, | |
2293 ptsp->samples_per_line, | |
2294 ptsp->interlaced?"Yes":"No", | |
2295 ptsp->count[0], | |
2296 ptsp->count[1]); | |
2297 | |
2298 return TVI_CONTROL_TRUE; | |
2299 } | |
2300 | |
2301 static void vbi_grabber(priv_t* priv) | |
2302 { | |
2303 grabber_ringbuffer_t *rb = priv->vbi_buf; | |
2304 int i; | |
2305 unsigned char* buf; | |
2306 if (!rb || !rb->ringbuffer) | |
2307 return; | |
2308 | |
2309 buf=calloc(1,rb->blocksize); | |
2310 for(i=0; i<23 && rb->count; i++){ | |
2311 memcpy(buf,rb->ringbuffer[rb->head],rb->blocksize); | |
2312 teletext_control(priv->priv_vbi,TV_VBI_CONTROL_DECODE_PAGE,&buf); | |
2313 rb->head = (rb->head + 1) % rb->buffersize; | |
2314 rb->count--; | |
2315 } | |
2316 free(buf); | |
2317 } | |
2318 #endif //HAVE_TV_TELETEXT | |
2319 | |
2320 /** | |
2321 * \brief fills given buffer with video data (usually one frame) | |
2322 * | |
2323 * \param priv driver's private data structure | |
2324 * \param buffer buffer to store data to | |
2325 * \param len buffer's size in bytes (usually one frame size) | |
2326 * | |
2327 * \return frame size if video present, 0 - otherwise | |
2328 */ | |
2329 static double grab_video_frame(priv_t * priv, char *buffer, int len) | |
2330 { | |
2331 int bytes = 0; | |
2332 int i; | |
2333 double pts; | |
2334 grabber_ringbuffer_t *rb = priv->v_buf; | |
2335 | |
2336 if (!rb || !rb->ringbuffer) | |
2337 return 1; | |
2338 if (len < rb->blocksize) | |
2339 bytes = len; | |
2340 else | |
2341 bytes = rb->blocksize; | |
2342 | |
2343 mp_msg(MSGT_TV, MSGL_DBG3,"tvi_dshow: FillBuffer (video) called. %d blocks in buffer, %d bytes requested\n", | |
2344 rb->count, len); | |
2345 if(!rb->count){ | |
2346 mp_msg(MSGT_TV,MSGL_DBG4,"tvi_dshow: waiting for frame\n"); | |
2347 for(i=0;i<1000 && !rb->count;i++) usec_sleep(1000); | |
2348 if(!rb->count){ | |
2349 mp_msg(MSGT_TV,MSGL_DBG4,"tvi_dshow: waiting timeout\n"); | |
2350 return 0; | |
2351 } | |
2352 mp_msg(MSGT_TV,MSGL_DBG4,"tvi_dshow: got frame!\n"); | |
2353 } | |
2354 EnterCriticalSection(rb->pMutex); | |
2355 if(rb->tStart<0) | |
2356 rb->tStart=rb->dpts[rb->head]; | |
2357 pts=rb->dpts[rb->head]-rb->tStart; | |
2358 memcpy(buffer, rb->ringbuffer[rb->head], bytes); | |
2359 rb->head = (rb->head + 1) % rb->buffersize; | |
2360 rb->count--; | |
2361 LeaveCriticalSection(rb->pMutex); | |
2362 | |
2363 #ifdef HAVE_TV_TELETEXT | |
2364 vbi_grabber(priv); | |
2365 #endif | |
2366 return pts; | |
2367 } | |
2368 | |
2369 /** | |
2370 * \brief returns frame size | |
2371 * | |
2372 * \param priv driver's private data structure | |
2373 * | |
2374 * \return frame size if video present, 0 - otherwise | |
2375 */ | |
2376 static int get_video_framesize(priv_t * priv) | |
2377 { | |
2378 // if(!priv->pmtVideo) return 1; //no video | |
2379 // return(priv->pmtVideo->lSampleSize); | |
2380 if (!priv->v_buf) | |
2381 return 1; //no video | |
2382 mp_msg(MSGT_TV,MSGL_DBG3,"geT_video_framesize: %d\n",priv->v_buf->blocksize); | |
2383 return priv->v_buf->blocksize; | |
2384 } | |
2385 | |
2386 /** | |
2387 * \brief calculate audio buffer size | |
2388 * \param video_buf_size size of video buffer in bytes | |
2389 * \param video_bitrate video bit rate | |
2390 * \param audio_bitrate audio bit rate | |
2391 * \return audio buffer isze in bytes | |
2392 * | |
2393 * \remarks length of video buffer and resulted audio buffer calculated in | |
2394 * seconds will be the same. | |
2395 */ | |
2396 static inline int audio_buf_size_from_video(int video_buf_size, int video_bitrate, int audio_bitrate) | |
2397 { | |
2398 int audio_buf_size = audio_bitrate * (video_buf_size / video_bitrate); | |
2399 mp_msg(MSGT_TV,MSGL_DBG2,"tvi_dshow: Audio capture buffer: %d * %d / %d = %d\n", | |
2400 audio_bitrate,video_buf_size,video_bitrate,audio_buf_size); | |
2401 return audio_buf_size; | |
2402 } | |
2403 | |
2404 /** | |
25057
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2405 * \brief build video stream chain in graph |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2406 * \param priv private data structure |
24744 | 2407 * |
25057
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2408 * \return S_OK if chain was built successfully, apropriate error code otherwise |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2409 */ |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2410 static HRESULT build_video_chain(priv_t *priv) |
24744 | 2411 { |
2412 HRESULT hr; | |
2413 | |
25059 | 2414 if(priv->v_buf) |
2415 return S_OK; | |
2416 | |
24744 | 2417 if (priv->pVideoStreamConfig) { |
2418 hr = OLE_CALL_ARGS(priv->pVideoStreamConfig, SetFormat, priv->pmtVideo); | |
2419 if (FAILED(hr)) { | |
2420 mp_msg(MSGT_TV,MSGL_ERR,MSGTR_TVI_DS_UnableSelectVideoFormat, (unsigned int)hr); | |
2421 } | |
2422 } | |
2423 | |
2424 priv->v_buf=calloc(1,sizeof(grabber_ringbuffer_t)); | |
25058 | 2425 if(!priv->v_buf) |
2426 return E_OUTOFMEMORY; | |
24744 | 2427 |
2428 if (priv->tv_param->buffer_size >= 0) { | |
2429 priv->v_buf->buffersize = priv->tv_param->buffer_size; | |
2430 } else { | |
2431 priv->v_buf->buffersize = 16; | |
2432 } | |
2433 | |
2434 priv->v_buf->buffersize *= 1024 * 1024; | |
2435 hr=build_sub_graph(priv, priv->pVideoFilter, priv->v_buf, priv->pmtVideo,&PIN_CATEGORY_CAPTURE); | |
2436 if(FAILED(hr)){ | |
2437 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_UnableBuildVideoSubGraph,(unsigned int)hr); | |
25057
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2438 return hr; |
24744 | 2439 } |
25057
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2440 return S_OK; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2441 } |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2442 |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2443 /** |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2444 * \brief build audio stream chain in graph |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2445 * \param priv private data structure |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2446 * |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2447 * \return S_OK if chain was built successfully, apropriate error code otherwise |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2448 */ |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2449 static HRESULT build_audio_chain(priv_t *priv) |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2450 { |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2451 HRESULT hr; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2452 |
25059 | 2453 if(priv->a_buf) |
2454 return S_OK; | |
2455 | |
25057
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2456 if(priv->immediate_mode) |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2457 return S_OK; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2458 |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2459 if (priv->pAudioStreamConfig) { |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2460 hr = OLE_CALL_ARGS(priv->pAudioStreamConfig, SetFormat, |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2461 priv->pmtAudio); |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2462 if (FAILED(hr)) { |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2463 mp_msg(MSGT_TV,MSGL_ERR,MSGTR_TVI_DS_UnableSelectAudioFormat, (unsigned int)hr); |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2464 } |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2465 } |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2466 |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2467 if(priv->pmtAudio){ |
24744 | 2468 priv->a_buf=calloc(1,sizeof(grabber_ringbuffer_t)); |
25058 | 2469 if(!priv->a_buf) |
2470 return E_OUTOFMEMORY; | |
24744 | 2471 |
2472 /* let the audio buffer be the same size (in seconds) than video one */ | |
2473 priv->a_buf->buffersize=audio_buf_size_from_video( | |
2474 priv->v_buf->buffersize, | |
2475 (((VIDEOINFOHEADER *) priv->pmtVideo->pbFormat)->dwBitRate), | |
2476 (((WAVEFORMATEX *) (priv->pmtAudio->pbFormat))->nAvgBytesPerSec)); | |
2477 | |
2478 hr=build_sub_graph(priv, priv->pAudioFilter, priv->a_buf,priv->pmtAudio,&PIN_CATEGORY_CAPTURE); | |
2479 if(FAILED(hr)){ | |
2480 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_UnableBuildAudioSubGraph,(unsigned int)hr); | |
2481 return 0; | |
2482 } | |
2483 } | |
25057
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2484 return S_OK; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2485 } |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2486 |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2487 /** |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2488 * \brief build VBI stream chain in graph |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2489 * \param priv private data structure |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2490 * |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2491 * \return S_OK if chain was built successfully, apropriate error code otherwise |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2492 */ |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2493 static HRESULT build_vbi_chain(priv_t *priv) |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2494 { |
24744 | 2495 #ifdef HAVE_TV_TELETEXT |
25057
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2496 HRESULT hr; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2497 |
25059 | 2498 if(priv->vbi_buf) |
2499 return S_OK; | |
2500 | |
24744 | 2501 if(priv->tv_param->tdevice) |
2502 { | |
2503 priv->vbi_buf=calloc(1,sizeof(grabber_ringbuffer_t)); | |
25058 | 2504 if(!priv->vbi_buf) |
2505 return E_OUTOFMEMORY; | |
2506 | |
24744 | 2507 init_ringbuffer(priv->vbi_buf,24,priv->tsp.bufsize); |
2508 | |
2509 priv->pmtVBI=calloc(1,sizeof(AM_MEDIA_TYPE)); | |
2510 priv->pmtVBI->majortype=MEDIATYPE_VBI; | |
2511 hr=build_sub_graph(priv, priv->pVideoFilter, priv->vbi_buf,priv->pmtVBI,&PIN_CATEGORY_VBI); | |
2512 if(FAILED(hr)){ | |
2513 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_UnableBuildVBISubGraph,(unsigned int)hr); | |
2514 return 0; | |
2515 } | |
2516 } | |
2517 #endif | |
25057
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2518 return S_OK; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2519 } |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2520 |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2521 /** |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2522 * \brief playback/capture real start |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2523 * |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2524 * \param priv driver's private data structure |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2525 * |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2526 * \return 1 if success, 0 - otherwise |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2527 * |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2528 * TODO: move some code from init() here |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2529 */ |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2530 static int start(priv_t * priv) |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2531 { |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2532 HRESULT hr; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2533 |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2534 hr = build_video_chain(priv); |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2535 if(FAILED(hr)) |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2536 return 0; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2537 |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2538 hr = build_audio_chain(priv); |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2539 if(FAILED(hr)) |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2540 return 0; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2541 |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2542 hr = build_vbi_chain(priv); |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2543 if(FAILED(hr)) |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2544 return 0; |
7d74c1a2c840
Move chains building code into separate routines.
voroshil
parents:
25054
diff
changeset
|
2545 |
24744 | 2546 /* |
2547 Graph is ready to capture. Starting graph. | |
2548 */ | |
2549 if (mp_msg_test(MSGT_TV, MSGL_DBG2)) { | |
2550 mp_msg(MSGT_TV, MSGL_DBG2, "Debug pause 10sec\n"); | |
2551 usec_sleep(10000000); | |
2552 mp_msg(MSGT_TV, MSGL_DBG2, "Debug pause end\n"); | |
2553 } | |
2554 if (!priv->pMediaControl) { | |
2555 mp_msg(MSGT_TV,MSGL_ERR,MSGTR_TVI_DS_UnableGetMediaControlInterface,(unsigned int)E_POINTER); | |
2556 return 0; | |
2557 } | |
2558 hr = OLE_CALL(priv->pMediaControl, Run); | |
2559 if (FAILED(hr)) { | |
2560 mp_msg(MSGT_TV,MSGL_ERR,MSGTR_TVI_DS_UnableStartGraph, (unsigned int)hr); | |
2561 return 0; | |
2562 } | |
2563 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Graph is started.\n"); | |
2564 priv->state = 1; | |
2565 | |
2566 return (1); | |
2567 } | |
2568 | |
2569 /** | |
2570 * \brief driver initialization | |
2571 * | |
2572 * \param priv driver's private data structure | |
2573 * | |
2574 * \return 1 if success, 0 - otherwise | |
2575 */ | |
2576 static int init(priv_t * priv) | |
2577 { | |
2578 HRESULT hr; | |
2579 int result = 0; | |
2580 long lInput, lTunerInput; | |
2581 IEnumFilters *pEnum; | |
2582 IBaseFilter *pFilter; | |
2583 IPin *pVPOutPin; | |
2584 | |
2585 priv->state=0; | |
2586 CoInitialize(NULL); | |
2587 do{ | |
2588 hr = CoCreateInstance((GUID *) & CLSID_FilterGraph, NULL, | |
2589 CLSCTX_INPROC_SERVER, &IID_IGraphBuilder, | |
2590 (void **) &priv->pGraph); | |
2591 if(FAILED(hr)){ | |
2592 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: CoCreateInstance(FilterGraph) call failed. Error:0x%x\n", (unsigned int)hr); | |
2593 break; | |
2594 } | |
2595 //Debug | |
2596 if (mp_msg_test(MSGT_TV, MSGL_DBG2)) { | |
2597 AddToRot((IUnknown *) priv->pGraph, &(priv->dwRegister)); | |
2598 } | |
2599 | |
2600 hr = CoCreateInstance((GUID *) & CLSID_CaptureGraphBuilder2, NULL, | |
2601 CLSCTX_INPROC_SERVER, &IID_ICaptureGraphBuilder2, | |
2602 (void **) &priv->pBuilder); | |
2603 if(FAILED(hr)){ | |
2604 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: CoCreateInstance(CaptureGraphBuilder) call failed. Error:0x%x\n", (unsigned int)hr); | |
2605 break; | |
2606 } | |
2607 | |
2608 hr = OLE_CALL_ARGS(priv->pBuilder, SetFiltergraph, priv->pGraph); | |
2609 if(FAILED(hr)){ | |
2610 mp_msg(MSGT_TV,MSGL_ERR, "tvi_dshow: SetFiltergraph call failed. Error:0x%x\n",(unsigned int)hr); | |
2611 break; | |
2612 } | |
2613 | |
2614 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Searching for available video capture devices\n"); | |
2615 priv->pVideoFilter = find_capture_device(priv->dev_index, &CLSID_VideoInputDeviceCategory); | |
2616 if(!priv->pVideoFilter){ | |
2617 mp_msg(MSGT_TV,MSGL_ERR, MSGTR_TVI_DS_NoVideoCaptureDevice); | |
2618 break; | |
2619 } | |
2620 hr = OLE_CALL_ARGS(priv->pGraph, AddFilter, priv->pVideoFilter, NULL); | |
2621 if(FAILED(hr)){ | |
2622 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Unable to add video capture device to Directshow graph. Error:0x%x\n", (unsigned int)hr); | |
2623 break; | |
2624 } | |
2625 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Searching for available audio capture devices\n"); | |
2626 if (priv->adev_index != -1) { | |
2627 priv->pAudioFilter = find_capture_device(priv->adev_index, &CLSID_AudioInputDeviceCategory); //output available audio edevices | |
2628 if(!priv->pAudioFilter){ | |
2629 mp_msg(MSGT_TV,MSGL_ERR, MSGTR_TVI_DS_NoAudioCaptureDevice); | |
2630 break; | |
2631 } | |
2632 | |
2633 hr = OLE_CALL_ARGS(priv->pGraph, AddFilter, priv->pAudioFilter, NULL); | |
2634 if(FAILED(hr)){ | |
2635 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: Unable to add audio capture device to Directshow graph. Error:0x%x\n", (unsigned int)hr); | |
2636 break; | |
2637 } | |
2638 } else | |
2639 hr = OLE_QUERYINTERFACE(priv->pVideoFilter, IID_IBaseFilter, priv->pAudioFilter); | |
2640 | |
2641 hr = OLE_QUERYINTERFACE(priv->pVideoFilter, IID_IAMVideoProcAmp,priv->pVideoProcAmp); | |
2642 if (FAILED(hr) && hr != E_NOINTERFACE) | |
2643 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Get IID_IAMVideoProcAmp failed (0x%x).\n", (unsigned int)hr); | |
2644 | |
2645 if (hr != S_OK) { | |
2646 mp_msg(MSGT_TV, MSGL_INFO, MSGTR_TVI_DS_VideoAdjustigNotSupported); | |
2647 priv->pVideoProcAmp = NULL; | |
2648 } | |
2649 | |
2650 hr = OLE_CALL_ARGS(priv->pBuilder, FindInterface, | |
2651 &PIN_CATEGORY_CAPTURE, | |
2652 &MEDIATYPE_Video, | |
2653 priv->pVideoFilter, | |
2654 &IID_IAMCrossbar, (void **) &(priv->pCrossbar)); | |
2655 if (FAILED(hr)) { | |
2656 mp_msg(MSGT_TV, MSGL_INFO, MSGTR_TVI_DS_SelectingInputNotSupported); | |
2657 priv->pCrossbar = NULL; | |
2658 } | |
2659 | |
2660 hr = OLE_CALL_ARGS(priv->pBuilder, FindInterface, | |
2661 &PIN_CATEGORY_CAPTURE, | |
2662 &MEDIATYPE_Video, | |
2663 priv->pVideoFilter, | |
2664 &IID_IAMStreamConfig, | |
2665 (void **) &(priv->pVideoStreamConfig)); | |
2666 if (FAILED(hr)) { | |
2667 mp_msg(MSGT_TV, MSGL_INFO, MSGTR_TVI_DS_ChangingWidthHeightNotSupported); | |
2668 priv->pVideoStreamConfig = NULL; | |
2669 } | |
2670 | |
2671 hr = OLE_CALL_ARGS(priv->pBuilder, FindInterface, | |
2672 &PIN_CATEGORY_CAPTURE, | |
2673 &MEDIATYPE_Audio, | |
2674 priv->pAudioFilter, | |
2675 &IID_IAMStreamConfig, | |
2676 (void **) &(priv->pAudioStreamConfig)); | |
2677 if (FAILED(hr)) { | |
2678 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Get IAMStreamConfig(audio) failed (0x%x).\n", (unsigned int)hr); | |
2679 priv->pAudioStreamConfig = NULL; | |
2680 } | |
2681 | |
2682 if (priv->tv_param->amode >= 0) { | |
2683 IAMTVAudio *pTVAudio; | |
2684 hr = OLE_CALL_ARGS(priv->pBuilder, FindInterface, NULL, NULL,priv->pVideoFilter,&IID_IAMTVAudio, (void *) &pTVAudio); | |
2685 if (hr == S_OK) { | |
2686 switch (priv->tv_param->amode) { | |
2687 case 0: | |
2688 hr = OLE_CALL_ARGS(pTVAudio, put_TVAudioMode, AMTVAUDIO_MODE_MONO); | |
2689 break; | |
2690 case 1: | |
2691 hr = OLE_CALL_ARGS(pTVAudio, put_TVAudioMode, AMTVAUDIO_MODE_STEREO); | |
2692 break; | |
2693 case 2: | |
2694 hr = OLE_CALL_ARGS(pTVAudio, put_TVAudioMode, | |
2695 AMTVAUDIO_MODE_LANG_A); | |
2696 break; | |
2697 case 3: | |
2698 hr = OLE_CALL_ARGS(pTVAudio, put_TVAudioMode, | |
2699 AMTVAUDIO_MODE_LANG_B); | |
2700 break; | |
2701 } | |
2702 OLE_RELEASE_SAFE(pTVAudio); | |
2703 if (FAILED(hr)) | |
2704 mp_msg(MSGT_TV, MSGL_WARN, MSGTR_TVI_DS_UnableSetAudioMode, priv->tv_param->amode,(unsigned int)hr); | |
2705 } | |
2706 } | |
2707 /* | |
2708 Getting available video formats (last pointer in array will be NULL) | |
2709 First tryin to call IAMStreamConfig::GetStreamCaos. this will give us additional information such as | |
2710 min/max picture dimensions, etc. If this call fails trying IPIn::EnumMediaTypes with default | |
2711 min/max values. | |
2712 */ | |
2713 | |
2714 hr = get_available_formats_stream(priv->pVideoStreamConfig, | |
2715 &MEDIATYPE_Video, | |
2716 &(priv->arpmtVideo), | |
2717 (void ***) &(priv->arVideoCaps)); | |
2718 if (FAILED(hr)) { | |
2719 mp_msg(MSGT_TV, MSGL_DBG2, "Unable to use IAMStreamConfig for retriving available video formats (Error:0x%x). Using EnumMediaTypes instead\n", (unsigned int)hr); | |
2720 hr = get_available_formats_pin(priv->pBuilder, priv->pVideoFilter, | |
2721 &MEDIATYPE_Video, | |
2722 &(priv->arpmtVideo), | |
2723 (void ***) &(priv->arVideoCaps)); | |
2724 if(FAILED(hr)){ | |
2725 mp_msg(MSGT_TV,MSGL_ERR, MSGTR_TVI_DS_UnableGetsupportedVideoFormats, (unsigned int)hr); | |
2726 break; | |
2727 } | |
2728 } | |
2729 priv->nVideoFormatUsed = 0; | |
2730 | |
2731 if (!priv->arpmtVideo[priv->nVideoFormatUsed] | |
2732 || !extract_video_format(priv->arpmtVideo[priv->nVideoFormatUsed], | |
2733 &(priv->fcc), &(priv->width), | |
2734 &(priv->height))) { | |
2735 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_ErrorParsingVideoFormatStruct); | |
2736 break; | |
2737 } | |
2738 priv->pmtVideo = CreateMediaType(priv->arpmtVideo[priv->nVideoFormatUsed]); | |
2739 /* | |
2740 Getting available audio formats (last pointer in array will be NULL) | |
2741 */ | |
2742 hr = get_available_formats_stream(priv->pAudioStreamConfig, | |
2743 &MEDIATYPE_Audio, | |
2744 &(priv->arpmtAudio), | |
2745 (void ***) &(priv->arAudioCaps)); | |
2746 if (FAILED(hr)) { | |
2747 mp_msg(MSGT_TV, MSGL_DBG2, "Unable to use IAMStreamConfig for retriving available audio formats (Error:0x%x). Using EnumMediaTypes instead\n", (unsigned int)hr); | |
2748 hr = get_available_formats_pin(priv->pBuilder, priv->pAudioFilter, | |
2749 &MEDIATYPE_Audio, | |
2750 &(priv->arpmtAudio), | |
2751 (void ***) &(priv->arAudioCaps)); | |
2752 if (FAILED(hr)) { | |
2753 mp_msg(MSGT_TV,MSGL_WARN, MSGTR_TVI_DS_UnableGetsupportedAudioFormats, (unsigned int)hr); | |
2754 /* | |
2755 Following combination will disable sound | |
2756 */ | |
2757 priv->arpmtAudio = (AM_MEDIA_TYPE **) malloc(sizeof(AM_MEDIA_TYPE *)); | |
2758 priv->arpmtAudio[0] = NULL; | |
2759 priv->pmtAudio = NULL; | |
2760 priv->pAudioStreamConfig = NULL; | |
2761 } | |
2762 } | |
2763 /* debug */ | |
2764 { | |
2765 int i; | |
2766 for (i = 0; priv->arpmtVideo[i]; i++) { | |
2767 DisplayMediaType("Available video format", priv->arpmtVideo[i]); | |
2768 } | |
2769 for (i = 0; priv->arpmtAudio[i]; i++) { | |
2770 DisplayMediaType("Available audio format", priv->arpmtAudio[i]); | |
2771 } | |
2772 } | |
2773 priv->nAudioFormatUsed = 0; | |
2774 if (priv->arpmtAudio[priv->nAudioFormatUsed]) { | |
2775 priv->pmtAudio = CreateMediaType(priv->arpmtAudio[priv->nAudioFormatUsed]); | |
2776 if (!extract_audio_format(priv->pmtAudio, &(priv->samplerate), NULL, NULL)) { | |
2777 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_ErrorParsingAudioFormatStruct); | |
2778 DisplayMediaType("audio format failed",priv->arpmtAudio[priv->nAudioFormatUsed]); | |
2779 break; | |
2780 } | |
2781 } | |
2782 show_filter_info(priv->pVideoFilter); | |
2783 | |
2784 hr = OLE_QUERYINTERFACE(priv->pGraph, IID_IMediaControl,priv->pMediaControl); | |
2785 if(FAILED(hr)){ | |
2786 mp_msg(MSGT_TV,MSGL_ERR, MSGTR_TVI_DS_UnableGetMediaControlInterface,(unsigned int)hr); | |
2787 break; | |
2788 } | |
2789 hr = OLE_CALL_ARGS(priv->pBuilder, FindInterface, | |
2790 &PIN_CATEGORY_CAPTURE, NULL, | |
2791 priv->pVideoFilter, | |
2792 &IID_IAMTVTuner, (void **) &(priv->pTVTuner)); | |
2793 | |
2794 if (!priv->pTVTuner) { | |
2795 mp_msg(MSGT_TV, MSGL_DBG2, "tvi_dshow: Unable to access IAMTVTuner (0x%x)\n", (unsigned int)hr); | |
2796 } | |
2797 | |
2798 // shows Tuner capabilities | |
2799 get_capabilities(priv); | |
2800 | |
2801 if (priv->pTVTuner) { | |
2802 hr = OLE_CALL_ARGS(priv->pTVTuner, put_CountryCode, | |
2803 chanlist2country(priv->tv_param->chanlist)); | |
2804 if(FAILED(hr)){ | |
2805 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: Call to put_CountryCode failed. Error:0x%x\n",(unsigned int)hr); | |
2806 } | |
2807 | |
2808 hr = OLE_CALL_ARGS(priv->pTVTuner, put_Mode, AMTUNER_MODE_TV); | |
2809 if(FAILED(hr)){ | |
2810 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: Call to put_Mode failed. Error:0x%x\n",(unsigned int)hr); | |
2811 break; | |
2812 } | |
2813 | |
2814 hr = OLE_CALL_ARGS(priv->pTVTuner, get_ConnectInput, &lInput); | |
2815 if(FAILED(hr)){ | |
2816 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: Call to get_ConnectInput failed. Error:0x%x\n",(unsigned int)hr); | |
2817 break; | |
2818 } | |
2819 | |
2820 /* small hack */ | |
2821 lTunerInput = strstr(priv->tv_param->chanlist, "cable") ? TunerInputCable : TunerInputAntenna; | |
2822 | |
2823 hr = OLE_CALL_ARGS(priv->pTVTuner, put_InputType, lInput, lTunerInput); | |
2824 if(FAILED(hr)){ | |
2825 mp_msg(MSGT_TV,MSGL_DBG2, "tvi_dshow: Call to put_InputType failed. Error:0x%x\n",(unsigned int)hr); | |
2826 break; | |
2827 } | |
2828 | |
2829 } | |
2830 | |
2831 /** | |
2832 for VIVO cards we should check if preview pin is available on video capture device. | |
2833 If it is not, we have to connect Video Port Manager filter to VP pin of capture device filter. | |
2834 Otherwise we will get 0x8007001f (Device is not functioning properly) when attempting to start graph | |
2835 */ | |
2836 hr = OLE_CALL_ARGS(priv->pBuilder, FindPin, | |
2837 (IUnknown *) priv->pVideoFilter, | |
2838 PINDIR_OUTPUT, | |
2839 &PIN_CATEGORY_VIDEOPORT, NULL, FALSE, | |
2840 0, (IPin **) & pVPOutPin); | |
2841 if (SUCCEEDED(hr)) { | |
2842 hr = OLE_CALL_ARGS(priv->pGraph, Render, pVPOutPin); | |
2843 OLE_RELEASE_SAFE(pVPOutPin); | |
2844 | |
2845 if (FAILED(hr)) { | |
2846 mp_msg(MSGT_TV,MSGL_ERR, MSGTR_TVI_DS_UnableTerminateVPPin, (unsigned int)hr); | |
2847 break; | |
2848 } | |
2849 } | |
2850 | |
2851 OLE_CALL_ARGS(priv->pGraph, EnumFilters, &pEnum); | |
2852 while (OLE_CALL_ARGS(pEnum, Next, 1, &pFilter, NULL) == S_OK) { | |
2853 LPVIDEOWINDOW pVideoWindow; | |
2854 hr = OLE_QUERYINTERFACE(pFilter, IID_IVideoWindow, pVideoWindow); | |
2855 if (SUCCEEDED(hr)) | |
2856 { | |
2857 if(priv->tv_param->hidden_vp_renderer){ | |
2858 OLE_CALL_ARGS(pVideoWindow,put_Visible,/* OAFALSE*/ 0); | |
2859 OLE_CALL_ARGS(pVideoWindow,put_AutoShow,/* OAFALSE*/ 0); | |
2860 }else | |
2861 { | |
2862 OLE_CALL_ARGS(priv->pGraph, RemoveFilter, pFilter); | |
2863 } | |
2864 OLE_RELEASE_SAFE(pVideoWindow); | |
2865 } | |
2866 OLE_RELEASE_SAFE(pFilter); | |
2867 } | |
2868 OLE_RELEASE_SAFE(pEnum); | |
2869 if(priv->tv_param->system_clock) | |
2870 { | |
2871 LPREFERENCECLOCK rc; | |
2872 IBaseFilter* pBF; | |
2873 hr = CoCreateInstance((GUID *) & CLSID_SystemClock, NULL, | |
2874 CLSCTX_INPROC_SERVER, &IID_IReferenceClock, | |
2875 (void *) &rc); | |
2876 | |
2877 OLE_QUERYINTERFACE(priv->pBuilder,IID_IBaseFilter,pBF); | |
2878 OLE_CALL_ARGS(pBF,SetSyncSource,rc); | |
2879 } | |
2880 #ifdef HAVE_TV_TELETEXT | |
2881 if(vbi_get_props(priv,&(priv->tsp))!=TVI_CONTROL_TRUE) | |
2882 break; | |
2883 #endif | |
2884 result = 1; | |
2885 } while(0); | |
2886 | |
2887 if (!result){ | |
2888 mp_msg(MSGT_TV,MSGL_ERR, MSGTR_TVI_DS_GraphInitFailure); | |
2889 uninit(priv); | |
2890 } | |
2891 return result; | |
2892 } | |
2893 | |
2894 /** | |
2895 * \brief driver uninitialization | |
2896 * | |
2897 * \param priv driver's private data structure | |
2898 * | |
2899 * \return always 1 | |
2900 */ | |
2901 static int uninit(priv_t * priv) | |
2902 { | |
2903 int i; | |
2904 if (!priv) | |
2905 return (1); | |
2906 //Debug | |
2907 if (priv->dwRegister) { | |
2908 RemoveFromRot(priv->dwRegister); | |
2909 } | |
2910 #ifdef HAVE_TV_TELETEXT | |
2911 teletext_control(priv->priv_vbi,TV_VBI_CONTROL_STOP,(void*)1); | |
2912 #endif | |
2913 //stop audio grabber thread | |
2914 | |
2915 if (priv->state && priv->pMediaControl) { | |
2916 OLE_CALL(priv->pMediaControl, Stop); | |
2917 } | |
2918 OLE_RELEASE_SAFE(priv->pMediaControl); | |
2919 priv->state = 0; | |
2920 | |
2921 if (priv->pGraph) { | |
2922 if (priv->pVideoFilter) | |
2923 OLE_CALL_ARGS(priv->pGraph, RemoveFilter, priv->pVideoFilter); | |
2924 if (priv->pAudioFilter) | |
2925 OLE_CALL_ARGS(priv->pGraph, RemoveFilter, priv->pAudioFilter); | |
2926 } | |
2927 OLE_RELEASE_SAFE(priv->pCrossbar); | |
2928 OLE_RELEASE_SAFE(priv->pVideoStreamConfig); | |
2929 OLE_RELEASE_SAFE(priv->pAudioStreamConfig); | |
2930 OLE_RELEASE_SAFE(priv->pVideoProcAmp); | |
2931 OLE_RELEASE_SAFE(priv->pVideoFilter); | |
2932 OLE_RELEASE_SAFE(priv->pAudioFilter); | |
2933 OLE_RELEASE_SAFE(priv->pGraph); | |
2934 OLE_RELEASE_SAFE(priv->pBuilder); | |
2935 OLE_RELEASE_SAFE(priv->pCSGCB); | |
2936 | |
2937 if (priv->pmtVideo) | |
2938 DeleteMediaType(priv->pmtVideo); | |
2939 if (priv->pmtAudio) | |
2940 DeleteMediaType(priv->pmtAudio); | |
2941 if (priv->pmtVBI) | |
2942 DeleteMediaType(priv->pmtVBI); | |
2943 | |
2944 if (priv->arpmtVideo) { | |
2945 for (i = 0; priv->arpmtVideo[i]; i++) { | |
2946 DeleteMediaType(priv->arpmtVideo[i]); | |
2947 } | |
2948 free(priv->arpmtVideo); | |
2949 } | |
2950 if (priv->arVideoCaps) { | |
2951 for (i = 0; priv->arVideoCaps[i]; i++) { | |
2952 free(priv->arVideoCaps[i]); | |
2953 } | |
2954 free(priv->arVideoCaps); | |
2955 } | |
2956 if (priv->arpmtAudio) { | |
2957 for (i = 0; priv->arpmtAudio[i]; i++) { | |
2958 DeleteMediaType(priv->arpmtAudio[i]); | |
2959 } | |
2960 free(priv->arpmtAudio); | |
2961 } | |
2962 if (priv->arAudioCaps) { | |
2963 for (i = 0; priv->arAudioCaps[i]; i++) { | |
2964 free(priv->arAudioCaps[i]); | |
2965 } | |
2966 free(priv->arAudioCaps); | |
2967 } | |
2968 if (priv->a_buf) { | |
2969 destroy_ringbuffer(priv->a_buf); | |
2970 free(priv->a_buf); | |
2971 priv->a_buf = NULL; | |
2972 } | |
2973 if (priv->v_buf) { | |
2974 destroy_ringbuffer(priv->v_buf); | |
2975 free(priv->v_buf); | |
2976 priv->v_buf = NULL; | |
2977 } | |
2978 if (priv->vbi_buf) { | |
2979 destroy_ringbuffer(priv->vbi_buf); | |
2980 free(priv->vbi_buf); | |
2981 priv->vbi_buf = NULL; | |
2982 } | |
2983 if(priv->freq_table){ | |
2984 priv->freq_table_len=-1; | |
2985 free(priv->freq_table); | |
2986 priv->freq_table=NULL; | |
2987 } | |
2988 CoUninitialize(); | |
2989 return (1); | |
2990 } | |
2991 | |
2992 /** | |
2993 * \brief driver pre-initialization | |
2994 * | |
2995 * \param device string, containing device name in form "x[.y]", where x is video capture device | |
2996 * (default: 0, first available); y (if given) sets audio capture device | |
2997 * | |
2998 * \return 1 if success,0 - otherwise | |
2999 */ | |
3000 static tvi_handle_t *tvi_init_dshow(tv_param_t* tv_param) | |
3001 { | |
3002 tvi_handle_t *h; | |
3003 priv_t *priv; | |
3004 int a; | |
3005 | |
3006 h = new_handle(); | |
3007 if (!h) | |
3008 return (NULL); | |
3009 | |
3010 priv = h->priv; | |
3011 | |
3012 memset(priv, 0, sizeof(priv_t)); | |
3013 priv->direct_setfreq_call = 1; //first using direct call. if it fails, workaround will be enabled | |
3014 priv->direct_getfreq_call = 1; //first using direct call. if it fails, workaround will be enabled | |
3015 priv->adev_index = -1; | |
3016 priv->freq_table_len=-1; | |
3017 priv->tv_param=tv_param; | |
3018 | |
3019 if (tv_param->device) { | |
3020 if (sscanf(tv_param->device, "%d", &a) == 1) { | |
3021 priv->dev_index = a; | |
3022 } else { | |
3023 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_WrongDeviceParam, tv_param->device); | |
3024 free_handle(h); | |
3025 return NULL; | |
3026 } | |
3027 if (priv->dev_index < 0) { | |
3028 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_WrongDeviceIndex, a); | |
3029 free_handle(h); | |
3030 return NULL; | |
3031 } | |
3032 } | |
3033 if (tv_param->adevice) { | |
3034 if (sscanf(tv_param->adevice, "%d", &a) == 1) { | |
3035 priv->adev_index = a; | |
3036 } else { | |
3037 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_WrongADeviceParam, tv_param->adevice); | |
3038 free_handle(h); | |
3039 return NULL; | |
3040 } | |
3041 if (priv->dev_index < 0) { | |
3042 mp_msg(MSGT_TV, MSGL_ERR, MSGTR_TVI_DS_WrongADeviceIndex, a); | |
3043 free_handle(h); | |
3044 return NULL; | |
3045 } | |
3046 } | |
3047 return h; | |
3048 } | |
3049 | |
3050 /** | |
3051 * \brief driver's ioctl handler | |
3052 * | |
3053 * \param priv driver's private data structure | |
3054 * \param cmd ioctl command | |
3055 * \param arg ioct command's parameter | |
3056 * | |
3057 * \return TVI_CONTROL_TRUE if success | |
3058 * \return TVI_CONTROL_FALSE if failure | |
3059 * \return TVI_CONTROL_UNKNOWN if unknowm cmd called | |
3060 */ | |
3061 static int control(priv_t * priv, int cmd, void *arg) | |
3062 { | |
3063 switch (cmd) { | |
3064 /* need rewrite */ | |
3065 case TVI_CONTROL_VID_SET_FORMAT: | |
3066 { | |
3067 int fcc, i; | |
3068 if (priv->state) | |
3069 return TVI_CONTROL_FALSE; | |
3070 fcc = *(int *) arg; | |
3071 | |
3072 if(!priv->arpmtVideo) | |
3073 return TVI_CONTROL_FALSE; | |
3074 for (i = 0; priv->arpmtVideo[i]; i++) | |
3075 if (check_video_format | |
3076 (priv->arpmtVideo[i], fcc, priv->width, priv->height)) | |
3077 break; | |
3078 if (!priv->arpmtVideo[i]) | |
3079 return TVI_CONTROL_FALSE; | |
3080 | |
3081 priv->nVideoFormatUsed = i; | |
3082 | |
3083 if (priv->pmtVideo) | |
3084 DeleteMediaType(priv->pmtVideo); | |
3085 priv->pmtVideo = | |
3086 CreateMediaType(priv->arpmtVideo[priv->nVideoFormatUsed]); | |
3087 DisplayMediaType("VID_SET_FORMAT", priv->pmtVideo); | |
3088 /* | |
3089 Setting width & height to preferred by driver values | |
3090 */ | |
3091 extract_video_format(priv->arpmtVideo[priv->nVideoFormatUsed], | |
3092 &(priv->fcc), &(priv->width), | |
3093 &(priv->height)); | |
3094 return TVI_CONTROL_TRUE; | |
3095 } | |
3096 case TVI_CONTROL_VID_GET_FORMAT: | |
3097 { | |
3098 if(!priv->pmtVideo) | |
3099 return TVI_CONTROL_FALSE; | |
3100 DisplayMediaType("VID_GET_FORMAT", priv->pmtVideo); | |
3101 if (priv->fcc) { | |
3102 *(int *) arg = priv->fcc; | |
3103 return (TVI_CONTROL_TRUE); | |
3104 } else | |
3105 return (TVI_CONTROL_FALSE); | |
3106 } | |
3107 case TVI_CONTROL_VID_SET_WIDTH: | |
3108 { | |
3109 VIDEO_STREAM_CONFIG_CAPS *pCaps; | |
3110 VIDEOINFOHEADER *Vhdr; | |
3111 int width = *(int *) arg; | |
3112 if (priv->state) | |
3113 return TVI_CONTROL_FALSE; | |
3114 | |
3115 pCaps = priv->arVideoCaps[priv->nVideoFormatUsed]; | |
3116 if (!pCaps) | |
3117 return TVI_CONTROL_FALSE; | |
3118 if (width < pCaps->MinOutputSize.cx | |
3119 || width > pCaps->MaxOutputSize.cx) | |
3120 return TVI_CONTROL_FALSE; | |
3121 | |
3122 if (width % pCaps->OutputGranularityX) | |
3123 return TVI_CONTROL_FALSE; | |
3124 | |
3125 if (!priv->pmtVideo || !priv->pmtVideo->pbFormat) | |
3126 return TVI_CONTROL_FALSE; | |
3127 Vhdr = (VIDEOINFOHEADER *) priv->pmtVideo->pbFormat; | |
3128 Vhdr->bmiHeader.biWidth = width; | |
3129 priv->pmtVideo->lSampleSize = Vhdr->bmiHeader.biSizeImage = | |
3130 labs(Vhdr->bmiHeader.biBitCount * Vhdr->bmiHeader.biWidth * | |
3131 Vhdr->bmiHeader.biHeight) >> 3; | |
3132 | |
3133 priv->width = width; | |
3134 | |
3135 return (TVI_CONTROL_TRUE); | |
3136 } | |
3137 case TVI_CONTROL_VID_GET_WIDTH: | |
3138 { | |
3139 if (priv->width) { | |
3140 *(int *) arg = priv->width; | |
3141 return (TVI_CONTROL_TRUE); | |
3142 } else | |
3143 return TVI_CONTROL_FALSE; | |
3144 } | |
3145 case TVI_CONTROL_VID_CHK_WIDTH: | |
3146 { | |
3147 VIDEO_STREAM_CONFIG_CAPS *pCaps; | |
3148 int width = *(int *) arg; | |
3149 pCaps = priv->arVideoCaps[priv->nVideoFormatUsed]; | |
3150 if (!pCaps) | |
3151 return TVI_CONTROL_FALSE; | |
3152 if (width < pCaps->MinOutputSize.cx | |
3153 || width > pCaps->MaxOutputSize.cx) | |
3154 return TVI_CONTROL_FALSE; | |
3155 | |
3156 if (width % pCaps->OutputGranularityX) | |
3157 return TVI_CONTROL_FALSE; | |
3158 return (TVI_CONTROL_TRUE); | |
3159 } | |
3160 case TVI_CONTROL_VID_SET_HEIGHT: | |
3161 { | |
3162 VIDEO_STREAM_CONFIG_CAPS *pCaps; | |
3163 VIDEOINFOHEADER *Vhdr; | |
3164 int height = *(int *) arg; | |
3165 if (priv->state) | |
3166 return TVI_CONTROL_FALSE; | |
3167 | |
3168 pCaps = priv->arVideoCaps[priv->nVideoFormatUsed]; | |
3169 if (!pCaps) | |
3170 return TVI_CONTROL_FALSE; | |
3171 if (height < pCaps->MinOutputSize.cy | |
3172 || height > pCaps->MaxOutputSize.cy) | |
3173 return TVI_CONTROL_FALSE; | |
3174 | |
3175 if (height % pCaps->OutputGranularityY) | |
3176 return TVI_CONTROL_FALSE; | |
3177 | |
3178 if (!priv->pmtVideo || !priv->pmtVideo->pbFormat) | |
3179 return TVI_CONTROL_FALSE; | |
3180 Vhdr = (VIDEOINFOHEADER *) priv->pmtVideo->pbFormat; | |
3181 | |
3182 if (Vhdr->bmiHeader.biHeight < 0) | |
3183 Vhdr->bmiHeader.biHeight = -height; | |
3184 else | |
3185 Vhdr->bmiHeader.biHeight = height; | |
3186 priv->pmtVideo->lSampleSize = Vhdr->bmiHeader.biSizeImage = | |
3187 labs(Vhdr->bmiHeader.biBitCount * Vhdr->bmiHeader.biWidth * | |
3188 Vhdr->bmiHeader.biHeight) >> 3; | |
3189 | |
3190 priv->height = height; | |
3191 return (TVI_CONTROL_TRUE); | |
3192 } | |
3193 case TVI_CONTROL_VID_GET_HEIGHT: | |
3194 { | |
3195 if (priv->height) { | |
3196 *(int *) arg = priv->height; | |
3197 return (TVI_CONTROL_TRUE); | |
3198 } else | |
3199 return TVI_CONTROL_FALSE; | |
3200 } | |
3201 case TVI_CONTROL_VID_CHK_HEIGHT: | |
3202 { | |
3203 VIDEO_STREAM_CONFIG_CAPS *pCaps; | |
3204 int height = *(int *) arg; | |
3205 pCaps = priv->arVideoCaps[priv->nVideoFormatUsed]; | |
3206 if (!pCaps) | |
3207 return TVI_CONTROL_FALSE; | |
3208 if (height < pCaps->MinOutputSize.cy | |
3209 || height > pCaps->MaxOutputSize.cy) | |
3210 return TVI_CONTROL_FALSE; | |
3211 | |
3212 if (height % pCaps->OutputGranularityY) | |
3213 return TVI_CONTROL_FALSE; | |
3214 | |
3215 return (TVI_CONTROL_TRUE); | |
3216 } | |
3217 case TVI_CONTROL_IS_AUDIO: | |
3218 if (!priv->pmtAudio) | |
3219 return TVI_CONTROL_FALSE; | |
3220 else | |
3221 return TVI_CONTROL_TRUE; | |
3222 case TVI_CONTROL_IS_VIDEO: | |
3223 return TVI_CONTROL_TRUE; | |
3224 case TVI_CONTROL_AUD_GET_FORMAT: | |
3225 { | |
3226 *(int *) arg = AF_FORMAT_S16_LE; | |
3227 if (!priv->pmtAudio) | |
3228 return TVI_CONTROL_FALSE; | |
3229 else | |
3230 return TVI_CONTROL_TRUE; | |
3231 } | |
3232 case TVI_CONTROL_AUD_GET_CHANNELS: | |
3233 { | |
3234 *(int *) arg = priv->channels; | |
3235 if (!priv->pmtAudio) | |
3236 return TVI_CONTROL_FALSE; | |
3237 else | |
3238 return TVI_CONTROL_TRUE; | |
3239 } | |
3240 case TVI_CONTROL_AUD_SET_SAMPLERATE: | |
3241 { | |
3242 int i, samplerate; | |
3243 if (priv->state) | |
3244 return TVI_CONTROL_FALSE; | |
3245 if (!priv->arpmtAudio[0]) | |
3246 return TVI_CONTROL_FALSE; | |
3247 | |
3248 samplerate = *(int *) arg;; | |
3249 | |
3250 for (i = 0; priv->arpmtAudio[i]; i++) | |
3251 if (check_audio_format | |
3252 (priv->arpmtAudio[i], samplerate, 16, priv->channels)) | |
3253 break; | |
3254 if (!priv->arpmtAudio[i]) { | |
3255 //request not found. failing back to first available | |
3256 mp_msg(MSGT_TV, MSGL_WARN, MSGTR_TVI_DS_SamplerateNotsupported, samplerate); | |
3257 i = 0; | |
3258 } | |
3259 if (priv->pmtAudio) | |
3260 DeleteMediaType(priv->pmtAudio); | |
3261 priv->pmtAudio = CreateMediaType(priv->arpmtAudio[i]); | |
3262 extract_audio_format(priv->arpmtAudio[i], &(priv->samplerate), | |
3263 NULL, &(priv->channels)); | |
3264 return TVI_CONTROL_TRUE; | |
3265 } | |
3266 case TVI_CONTROL_AUD_GET_SAMPLERATE: | |
3267 { | |
3268 *(int *) arg = priv->samplerate; | |
3269 if (!priv->samplerate) | |
3270 return TVI_CONTROL_FALSE; | |
3271 if (!priv->pmtAudio) | |
3272 return TVI_CONTROL_FALSE; | |
3273 else | |
3274 return TVI_CONTROL_TRUE; | |
3275 } | |
3276 case TVI_CONTROL_AUD_GET_SAMPLESIZE: | |
3277 { | |
3278 WAVEFORMATEX *pWF; | |
3279 if (!priv->pmtAudio) | |
3280 return TVI_CONTROL_FALSE; | |
3281 if (!priv->pmtAudio->pbFormat) | |
3282 return TVI_CONTROL_FALSE; | |
3283 pWF = (WAVEFORMATEX *) priv->pmtAudio->pbFormat; | |
3284 *(int *) arg = pWF->wBitsPerSample / 8; | |
3285 return TVI_CONTROL_TRUE; | |
3286 } | |
3287 case TVI_CONTROL_IS_TUNER: | |
3288 { | |
3289 if (!priv->pTVTuner) | |
3290 return TVI_CONTROL_FALSE; | |
3291 | |
3292 return (TVI_CONTROL_TRUE); | |
3293 } | |
3294 case TVI_CONTROL_TUN_SET_NORM: | |
3295 { | |
3296 IAMAnalogVideoDecoder *pVD; | |
3297 long lAnalogFormat; | |
3298 int i; | |
3299 HRESULT hr; | |
3300 | |
3301 i = *(int *) arg; | |
3302 i--; | |
3303 if (i < 0 || i >= tv_available_norms_count) | |
3304 return TVI_CONTROL_FALSE; | |
3305 lAnalogFormat = tv_norms[tv_available_norms[i]].index; | |
3306 | |
3307 hr = OLE_QUERYINTERFACE(priv->pVideoFilter,IID_IAMAnalogVideoDecoder, pVD); | |
3308 if (hr != S_OK) | |
3309 return TVI_CONTROL_FALSE; | |
3310 hr = OLE_CALL_ARGS(pVD, get_TVFormat, &lAnalogFormat); | |
3311 OLE_RELEASE_SAFE(pVD); | |
3312 if (FAILED(hr)) | |
3313 return (TVI_CONTROL_FALSE); | |
3314 else | |
3315 return (TVI_CONTROL_TRUE); | |
3316 } | |
3317 case TVI_CONTROL_TUN_GET_NORM: | |
3318 { | |
3319 long lAnalogFormat; | |
3320 int i; | |
3321 HRESULT hr; | |
3322 IAMAnalogVideoDecoder *pVD; | |
3323 | |
3324 hr = OLE_QUERYINTERFACE(priv->pVideoFilter,IID_IAMAnalogVideoDecoder, pVD); | |
3325 if (hr == S_OK) { | |
3326 hr = OLE_CALL_ARGS(pVD, get_TVFormat, &lAnalogFormat); | |
3327 OLE_RELEASE_SAFE(pVD); | |
3328 } | |
3329 | |
3330 if (FAILED(hr)) { //trying another method | |
3331 if (!priv->pTVTuner) | |
3332 return TVI_CONTROL_FALSE; | |
3333 hr=OLE_CALL_ARGS(priv->pTVTuner, get_TVFormat, &lAnalogFormat); | |
3334 if (FAILED(hr)) | |
3335 return TVI_CONTROL_FALSE; | |
3336 } | |
3337 for (i = 0; i < tv_available_norms_count; i++) { | |
3338 if (tv_norms[tv_available_norms[i]].index == lAnalogFormat) { | |
3339 *(int *) arg = i + 1; | |
3340 return TVI_CONTROL_TRUE; | |
3341 } | |
3342 } | |
3343 return (TVI_CONTROL_FALSE); | |
3344 } | |
3345 case TVI_CONTROL_SPC_GET_NORMID: | |
3346 { | |
3347 int i; | |
3348 if (!priv->pTVTuner) | |
3349 return TVI_CONTROL_FALSE; | |
3350 for (i = 0; i < tv_available_norms_count; i++) { | |
3351 if (!strcasecmp | |
3352 (tv_norms[tv_available_norms[i]].name, (char *) arg)) { | |
3353 *(int *) arg = i + 1; | |
3354 return TVI_CONTROL_TRUE; | |
3355 } | |
3356 } | |
3357 return TVI_CONTROL_FALSE; | |
3358 } | |
3359 case TVI_CONTROL_SPC_SET_INPUT: | |
3360 { | |
3361 return set_crossbar_input(priv, *(int *) arg); | |
3362 } | |
3363 case TVI_CONTROL_TUN_GET_FREQ: | |
3364 { | |
3365 unsigned long lFreq; | |
3366 int ret; | |
3367 if (!priv->pTVTuner) | |
3368 return TVI_CONTROL_FALSE; | |
3369 | |
3370 ret = get_frequency(priv, &lFreq); | |
3371 lFreq = lFreq * 16 / 1000000; //convert from Hz to 1/16 MHz units | |
3372 | |
3373 *(unsigned long *) arg = lFreq; | |
3374 return ret; | |
3375 } | |
3376 case TVI_CONTROL_TUN_SET_FREQ: | |
3377 { | |
3378 unsigned long nFreq = *(unsigned long *) arg; | |
3379 if (!priv->pTVTuner) | |
3380 return TVI_CONTROL_FALSE; | |
3381 //convert to Hz | |
3382 nFreq = 1000000 * nFreq / 16; //convert from 1/16 MHz units to Hz | |
3383 return set_frequency(priv, nFreq); | |
3384 } | |
3385 case TVI_CONTROL_VID_SET_HUE: | |
3386 return set_control(priv, VideoProcAmp_Hue, *(int *) arg); | |
3387 case TVI_CONTROL_VID_GET_HUE: | |
3388 return get_control(priv, VideoProcAmp_Hue, (int *) arg); | |
3389 case TVI_CONTROL_VID_SET_CONTRAST: | |
3390 return set_control(priv, VideoProcAmp_Contrast, *(int *) arg); | |
3391 case TVI_CONTROL_VID_GET_CONTRAST: | |
3392 return get_control(priv, VideoProcAmp_Contrast, (int *) arg); | |
3393 case TVI_CONTROL_VID_SET_SATURATION: | |
3394 return set_control(priv, VideoProcAmp_Saturation, *(int *) arg); | |
3395 case TVI_CONTROL_VID_GET_SATURATION: | |
3396 return get_control(priv, VideoProcAmp_Saturation, (int *) arg); | |
3397 case TVI_CONTROL_VID_SET_BRIGHTNESS: | |
3398 return set_control(priv, VideoProcAmp_Brightness, *(int *) arg); | |
3399 case TVI_CONTROL_VID_GET_BRIGHTNESS: | |
3400 return get_control(priv, VideoProcAmp_Brightness, (int *) arg); | |
3401 | |
3402 case TVI_CONTROL_VID_GET_FPS: | |
3403 { | |
3404 VIDEOINFOHEADER *Vhdr; | |
3405 if (!priv->pmtVideo) | |
3406 return TVI_CONTROL_FALSE; | |
3407 if (!priv->pmtVideo->pbFormat) | |
3408 return TVI_CONTROL_FALSE; | |
3409 Vhdr = (VIDEOINFOHEADER *) priv->pmtVideo->pbFormat; | |
3410 *(float *) arg = | |
25028
2cae9470f53b
Fix FPS from bitrate calculation (was 8 times larger than real value).
voroshil
parents:
25019
diff
changeset
|
3411 (1.0 * Vhdr->dwBitRate) / (Vhdr->bmiHeader.biSizeImage * 8); |
24744 | 3412 return TVI_CONTROL_TRUE; |
3413 } | |
3414 case TVI_CONTROL_IMMEDIATE: | |
3415 priv->immediate_mode = 1; | |
3416 return TVI_CONTROL_TRUE; | |
3417 #ifdef HAVE_TV_TELETEXT | |
3418 case TVI_CONTROL_VBI_INIT: | |
3419 { | |
3420 void* ptr; | |
3421 ptr=&(priv->tsp); | |
3422 if(teletext_control(NULL,TV_VBI_CONTROL_START,&ptr)==TVI_CONTROL_TRUE) | |
3423 priv->priv_vbi=ptr; | |
3424 else | |
3425 priv->priv_vbi=NULL; | |
3426 return TVI_CONTROL_TRUE; | |
3427 } | |
3428 default: | |
3429 return teletext_control(priv->priv_vbi,cmd,arg); | |
3430 #endif | |
3431 } | |
3432 return (TVI_CONTROL_UNKNOWN); | |
3433 } |