ws2_32: Allow arbitrary-sized iovecs again in WSASendTo and WSARecvFrom.
[wine/multimedia.git] / dlls / msvfw32 / msvideo16.c
blob221542c5cc36bc12196b7ab34b5123f4d65367f2
1 /*
2 * msvideo 16-bit functions
4 * Copyright 1998 Marcus Meissner
5 * Copyright 2000 Bradley Baetz
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #include <stdarg.h>
23 #include <stdio.h>
24 #include <string.h>
26 #include "windef.h"
27 #include "winbase.h"
28 #include "winver.h"
29 #include "winnls.h"
30 #include "winreg.h"
31 #include "winuser.h"
32 #include "vfw16.h"
33 #include "msvideo_private.h"
34 #include "wine/debug.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(msvideo);
38 /* Drivers32 settings */
39 #define HKLM_DRIVERS32 "Software\\Microsoft\\Windows NT\\CurrentVersion\\Drivers32"
41 /***********************************************************************
42 * DrawDibOpen [MSVIDEO.102]
44 HDRAWDIB16 VFWAPI DrawDibOpen16(void)
46 return HDRAWDIB_16(DrawDibOpen());
49 /***********************************************************************
50 * DrawDibClose [MSVIDEO.103]
52 BOOL16 VFWAPI DrawDibClose16(HDRAWDIB16 hdd)
54 return DrawDibClose(HDRAWDIB_32(hdd));
57 /************************************************************************
58 * DrawDibBegin [MSVIDEO.104]
60 BOOL16 VFWAPI DrawDibBegin16(HDRAWDIB16 hdd, HDC16 hdc, INT16 dxDst,
61 INT16 dyDst, LPBITMAPINFOHEADER lpbi, INT16 dxSrc,
62 INT16 dySrc, UINT16 wFlags)
64 return DrawDibBegin(HDRAWDIB_32(hdd), HDC_32(hdc), dxDst, dyDst, lpbi,
65 dxSrc, dySrc, wFlags);
68 /***********************************************************************
69 * DrawDibEnd [MSVIDEO.105]
71 BOOL16 VFWAPI DrawDibEnd16(HDRAWDIB16 hdd)
73 return DrawDibEnd(HDRAWDIB_32(hdd));
76 /**********************************************************************
77 * DrawDibDraw [MSVIDEO.106]
79 BOOL16 VFWAPI DrawDibDraw16(HDRAWDIB16 hdd, HDC16 hdc, INT16 xDst, INT16 yDst,
80 INT16 dxDst, INT16 dyDst, LPBITMAPINFOHEADER lpbi,
81 LPVOID lpBits, INT16 xSrc, INT16 ySrc, INT16 dxSrc,
82 INT16 dySrc, UINT16 wFlags)
84 return DrawDibDraw(HDRAWDIB_32(hdd), HDC_32(hdc), xDst, yDst, dxDst,
85 dyDst, lpbi, lpBits, xSrc, ySrc, dxSrc, dySrc, wFlags);
88 /***********************************************************************
89 * DrawDibGetPalette [MSVIDEO.108]
91 HPALETTE16 VFWAPI DrawDibGetPalette16(HDRAWDIB16 hdd)
93 return HPALETTE_16(DrawDibGetPalette(HDRAWDIB_32(hdd)));
96 /***********************************************************************
97 * DrawDibSetPalette [MSVIDEO.110]
99 BOOL16 VFWAPI DrawDibSetPalette16(HDRAWDIB16 hdd, HPALETTE16 hpal)
101 return DrawDibSetPalette(HDRAWDIB_32(hdd), HPALETTE_32(hpal));
104 /***********************************************************************
105 * DrawDibRealize [MSVIDEO.112]
107 UINT16 VFWAPI DrawDibRealize16(HDRAWDIB16 hdd, HDC16 hdc,
108 BOOL16 fBackground)
110 return (UINT16)DrawDibRealize(HDRAWDIB_32(hdd), HDC_32(hdc), fBackground);
113 /*************************************************************************
114 * DrawDibStart [MSVIDEO.118]
116 BOOL16 VFWAPI DrawDibStart16(HDRAWDIB16 hdd, DWORD rate)
118 return DrawDibStart(HDRAWDIB_32(hdd), rate);
121 /*************************************************************************
122 * DrawDibStop [MSVIDEO.119]
124 BOOL16 VFWAPI DrawDibStop16(HDRAWDIB16 hdd)
126 return DrawDibStop(HDRAWDIB_32(hdd));
129 /***********************************************************************
130 * ICOpen [MSVIDEO.203]
132 HIC16 VFWAPI ICOpen16(DWORD fccType, DWORD fccHandler, UINT16 wMode)
134 return HIC_16(ICOpen(fccType, fccHandler, wMode));
137 /***********************************************************************
138 * ICClose [MSVIDEO.204]
140 LRESULT WINAPI ICClose16(HIC16 hic)
142 return ICClose(HIC_32(hic));
145 /***********************************************************************
146 * _ICMessage [MSVIDEO.207]
148 LRESULT VFWAPIV ICMessage16( HIC16 hic, UINT16 msg, UINT16 cb, VA_LIST16 valist )
150 LPWORD lpData;
151 SEGPTR segData;
152 LRESULT ret;
153 UINT16 i;
155 lpData = HeapAlloc(GetProcessHeap(), 0, cb);
157 TRACE("0x%08x, %u, %u, ...)\n", (DWORD) hic, msg, cb);
159 for (i = 0; i < cb / sizeof(WORD); i++)
161 lpData[i] = VA_ARG16(valist, WORD);
164 segData = MapLS(lpData);
165 ret = ICSendMessage16(hic, msg, segData, (DWORD) cb);
166 UnMapLS(segData);
167 HeapFree(GetProcessHeap(), 0, lpData);
168 return ret;
171 /***********************************************************************
172 * ICGetInfo [MSVIDEO.212]
174 LRESULT VFWAPI ICGetInfo16(HIC16 hic, ICINFO16 * picinfo, DWORD cb)
176 LRESULT ret;
178 TRACE("(0x%08x,%p,%d)\n", (DWORD) hic, picinfo, cb);
179 ret = ICSendMessage16(hic, ICM_GETINFO, (DWORD) picinfo, cb);
180 TRACE(" -> 0x%08lx\n", ret);
181 return ret;
184 /***********************************************************************
185 * ICLocate [MSVIDEO.213]
187 HIC16 VFWAPI ICLocate16(DWORD fccType, DWORD fccHandler,
188 LPBITMAPINFOHEADER lpbiIn, LPBITMAPINFOHEADER lpbiOut,
189 WORD wFlags)
191 return HIC_16(ICLocate(fccType, fccHandler, lpbiIn, lpbiOut, wFlags));
194 /***********************************************************************
195 * _ICCompress [MSVIDEO.224]
197 DWORD VFWAPIV ICCompress16(HIC16 hic, DWORD dwFlags,
198 LPBITMAPINFOHEADER lpbiOutput, LPVOID lpData,
199 LPBITMAPINFOHEADER lpbiInput, LPVOID lpBits,
200 LPDWORD lpckid, LPDWORD lpdwFlags,
201 LONG lFrameNum, DWORD dwFrameSize,
202 DWORD dwQuality, LPBITMAPINFOHEADER lpbiPrev,
203 LPVOID lpPrev)
205 DWORD ret;
206 ICCOMPRESS iccmp;
207 SEGPTR seg_iccmp;
209 TRACE("(0x%08x,%d,%p,%p,%p,%p,...)\n", (DWORD) hic, dwFlags,
210 lpbiOutput, lpData, lpbiInput, lpBits);
212 iccmp.dwFlags = dwFlags;
214 iccmp.lpbiOutput = lpbiOutput;
215 iccmp.lpOutput = lpData;
216 iccmp.lpbiInput = lpbiInput;
217 iccmp.lpInput = lpBits;
219 iccmp.lpckid = lpckid;
220 iccmp.lpdwFlags = lpdwFlags;
221 iccmp.lFrameNum = lFrameNum;
222 iccmp.dwFrameSize = dwFrameSize;
223 iccmp.dwQuality = dwQuality;
224 iccmp.lpbiPrev = lpbiPrev;
225 iccmp.lpPrev = lpPrev;
226 seg_iccmp = MapLS(&iccmp);
227 ret = ICSendMessage16(hic, ICM_COMPRESS, seg_iccmp, sizeof(ICCOMPRESS));
228 UnMapLS(seg_iccmp);
229 return ret;
232 /***********************************************************************
233 * _ICDecompress [MSVIDEO.230]
235 DWORD VFWAPIV ICDecompress16(HIC16 hic, DWORD dwFlags,
236 LPBITMAPINFOHEADER lpbiFormat, LPVOID lpData,
237 LPBITMAPINFOHEADER lpbi, LPVOID lpBits)
239 ICDECOMPRESS icd;
240 SEGPTR segptr;
241 DWORD ret;
243 TRACE("(0x%08x,%d,%p,%p,%p,%p)\n", (DWORD) hic, dwFlags, lpbiFormat,
244 lpData, lpbi, lpBits);
246 icd.dwFlags = dwFlags;
247 icd.lpbiInput = lpbiFormat;
248 icd.lpInput = lpData;
249 icd.lpbiOutput = lpbi;
250 icd.lpOutput = lpBits;
251 icd.ckid = 0;
252 segptr = MapLS(&icd);
253 ret = ICSendMessage16(hic, ICM_DECOMPRESS, segptr, sizeof(ICDECOMPRESS));
254 UnMapLS(segptr);
255 return ret;
258 /***********************************************************************
259 * _ICDrawBegin [MSVIDEO.232]
261 DWORD VFWAPIV ICDrawBegin16(HIC16 hic, /* [in] */
262 DWORD dwFlags, /* [in] flags */
263 HPALETTE16 hpal, /* [in] palette to draw with */
264 HWND16 hwnd, /* [in] window to draw to */
265 HDC16 hdc, /* [in] HDC to draw to */
266 INT16 xDst, /* [in] destination rectangle */
267 INT16 yDst, /* [in] */
268 INT16 dxDst, /* [in] */
269 INT16 dyDst, /* [in] */
270 LPBITMAPINFOHEADER lpbi, /* [in] format of frame to draw NOTE: SEGPTR */
271 INT16 xSrc, /* [in] source rectangle */
272 INT16 ySrc, /* [in] */
273 INT16 dxSrc, /* [in] */
274 INT16 dySrc, /* [in] */
275 DWORD dwRate, /* [in] frames/second = (dwRate/dwScale) */
276 DWORD dwScale) /* [in] */
278 DWORD ret;
279 ICDRAWBEGIN16 icdb;
280 SEGPTR seg_icdb;
282 TRACE ("(0x%08x,%d,0x%08x,0x%08x,0x%08x,%u,%u,%u,%u,%p,%u,%u,%u,%u,%d,%d)\n",
283 (DWORD) hic, dwFlags, (DWORD) hpal, (DWORD) hwnd, (DWORD) hdc,
284 xDst, yDst, dxDst, dyDst, lpbi, xSrc, ySrc, dxSrc, dySrc, dwRate,
285 dwScale);
287 icdb.dwFlags = dwFlags;
288 icdb.hpal = hpal;
289 icdb.hwnd = hwnd;
290 icdb.hdc = hdc;
291 icdb.xDst = xDst;
292 icdb.yDst = yDst;
293 icdb.dxDst = dxDst;
294 icdb.dyDst = dyDst;
295 icdb.lpbi = lpbi; /* Keep this as SEGPTR for the mapping code to deal with */
296 icdb.xSrc = xSrc;
297 icdb.ySrc = ySrc;
298 icdb.dxSrc = dxSrc;
299 icdb.dySrc = dySrc;
300 icdb.dwRate = dwRate;
301 icdb.dwScale = dwScale;
302 seg_icdb = MapLS(&icdb);
303 ret = (DWORD) ICSendMessage16(hic, ICM_DRAW_BEGIN, seg_icdb,
304 sizeof(ICDRAWBEGIN16));
305 UnMapLS(seg_icdb);
306 return ret;
309 /***********************************************************************
310 * _ICDraw [MSVIDEO.234]
312 DWORD VFWAPIV ICDraw16(HIC16 hic, DWORD dwFlags,
313 LPVOID lpFormat, /* [???] NOTE: SEGPTR */
314 LPVOID lpData, /* [???] NOTE: SEGPTR */
315 DWORD cbData, LONG lTime)
317 DWORD ret;
318 ICDRAW icd;
319 SEGPTR seg_icd;
321 TRACE("(0x%08x,0x%08x,%p,%p,%d,%d)\n", (DWORD) hic, dwFlags,
322 lpFormat, lpData, cbData, lTime);
323 icd.dwFlags = dwFlags;
324 icd.lpFormat = lpFormat;
325 icd.lpData = lpData;
326 icd.cbData = cbData;
327 icd.lTime = lTime;
328 seg_icd = MapLS(&icd);
329 ret = ICSendMessage16(hic, ICM_DRAW, seg_icd, sizeof(ICDRAW));
330 UnMapLS(seg_icd);
331 return ret;
334 /***********************************************************************
335 * ICGetDisplayFormat [MSVIDEO.239]
337 HIC16 VFWAPI ICGetDisplayFormat16(HIC16 hic, LPBITMAPINFOHEADER lpbiIn,
338 LPBITMAPINFOHEADER lpbiOut, INT16 depth,
339 INT16 dx, INT16 dy)
341 return HIC_16(ICGetDisplayFormat(HIC_32(hic), lpbiIn, lpbiOut, depth,
342 dx, dy));
345 #define COPY(x,y) (x->y = x##16->y);
346 #define COPYPTR(x,y) (x->y = MapSL((SEGPTR)x##16->y));
348 /******************************************************************
349 * MSVIDEO_MapICDEX16To32
353 static LPVOID MSVIDEO_MapICDEX16To32(LPDWORD lParam)
355 LPVOID ret;
357 ICDECOMPRESSEX *icdx = HeapAlloc(GetProcessHeap(), 0, sizeof(ICDECOMPRESSEX));
358 ICDECOMPRESSEX16 *icdx16 = MapSL(*lParam);
359 ret = icdx16;
361 COPY(icdx, dwFlags);
362 COPYPTR(icdx, lpbiSrc);
363 COPYPTR(icdx, lpSrc);
364 COPYPTR(icdx, lpbiDst);
365 COPYPTR(icdx, lpDst);
366 COPY(icdx, xDst);
367 COPY(icdx, yDst);
368 COPY(icdx, dxDst);
369 COPY(icdx, dyDst);
370 COPY(icdx, xSrc);
371 COPY(icdx, ySrc);
372 COPY(icdx, dxSrc);
373 COPY(icdx, dySrc);
375 *lParam = (DWORD)(icdx);
376 return ret;
379 /******************************************************************
380 * MSVIDEO_MapMsg16To32
384 static LPVOID MSVIDEO_MapMsg16To32(UINT msg, LPDWORD lParam1, LPDWORD lParam2)
386 LPVOID ret = 0;
388 TRACE("Mapping %d\n", msg);
390 switch (msg)
392 case DRV_LOAD:
393 case DRV_ENABLE:
394 case DRV_CLOSE:
395 case DRV_DISABLE:
396 case DRV_FREE:
397 case ICM_ABOUT:
398 case ICM_CONFIGURE:
399 case ICM_COMPRESS_END:
400 case ICM_DECOMPRESS_END:
401 case ICM_DECOMPRESSEX_END:
402 case ICM_SETQUALITY:
403 case ICM_DRAW_START_PLAY:
404 case ICM_DRAW_STOP_PLAY:
405 case ICM_DRAW_REALIZE:
406 case ICM_DRAW_RENDERBUFFER:
407 case ICM_DRAW_END:
408 break;
409 case DRV_OPEN:
410 case ICM_GETDEFAULTQUALITY:
411 case ICM_GETQUALITY:
412 case ICM_SETSTATE:
413 case ICM_DRAW_WINDOW:
414 case ICM_GETBUFFERSWANTED:
415 *lParam1 = (DWORD)MapSL(*lParam1);
416 break;
417 case ICM_GETINFO:
419 ICINFO *ici = HeapAlloc(GetProcessHeap(), 0, sizeof(ICINFO));
420 ICINFO16 *ici16;
422 ici16 = MapSL(*lParam1);
423 ret = ici16;
425 ici->dwSize = sizeof(ICINFO);
426 COPY(ici, fccType);
427 COPY(ici, fccHandler);
428 COPY(ici, dwFlags);
429 COPY(ici, dwVersion);
430 COPY(ici, dwVersionICM);
431 MultiByteToWideChar( CP_ACP, 0, ici16->szName, -1, ici->szName, 16 );
432 MultiByteToWideChar( CP_ACP, 0, ici16->szDescription, -1, ici->szDescription, 128 );
433 MultiByteToWideChar( CP_ACP, 0, ici16->szDriver, -1, ici->szDriver, 128 );
434 *lParam1 = (DWORD)(ici);
435 *lParam2 = sizeof(ICINFO);
437 break;
438 case ICM_COMPRESS:
440 ICCOMPRESS *icc = HeapAlloc(GetProcessHeap(), 0, sizeof(ICCOMPRESS));
441 ICCOMPRESS *icc16;
443 icc16 = MapSL(*lParam1);
444 ret = icc16;
446 COPY(icc, dwFlags);
447 COPYPTR(icc, lpbiOutput);
448 COPYPTR(icc, lpOutput);
449 COPYPTR(icc, lpbiInput);
450 COPYPTR(icc, lpInput);
451 COPYPTR(icc, lpckid);
452 COPYPTR(icc, lpdwFlags);
453 COPY(icc, lFrameNum);
454 COPY(icc, dwFrameSize);
455 COPY(icc, dwQuality);
456 COPYPTR(icc, lpbiPrev);
457 COPYPTR(icc, lpPrev);
459 *lParam1 = (DWORD)(icc);
460 *lParam2 = sizeof(ICCOMPRESS);
462 break;
463 case ICM_DECOMPRESS:
465 ICDECOMPRESS *icd = HeapAlloc(GetProcessHeap(), 0, sizeof(ICDECOMPRESS));
466 ICDECOMPRESS *icd16; /* Same structure except for the pointers */
468 icd16 = MapSL(*lParam1);
469 ret = icd16;
471 COPY(icd, dwFlags);
472 COPYPTR(icd, lpbiInput);
473 COPYPTR(icd, lpInput);
474 COPYPTR(icd, lpbiOutput);
475 COPYPTR(icd, lpOutput);
476 COPY(icd, ckid);
478 *lParam1 = (DWORD)(icd);
479 *lParam2 = sizeof(ICDECOMPRESS);
481 break;
482 case ICM_COMPRESS_BEGIN:
483 case ICM_COMPRESS_GET_FORMAT:
484 case ICM_COMPRESS_GET_SIZE:
485 case ICM_COMPRESS_QUERY:
486 case ICM_DECOMPRESS_GET_FORMAT:
487 case ICM_DECOMPRESS_QUERY:
488 case ICM_DECOMPRESS_BEGIN:
489 case ICM_DECOMPRESS_SET_PALETTE:
490 case ICM_DECOMPRESS_GET_PALETTE:
491 *lParam1 = (DWORD)MapSL(*lParam1);
492 *lParam2 = (DWORD)MapSL(*lParam2);
493 break;
494 case ICM_DECOMPRESSEX_QUERY:
495 if ((*lParam2 != sizeof(ICDECOMPRESSEX16)) && (*lParam2 != 0))
496 WARN("*lParam2 has unknown value %p\n", (ICDECOMPRESSEX16*)*lParam2);
497 /* FIXME: *lParm2 is meant to be 0 or an ICDECOMPRESSEX16*, but is sizeof(ICDECOMRPESSEX16)
498 * This is because of ICMessage(). Special case it?
500 LPVOID* addr = HeapAlloc(GetProcessHeap(), 0, 2*sizeof(LPVOID));
501 addr[0] = MSVIDEO_MapICDEX16To32(lParam1);
502 if (*lParam2)
503 addr[1] = MSVIDEO_MapICDEX16To32(lParam2);
504 else
505 addr[1] = 0;
507 ret = addr;
509 break;*/
510 case ICM_DECOMPRESSEX_BEGIN:
511 case ICM_DECOMPRESSEX:
512 ret = MSVIDEO_MapICDEX16To32(lParam1);
513 *lParam2 = sizeof(ICDECOMPRESSEX);
514 break;
515 case ICM_DRAW_BEGIN:
517 ICDRAWBEGIN *icdb = HeapAlloc(GetProcessHeap(), 0, sizeof(ICDRAWBEGIN));
518 ICDRAWBEGIN16 *icdb16 = MapSL(*lParam1);
519 ret = icdb16;
521 COPY(icdb, dwFlags);
522 icdb->hpal = HPALETTE_32(icdb16->hpal);
523 icdb->hwnd = HWND_32(icdb16->hwnd);
524 icdb->hdc = HDC_32(icdb16->hdc);
525 COPY(icdb, xDst);
526 COPY(icdb, yDst);
527 COPY(icdb, dxDst);
528 COPY(icdb, dyDst);
529 COPYPTR(icdb, lpbi);
530 COPY(icdb, xSrc);
531 COPY(icdb, ySrc);
532 COPY(icdb, dxSrc);
533 COPY(icdb, dySrc);
534 COPY(icdb, dwRate);
535 COPY(icdb, dwScale);
537 *lParam1 = (DWORD)(icdb);
538 *lParam2 = sizeof(ICDRAWBEGIN);
540 break;
541 case ICM_DRAW_SUGGESTFORMAT:
543 ICDRAWSUGGEST *icds = HeapAlloc(GetProcessHeap(), 0, sizeof(ICDRAWSUGGEST));
544 ICDRAWSUGGEST16 *icds16 = MapSL(*lParam1);
546 ret = icds16;
548 COPY(icds, dwFlags);
549 COPYPTR(icds, lpbiIn);
550 COPYPTR(icds, lpbiSuggest);
551 COPY(icds, dxSrc);
552 COPY(icds, dySrc);
553 COPY(icds, dxDst);
554 COPY(icds, dyDst);
555 icds->hicDecompressor = HIC_32(icds16->hicDecompressor);
557 *lParam1 = (DWORD)(icds);
558 *lParam2 = sizeof(ICDRAWSUGGEST);
560 break;
561 case ICM_DRAW:
563 ICDRAW *icd = HeapAlloc(GetProcessHeap(), 0, sizeof(ICDRAW));
564 ICDRAW *icd16 = MapSL(*lParam1);
565 ret = icd16;
567 COPY(icd, dwFlags);
568 COPYPTR(icd, lpFormat);
569 COPYPTR(icd, lpData);
570 COPY(icd, cbData);
571 COPY(icd, lTime);
573 *lParam1 = (DWORD)(icd);
574 *lParam2 = sizeof(ICDRAW);
576 break;
577 case ICM_DRAW_START:
578 case ICM_DRAW_STOP:
579 break;
580 default:
581 FIXME("%d is not yet handled. Expect a crash.\n", msg);
583 return ret;
586 #undef COPY
587 #undef COPYPTR
589 /******************************************************************
590 * MSVIDEO_UnmapMsg16To32
594 static void MSVIDEO_UnmapMsg16To32(UINT msg, LPVOID data16, LPDWORD lParam1, LPDWORD lParam2)
596 TRACE("Unmapping %d\n", msg);
598 #define UNCOPY(x, y) (x##16->y = x->y);
600 switch (msg)
602 case ICM_GETINFO:
604 ICINFO *ici = (ICINFO*)(*lParam1);
605 ICINFO16 *ici16 = (ICINFO16*)data16;
607 UNCOPY(ici, fccType);
608 UNCOPY(ici, fccHandler);
609 UNCOPY(ici, dwFlags);
610 UNCOPY(ici, dwVersion);
611 UNCOPY(ici, dwVersionICM);
612 WideCharToMultiByte( CP_ACP, 0, ici->szName, -1, ici16->szName,
613 sizeof(ici16->szName), NULL, NULL );
614 ici16->szName[sizeof(ici16->szName)-1] = 0;
615 WideCharToMultiByte( CP_ACP, 0, ici->szDescription, -1, ici16->szDescription,
616 sizeof(ici16->szDescription), NULL, NULL );
617 ici16->szDescription[sizeof(ici16->szDescription)-1] = 0;
618 /* This just gives garbage for some reason - BB
619 lstrcpynWtoA(ici16->szDriver, ici->szDriver, 128);*/
621 HeapFree(GetProcessHeap(), 0, ici);
623 break;
624 case ICM_DECOMPRESS_QUERY:
626 LPVOID* x = data16;
627 HeapFree(GetProcessHeap(), 0, x[0]);
628 if (x[1])
629 HeapFree(GetProcessHeap(), 0, x[1]);
631 break;*/
632 case ICM_COMPRESS:
633 case ICM_DECOMPRESS:
634 case ICM_DECOMPRESSEX_QUERY:
635 case ICM_DECOMPRESSEX_BEGIN:
636 case ICM_DECOMPRESSEX:
637 case ICM_DRAW_BEGIN:
638 case ICM_DRAW_SUGGESTFORMAT:
639 case ICM_DRAW:
640 HeapFree(GetProcessHeap(), 0, data16);
641 break;
642 default:
643 ERR("Unmapping unmapped msg %d\n", msg);
645 #undef UNCOPY
648 /***********************************************************************
649 * ICInfo [MSVIDEO.200]
651 BOOL16 VFWAPI ICInfo16(DWORD fccType, DWORD fccHandler, ICINFO16 *lpicinfo)
653 BOOL16 ret;
654 LPVOID lpv;
655 DWORD lParam = (DWORD)lpicinfo;
656 DWORD size = ((ICINFO*)(MapSL((SEGPTR)lpicinfo)))->dwSize;
658 /* Use the mapping functions to map the ICINFO structure */
659 lpv = MSVIDEO_MapMsg16To32(ICM_GETINFO, &lParam, &size);
661 ret = ICInfo(fccType, fccHandler, (ICINFO*)lParam);
663 MSVIDEO_UnmapMsg16To32(ICM_GETINFO, lpv, &lParam, &size);
665 return ret;
668 /******************************************************************
669 * IC_Callback3216
673 static LRESULT CALLBACK IC_Callback3216(HIC hic, HDRVR hdrv, UINT msg, DWORD lp1, DWORD lp2)
675 WINE_HIC* whic;
676 WORD args[8];
678 whic = MSVIDEO_GetHicPtr(hic);
679 if (whic)
681 DWORD ret = 0;
682 switch (msg)
684 case DRV_OPEN:
685 lp2 = (DWORD)MapLS((void*)lp2);
686 break;
688 args[7] = HIWORD(hic);
689 args[6] = LOWORD(hic);
690 args[5] = HDRVR_16(whic->hdrv);
691 args[4] = msg;
692 args[3] = HIWORD(lp1);
693 args[2] = LOWORD(lp1);
694 args[1] = HIWORD(lp2);
695 args[0] = LOWORD(lp2);
696 WOWCallback16Ex( whic->driverproc16, WCB16_PASCAL, sizeof(args), args, &ret );
698 switch (msg)
700 case DRV_OPEN:
701 UnMapLS(lp2);
702 break;
704 return ret;
706 else return ICERR_BADHANDLE;
709 /***********************************************************************
710 * ICOpenFunction [MSVIDEO.206]
712 HIC16 VFWAPI ICOpenFunction16(DWORD fccType, DWORD fccHandler, UINT16 wMode, FARPROC16 lpfnHandler)
714 HIC hic32;
716 hic32 = MSVIDEO_OpenFunction(fccType, fccHandler, wMode,
717 (DRIVERPROC)IC_Callback3216, (DWORD)lpfnHandler);
718 return HIC_16(hic32);
721 /***********************************************************************
722 * ICSendMessage [MSVIDEO.205]
724 LRESULT VFWAPI ICSendMessage16(HIC16 hic, UINT16 msg, DWORD lParam1, DWORD lParam2)
726 LRESULT ret = ICERR_BADHANDLE;
727 WINE_HIC* whic;
729 whic = MSVIDEO_GetHicPtr(HIC_32(hic));
730 if (whic)
732 /* we've got a 16 bit driver proc... call it directly */
733 if (whic->driverproc16)
735 WORD args[8];
736 DWORD result;
738 /* FIXME: original code was passing hdrv first and hic second */
739 /* but this doesn't match what IC_Callback3216 does */
740 args[7] = HIWORD(hic);
741 args[6] = LOWORD(hic);
742 args[5] = HDRVR_16(whic->hdrv);
743 args[4] = msg;
744 args[3] = HIWORD(lParam1);
745 args[2] = LOWORD(lParam1);
746 args[1] = HIWORD(lParam2);
747 args[0] = LOWORD(lParam2);
748 WOWCallback16Ex( whic->driverproc16, WCB16_PASCAL, sizeof(args), args, &result );
749 ret = result;
751 else
753 /* map the message for a 32 bit infrastructure, and pass it along */
754 void* data16 = MSVIDEO_MapMsg16To32(msg, &lParam1, &lParam2);
756 ret = MSVIDEO_SendMessage(whic, msg, lParam1, lParam2);
757 if (data16)
758 MSVIDEO_UnmapMsg16To32(msg, data16, &lParam1, &lParam2);
761 return ret;
764 /***********************************************************************
765 * VideoCapDriverDescAndVer [MSVIDEO.22]
767 DWORD WINAPI VideoCapDriverDescAndVer16(WORD nr, LPSTR buf1, WORD buf1len,
768 LPSTR buf2, WORD buf2len)
770 static const char version_info_spec[] = "\\StringFileInfo\\040904E4\\FileDescription";
771 DWORD verhandle;
772 DWORD infosize;
773 UINT subblocklen;
774 char *s, buf[2048], fn[260];
775 LPBYTE infobuf;
776 LPVOID subblock;
777 DWORD i, cnt = 0, lRet;
778 DWORD bufLen, fnLen;
779 FILETIME lastWrite;
780 HKEY hKey;
781 BOOL found = FALSE;
783 TRACE("(%d,%p,%d,%p,%d)\n", nr, buf1, buf1len, buf2, buf2len);
784 lRet = RegOpenKeyExA(HKEY_LOCAL_MACHINE, HKLM_DRIVERS32, 0, KEY_QUERY_VALUE, &hKey);
785 if (lRet == ERROR_SUCCESS)
787 RegQueryInfoKeyA( hKey, 0, 0, 0, &cnt, 0, 0, 0, 0, 0, 0, 0);
788 for (i = 0; i < cnt; i++)
790 bufLen = sizeof(buf) / sizeof(buf[0]);
791 lRet = RegEnumKeyExA(hKey, i, buf, &bufLen, 0, 0, 0, &lastWrite);
792 if (lRet != ERROR_SUCCESS) continue;
793 if (strncasecmp(buf, "vid", 3)) continue;
794 if (nr--) continue;
795 fnLen = sizeof(fn);
796 lRet = RegQueryValueExA(hKey, buf, 0, 0, (LPBYTE)fn, &fnLen);
797 if (lRet == ERROR_SUCCESS) found = TRUE;
798 break;
800 RegCloseKey( hKey );
803 /* search system.ini if not found in the registry */
804 if (!found && GetPrivateProfileStringA("drivers32", NULL, NULL, buf, sizeof(buf), "system.ini"))
806 for (s = buf; *s; s += strlen(s) + 1)
808 if (strncasecmp(s, "vid", 3)) continue;
809 if (nr--) continue;
810 if (GetPrivateProfileStringA("drivers32", s, NULL, fn, sizeof(fn), "system.ini"))
811 found = TRUE;
812 break;
816 if (!found)
818 TRACE("No more VID* entries found nr=%d\n", nr);
819 return 20;
821 infosize = GetFileVersionInfoSizeA(fn, &verhandle);
822 if (!infosize)
824 TRACE("%s has no fileversioninfo.\n", fn);
825 return 18;
827 infobuf = HeapAlloc(GetProcessHeap(), 0, infosize);
828 if (GetFileVersionInfoA(fn, verhandle, infosize, infobuf))
830 /* Yes, two space behind : */
831 /* FIXME: test for buflen */
832 snprintf(buf2, buf2len, "Version: %d.%d.%d.%d\n",
833 ((WORD*)infobuf)[0x0f],
834 ((WORD*)infobuf)[0x0e],
835 ((WORD*)infobuf)[0x11],
836 ((WORD*)infobuf)[0x10]
838 TRACE("version of %s is %s\n", fn, buf2);
840 else
842 TRACE("GetFileVersionInfoA failed for %s.\n", fn);
843 lstrcpynA(buf2, fn, buf2len); /* msvideo.dll appears to copy fn*/
845 /* FIXME: language problem? */
846 if (VerQueryValueA( infobuf,
847 version_info_spec,
848 &subblock,
849 &subblocklen
852 UINT copylen = min(subblocklen,buf1len-1);
853 memcpy(buf1, subblock, copylen);
854 buf1[copylen] = '\0';
855 TRACE("VQA returned %s\n", (LPCSTR)subblock);
857 else
859 TRACE("VQA did not return on query \\StringFileInfo\\040904E4\\FileDescription?\n");
860 lstrcpynA(buf1, fn, buf1len); /* msvideo.dll appears to copy fn*/
862 HeapFree(GetProcessHeap(), 0, infobuf);
863 return 0;
866 /******************************************************************
867 * IC_CallTo16
871 static LRESULT CALLBACK IC_CallTo16(HDRVR hdrv, HIC hic, UINT msg, LPARAM lp1, LPARAM lp2)
873 #if 0
874 WINE_HIC* whic = IC_GetPtr(hic);
875 LRESULT ret = 0;
878 if (whic->driverproc)
880 ret = whic->driverproc(hic, whic->hdrv, msg, lParam1, lParam2);
882 else
884 ret = SendDriverMessage(whic->hdrv, msg, lParam1, lParam2);
886 #else
887 FIXME("No 32=>16 conversion yet\n");
888 #endif
889 return 0;
892 /**************************************************************************
893 * DllEntryPoint (MSVIDEO.3)
895 * MSVIDEO DLL entry point
898 BOOL WINAPI VIDEO_LibMain(DWORD fdwReason, HINSTANCE hinstDLL, WORD ds,
899 WORD wHeapSize, DWORD dwReserved1, WORD wReserved2)
901 switch (fdwReason)
903 case DLL_PROCESS_ATTACH:
904 /* hook in our 16 bit management functions */
905 pFnCallTo16 = IC_CallTo16;
906 break;
907 case DLL_PROCESS_DETACH:
908 /* remove our 16 bit management functions */
909 pFnCallTo16 = NULL;
910 break;
911 case DLL_THREAD_ATTACH:
912 case DLL_THREAD_DETACH:
913 break;
915 return TRUE;