2 * Copyright 2002 Michael Günnewig
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 #define COM_NO_WINDOWS_H
34 #include "avifile_private.h"
36 #include "wine/debug.h"
38 WINE_DEFAULT_DEBUG_CHANNEL(avifile);
40 /***********************************************************************/
42 static HRESULT WINAPI ACMStream_fnQueryInterface(IAVIStream*iface,REFIID refiid,LPVOID *obj);
43 static ULONG WINAPI ACMStream_fnAddRef(IAVIStream*iface);
44 static ULONG WINAPI ACMStream_fnRelease(IAVIStream* iface);
45 static HRESULT WINAPI ACMStream_fnCreate(IAVIStream*iface,LPARAM lParam1,LPARAM lParam2);
46 static HRESULT WINAPI ACMStream_fnInfo(IAVIStream*iface,AVISTREAMINFOW *psi,LONG size);
47 static LONG WINAPI ACMStream_fnFindSample(IAVIStream*iface,LONG pos,LONG flags);
48 static HRESULT WINAPI ACMStream_fnReadFormat(IAVIStream*iface,LONG pos,LPVOID format,LONG *formatsize);
49 static HRESULT WINAPI ACMStream_fnSetFormat(IAVIStream*iface,LONG pos,LPVOID format,LONG formatsize);
50 static HRESULT WINAPI ACMStream_fnRead(IAVIStream*iface,LONG start,LONG samples,LPVOID buffer,LONG buffersize,LONG *bytesread,LONG *samplesread);
51 static HRESULT WINAPI ACMStream_fnWrite(IAVIStream*iface,LONG start,LONG samples,LPVOID buffer,LONG buffersize,DWORD flags,LONG *sampwritten,LONG *byteswritten);
52 static HRESULT WINAPI ACMStream_fnDelete(IAVIStream*iface,LONG start,LONG samples);
53 static HRESULT WINAPI ACMStream_fnReadData(IAVIStream*iface,DWORD fcc,LPVOID lp,LONG *lpread);
54 static HRESULT WINAPI ACMStream_fnWriteData(IAVIStream*iface,DWORD fcc,LPVOID lp,LONG size);
55 static HRESULT WINAPI ACMStream_fnSetInfo(IAVIStream*iface,AVISTREAMINFOW*info,LONG infolen);
57 struct IAVIStreamVtbl iacmst = {
58 ACMStream_fnQueryInterface,
63 ACMStream_fnFindSample,
64 ACMStream_fnReadFormat,
65 ACMStream_fnSetFormat,
70 ACMStream_fnWriteData,
74 typedef struct _IAVIStreamImpl {
76 IAVIStreamVtbl *lpVtbl;
79 /* IAVIStream stuff */
85 LPWAVEFORMATEX lpInFormat;
88 LPWAVEFORMATEX lpOutFormat;
91 ACMSTREAMHEADER acmStreamHdr;
94 /***********************************************************************/
96 #define CONVERT_STREAM_to_THIS(a) { \
97 acmStreamSize(This->has,(a)*This->lpInFormat->nBlockAlign,\
98 &(a), ACM_STREAMSIZEF_SOURCE); \
99 (a) /= This->lpOutFormat->nBlockAlign; }
100 #define CONVERT_THIS_to_STREAM(a) { \
101 acmStreamSize(This->has,(a)*This->lpOutFormat->nBlockAlign,\
102 &(a), ACM_STREAMSIZEF_DESTINATION); \
103 (a) /= This->lpInFormat->nBlockAlign; }
105 static HRESULT AVIFILE_OpenCompressor(IAVIStreamImpl *This);
107 HRESULT AVIFILE_CreateACMStream(REFIID riid, LPVOID *ppv)
109 IAVIStreamImpl *pstream;
112 assert(riid != NULL && ppv != NULL);
116 pstream = (IAVIStreamImpl*)LocalAlloc(LPTR, sizeof(IAVIStreamImpl));
118 return AVIERR_MEMORY;
120 pstream->lpVtbl = &iacmst;
122 hr = IAVIStream_QueryInterface((IAVIStream*)pstream, riid, ppv);
124 LocalFree((HLOCAL)pstream);
129 static HRESULT WINAPI ACMStream_fnQueryInterface(IAVIStream *iface,
130 REFIID refiid, LPVOID *obj)
132 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
134 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(refiid), obj);
136 if (IsEqualGUID(&IID_IUnknown, refiid) ||
137 IsEqualGUID(&IID_IAVIStream, refiid)) {
139 IAVIStream_AddRef(iface);
144 return OLE_E_ENUM_NOMORE;
147 static ULONG WINAPI ACMStream_fnAddRef(IAVIStream *iface)
149 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
150 ULONG ref = InterlockedIncrement(&This->ref);
152 TRACE("(%p) -> %ld\n", iface, ref);
154 /* also add reference to the nested stream */
155 if (This->pStream != NULL)
156 IAVIStream_AddRef(This->pStream);
161 static ULONG WINAPI ACMStream_fnRelease(IAVIStream* iface)
163 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
164 ULONG ref = InterlockedDecrement(&This->ref);
166 TRACE("(%p) -> %ld\n", iface, ref);
170 if (This->has != NULL) {
171 if (This->acmStreamHdr.fdwStatus & ACMSTREAMHEADER_STATUSF_PREPARED)
172 acmStreamUnprepareHeader(This->has, &This->acmStreamHdr, 0);
173 acmStreamClose(This->has, 0);
176 if (This->acmStreamHdr.pbSrc != NULL) {
177 GlobalFreePtr(This->acmStreamHdr.pbSrc);
178 This->acmStreamHdr.pbSrc = NULL;
180 if (This->acmStreamHdr.pbDst != NULL) {
181 GlobalFreePtr(This->acmStreamHdr.pbDst);
182 This->acmStreamHdr.pbDst = NULL;
184 if (This->lpInFormat != NULL) {
185 GlobalFreePtr(This->lpInFormat);
186 This->lpInFormat = NULL;
187 This->cbInFormat = 0;
189 if (This->lpOutFormat != NULL) {
190 GlobalFreePtr(This->lpOutFormat);
191 This->lpOutFormat = NULL;
192 This->cbOutFormat = 0;
194 if (This->pStream != NULL) {
195 IAVIStream_Release(This->pStream);
196 This->pStream = NULL;
198 LocalFree((HLOCAL)This);
203 /* also release reference to the nested stream */
204 if (This->pStream != NULL)
205 IAVIStream_Release(This->pStream);
210 /* lParam1: PAVISTREAM
211 * lParam2: LPAVICOMPRESSOPTIONS -- even if doc's say LPWAVEFORMAT
213 static HRESULT WINAPI ACMStream_fnCreate(IAVIStream *iface, LPARAM lParam1,
216 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
218 TRACE("(%p,0x%08lX,0x%08lX)\n", iface, lParam1, lParam2);
220 /* check for swapped parameters */
221 if ((LPVOID)lParam1 != NULL &&
222 ((LPAVICOMPRESSOPTIONS)lParam1)->fccType == streamtypeAUDIO) {
223 register LPARAM tmp = lParam1;
229 if ((LPVOID)lParam1 == NULL)
230 return AVIERR_BADPARAM;
232 IAVIStream_Info((PAVISTREAM)lParam1, &This->sInfo, sizeof(This->sInfo));
233 if (This->sInfo.fccType != streamtypeAUDIO)
234 return AVIERR_ERROR; /* error in registry or AVIMakeCompressedStream */
236 This->sInfo.fccHandler = 0; /* be paranoid */
238 /* FIXME: check ACM version? Which version does we need? */
240 if ((LPVOID)lParam2 != NULL) {
241 /* We only need the format from the compress-options */
242 if (((LPAVICOMPRESSOPTIONS)lParam2)->fccType == streamtypeAUDIO)
243 lParam2 = (LPARAM)((LPAVICOMPRESSOPTIONS)lParam2)->lpFormat;
245 if (((LPWAVEFORMATEX)lParam2)->wFormatTag != WAVE_FORMAT_PCM)
246 This->cbOutFormat = sizeof(WAVEFORMATEX) + ((LPWAVEFORMATEX)lParam2)->cbSize;
248 This->cbOutFormat = sizeof(PCMWAVEFORMAT);
250 This->lpOutFormat = (LPWAVEFORMATEX)GlobalAllocPtr(GHND, This->cbOutFormat);
251 if (This->lpOutFormat == NULL)
252 return AVIERR_MEMORY;
254 memcpy(This->lpOutFormat, (LPVOID)lParam2, This->cbOutFormat);
256 This->lpOutFormat = NULL;
257 This->cbOutFormat = 0;
260 This->pStream = (PAVISTREAM)lParam1;
261 IAVIStream_AddRef(This->pStream);
266 static HRESULT WINAPI ACMStream_fnInfo(IAVIStream *iface,LPAVISTREAMINFOW psi,
269 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
271 TRACE("(%p,%p,%ld)\n", iface, psi, size);
274 return AVIERR_BADPARAM;
276 return AVIERR_BADSIZE;
278 /* Need codec to correct some values in structure */
279 if (This->has == NULL) {
280 HRESULT hr = AVIFILE_OpenCompressor(This);
286 memcpy(psi, &This->sInfo, min(size, (LONG)sizeof(This->sInfo)));
288 if (size < (LONG)sizeof(This->sInfo))
289 return AVIERR_BUFFERTOOSMALL;
293 static LONG WINAPI ACMStream_fnFindSample(IAVIStream *iface, LONG pos,
296 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
298 TRACE("(%p,%ld,0x%08lX)\n",iface,pos,flags);
300 if (flags & FIND_FROM_START) {
301 pos = This->sInfo.dwStart;
302 flags &= ~(FIND_FROM_START|FIND_PREV);
306 /* convert pos from our 'space' to This->pStream's one */
307 CONVERT_THIS_to_STREAM(pos);
310 pos = IAVIStream_FindSample(This->pStream, pos, flags);
313 /* convert pos back to our 'space' if it's no size or physical pos */
314 if ((flags & FIND_RET) == 0)
315 CONVERT_STREAM_to_THIS(pos);
321 static HRESULT WINAPI ACMStream_fnReadFormat(IAVIStream *iface, LONG pos,
322 LPVOID format, LONG *formatsize)
324 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
326 TRACE("(%p,%ld,%p,%p)\n", iface, pos, format, formatsize);
328 if (formatsize == NULL)
329 return AVIERR_BADPARAM;
331 if (This->has == NULL) {
332 HRESULT hr = AVIFILE_OpenCompressor(This);
338 /* only interested in needed buffersize? */
339 if (format == NULL || *formatsize <= 0) {
340 *formatsize = This->cbOutFormat;
345 /* copy initial format (only as much as will fit) */
346 memcpy(format, This->lpOutFormat, min(*formatsize, This->cbOutFormat));
347 if (*formatsize < This->cbOutFormat) {
348 *formatsize = This->cbOutFormat;
349 return AVIERR_BUFFERTOOSMALL;
352 *formatsize = This->cbOutFormat;
356 static HRESULT WINAPI ACMStream_fnSetFormat(IAVIStream *iface, LONG pos,
357 LPVOID format, LONG formatsize)
359 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
363 TRACE("(%p,%ld,%p,%ld)\n", iface, pos, format, formatsize);
365 /* check parameters */
366 if (format == NULL || formatsize <= 0)
367 return AVIERR_BADPARAM;
369 /* Input format already known?
370 * Changing is unsupported, but be quiet if it's the same */
371 if (This->lpInFormat != NULL) {
372 if (This->cbInFormat != formatsize ||
373 memcmp(format, This->lpInFormat, formatsize) != 0)
374 return AVIERR_UNSUPPORTED;
379 /* Does the nested stream support writing? */
380 if ((This->sInfo.dwCaps & AVIFILECAPS_CANWRITE) == 0)
381 return AVIERR_READONLY;
383 This->lpInFormat = (LPWAVEFORMATEX)GlobalAllocPtr(GMEM_MOVEABLE, formatsize);
384 if (This->lpInFormat == NULL)
385 return AVIERR_MEMORY;
386 This->cbInFormat = formatsize;
387 memcpy(This->lpInFormat, format, formatsize);
389 /* initialize formats and get compressor */
390 hr = AVIFILE_OpenCompressor(This);
394 CONVERT_THIS_to_STREAM(pos);
396 /* tell the nested stream the new format */
397 return IAVIStream_SetFormat(This->pStream, pos, This->lpOutFormat,
401 static HRESULT WINAPI ACMStream_fnRead(IAVIStream *iface, LONG start,
402 LONG samples, LPVOID buffer,
403 LONG buffersize, LPLONG bytesread,
406 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
411 TRACE("(%p,%ld,%ld,%p,%ld,%p,%p)\n", iface, start, samples, buffer,
412 buffersize, bytesread, samplesread);
414 /* clear return parameters if given */
415 if (bytesread != NULL)
417 if (samplesread != NULL)
420 /* Do we have our compressor? */
421 if (This->has == NULL) {
422 hr = AVIFILE_OpenCompressor(This);
428 /* only need to pass through? */
429 if (This->cbInFormat == This->cbOutFormat &&
430 memcmp(This->lpInFormat, This->lpOutFormat, This->cbInFormat) == 0) {
431 return IAVIStream_Read(This->pStream, start, samples, buffer, buffersize,
432 bytesread, samplesread);
435 /* read as much as fit? */
437 samples = buffersize / This->lpOutFormat->nBlockAlign;
438 /* limit to buffersize */
439 if (samples * This->lpOutFormat->nBlockAlign > buffersize)
440 samples = buffersize / This->lpOutFormat->nBlockAlign;
442 /* only return needed size? */
443 if (buffer == NULL || buffersize <= 0 || samples == 0) {
444 if (bytesread == NULL && samplesread == NULL)
445 return AVIERR_BADPARAM;
447 if (bytesread != NULL)
448 *bytesread = samples * This->lpOutFormat->nBlockAlign;
449 if (samplesread != NULL)
450 *samplesread = samples;
455 /* map our positions to pStream positions */
456 CONVERT_THIS_to_STREAM(start);
458 /* our needed internal buffersize */
459 size = samples * This->lpInFormat->nBlockAlign;
461 /* Need to free destination buffer used for writing? */
462 if (This->acmStreamHdr.pbDst != NULL) {
463 GlobalFreePtr(This->acmStreamHdr.pbDst);
464 This->acmStreamHdr.pbDst = NULL;
465 This->acmStreamHdr.dwDstUser = 0;
468 /* need bigger source buffer? */
469 if (This->acmStreamHdr.pbSrc == NULL ||
470 This->acmStreamHdr.dwSrcUser < size) {
471 if (This->acmStreamHdr.pbSrc == NULL)
472 This->acmStreamHdr.pbSrc = GlobalAllocPtr(GMEM_MOVEABLE, size);
474 This->acmStreamHdr.pbSrc = GlobalReAllocPtr(This->acmStreamHdr.pbSrc,
475 size, GMEM_MOVEABLE);
476 if (This->acmStreamHdr.pbSrc == NULL)
477 return AVIERR_MEMORY;
478 This->acmStreamHdr.dwSrcUser = size;
481 This->acmStreamHdr.cbStruct = sizeof(This->acmStreamHdr);
482 This->acmStreamHdr.cbSrcLengthUsed = 0;
483 This->acmStreamHdr.cbDstLengthUsed = 0;
484 This->acmStreamHdr.cbSrcLength = size;
486 /* read source data */
487 hr = IAVIStream_Read(This->pStream, start, -1, This->acmStreamHdr.pbSrc,
488 This->acmStreamHdr.cbSrcLength,
489 &This->acmStreamHdr.cbSrcLength, NULL);
490 if (FAILED(hr) || This->acmStreamHdr.cbSrcLength == 0)
493 /* need to prepare stream? */
494 This->acmStreamHdr.pbDst = buffer;
495 This->acmStreamHdr.cbDstLength = buffersize;
496 if ((This->acmStreamHdr.fdwStatus & ACMSTREAMHEADER_STATUSF_PREPARED) == 0) {
497 if (acmStreamPrepareHeader(This->has, &This->acmStreamHdr, 0) != S_OK) {
498 This->acmStreamHdr.pbDst = NULL;
499 This->acmStreamHdr.cbDstLength = 0;
500 return AVIERR_COMPRESSOR;
504 /* now do the conversion */
505 /* FIXME: use ACM_CONVERTF_* flags */
506 if (acmStreamConvert(This->has, &This->acmStreamHdr, 0) != S_OK)
507 hr = AVIERR_COMPRESSOR;
509 This->acmStreamHdr.pbDst = NULL;
510 This->acmStreamHdr.cbDstLength = 0;
512 /* fill out return parameters if given */
513 if (bytesread != NULL)
514 *bytesread = This->acmStreamHdr.cbDstLengthUsed;
515 if (samplesread != NULL)
517 This->acmStreamHdr.cbDstLengthUsed / This->lpOutFormat->nBlockAlign;
522 static HRESULT WINAPI ACMStream_fnWrite(IAVIStream *iface, LONG start,
523 LONG samples, LPVOID buffer,
524 LONG buffersize, DWORD flags,
528 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
533 TRACE("(%p,%ld,%ld,%p,%ld,0x%08lX,%p,%p)\n", iface, start, samples,
534 buffer, buffersize, flags, sampwritten, byteswritten);
536 /* clear return parameters if given */
537 if (sampwritten != NULL)
539 if (byteswritten != NULL)
542 /* check parameters */
543 if (buffer == NULL && (buffersize > 0 || samples > 0))
544 return AVIERR_BADPARAM;
546 /* Have we write capability? */
547 if ((This->sInfo.dwCaps & AVIFILECAPS_CANWRITE) == 0)
548 return AVIERR_READONLY;
550 /* also need a compressor */
551 if (This->has == NULL)
552 return AVIERR_NOCOMPRESSOR;
554 /* map our sizes to pStream sizes */
556 CONVERT_THIS_to_STREAM(size);
557 CONVERT_THIS_to_STREAM(start);
559 /* no bytes to write? -- short circuit */
561 return IAVIStream_Write(This->pStream, -1, samples, buffer, size,
562 flags, sampwritten, byteswritten);
565 /* Need to free source buffer used for reading? */
566 if (This->acmStreamHdr.pbSrc != NULL) {
567 GlobalFreePtr(This->acmStreamHdr.pbSrc);
568 This->acmStreamHdr.pbSrc = NULL;
569 This->acmStreamHdr.dwSrcUser = 0;
572 /* Need bigger destination buffer? */
573 if (This->acmStreamHdr.pbDst == NULL ||
574 This->acmStreamHdr.dwDstUser < size) {
575 if (This->acmStreamHdr.pbDst == NULL)
576 This->acmStreamHdr.pbDst = GlobalAllocPtr(GMEM_MOVEABLE, size);
578 This->acmStreamHdr.pbDst = GlobalReAllocPtr(This->acmStreamHdr.pbDst,
579 size, GMEM_MOVEABLE);
580 if (This->acmStreamHdr.pbDst == NULL)
581 return AVIERR_MEMORY;
582 This->acmStreamHdr.dwDstUser = size;
584 This->acmStreamHdr.cbStruct = sizeof(This->acmStreamHdr);
585 This->acmStreamHdr.cbSrcLengthUsed = 0;
586 This->acmStreamHdr.cbDstLengthUsed = 0;
587 This->acmStreamHdr.cbDstLength = This->acmStreamHdr.dwDstUser;
589 /* need to prepare stream? */
590 This->acmStreamHdr.pbSrc = buffer;
591 This->acmStreamHdr.cbSrcLength = buffersize;
592 if ((This->acmStreamHdr.fdwStatus & ACMSTREAMHEADER_STATUSF_PREPARED) == 0) {
593 if (acmStreamPrepareHeader(This->has, &This->acmStreamHdr, 0) != S_OK) {
594 This->acmStreamHdr.pbSrc = NULL;
595 This->acmStreamHdr.cbSrcLength = 0;
596 return AVIERR_COMPRESSOR;
600 /* now do the conversion */
601 /* FIXME: use ACM_CONVERTF_* flags */
602 if (acmStreamConvert(This->has, &This->acmStreamHdr, 0) != S_OK)
603 hr = AVIERR_COMPRESSOR;
607 This->acmStreamHdr.pbSrc = NULL;
608 This->acmStreamHdr.cbSrcLength = 0;
613 return IAVIStream_Write(This->pStream,-1,This->acmStreamHdr.cbDstLengthUsed /
614 This->lpOutFormat->nBlockAlign,This->acmStreamHdr.pbDst,
615 This->acmStreamHdr.cbDstLengthUsed,flags,sampwritten,
619 static HRESULT WINAPI ACMStream_fnDelete(IAVIStream *iface, LONG start,
622 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
624 TRACE("(%p,%ld,%ld)\n", iface, start, samples);
626 /* check parameters */
627 if (start < 0 || samples < 0)
628 return AVIERR_BADPARAM;
630 /* Delete before start of stream? */
631 if ((DWORD)(start + samples) < This->sInfo.dwStart)
634 /* Delete after end of stream? */
635 if ((DWORD)start > This->sInfo.dwLength)
638 /* For the rest we need write capability */
639 if ((This->sInfo.dwCaps & AVIFILECAPS_CANWRITE) == 0)
640 return AVIERR_READONLY;
642 /* A compressor is also necessary */
643 if (This->has == NULL)
644 return AVIERR_NOCOMPRESSOR;
646 /* map our positions to pStream positions */
647 CONVERT_THIS_to_STREAM(start);
648 CONVERT_THIS_to_STREAM(samples);
650 return IAVIStream_Delete(This->pStream, start, samples);
653 static HRESULT WINAPI ACMStream_fnReadData(IAVIStream *iface, DWORD fcc,
654 LPVOID lp, LPLONG lpread)
656 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
658 TRACE("(%p,0x%08lX,%p,%p)\n", iface, fcc, lp, lpread);
660 assert(This->pStream != NULL);
662 return IAVIStream_ReadData(This->pStream, fcc, lp, lpread);
665 static HRESULT WINAPI ACMStream_fnWriteData(IAVIStream *iface, DWORD fcc,
666 LPVOID lp, LONG size)
668 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
670 TRACE("(%p,0x%08lx,%p,%ld)\n", iface, fcc, lp, size);
672 assert(This->pStream != NULL);
674 return IAVIStream_WriteData(This->pStream, fcc, lp, size);
677 static HRESULT WINAPI ACMStream_fnSetInfo(IAVIStream *iface,
678 LPAVISTREAMINFOW info, LONG infolen)
680 FIXME("(%p,%p,%ld): stub\n", iface, info, infolen);
685 /***********************************************************************/
687 static HRESULT AVIFILE_OpenCompressor(IAVIStreamImpl *This)
692 assert(This != NULL);
693 assert(This->pStream != NULL);
695 if (This->has != NULL)
698 if (This->lpInFormat == NULL) {
699 /* decode or encode the data from pStream */
700 hr = AVIStreamFormatSize(This->pStream, This->sInfo.dwStart, &This->cbInFormat);
703 This->lpInFormat = (LPWAVEFORMATEX)GlobalAllocPtr(GMEM_MOVEABLE, This->cbInFormat);
704 if (This->lpInFormat == NULL)
705 return AVIERR_MEMORY;
707 hr = IAVIStream_ReadFormat(This->pStream, This->sInfo.dwStart,
708 This->lpInFormat, &This->cbInFormat);
712 if (This->lpOutFormat == NULL) {
713 /* we must decode to default format */
714 This->cbOutFormat = sizeof(PCMWAVEFORMAT);
715 This->lpOutFormat = (LPWAVEFORMATEX)GlobalAllocPtr(GHND, This->cbOutFormat);
716 if (This->lpOutFormat == NULL)
717 return AVIERR_MEMORY;
719 This->lpOutFormat->wFormatTag = WAVE_FORMAT_PCM;
720 if (acmFormatSuggest(NULL, This->lpInFormat, This->lpOutFormat,
721 This->cbOutFormat, ACM_FORMATSUGGESTF_WFORMATTAG) != S_OK)
722 return AVIERR_NOCOMPRESSOR;
724 } else if (This->lpOutFormat == NULL)
725 return AVIERR_ERROR; /* To what should I encode? */
727 if (acmStreamOpen(&This->has, NULL, This->lpInFormat, This->lpOutFormat,
728 NULL, 0, 0, ACM_STREAMOPENF_NONREALTIME) != S_OK)
729 return AVIERR_NOCOMPRESSOR;
731 /* update AVISTREAMINFO structure */
732 This->sInfo.dwSampleSize = This->lpOutFormat->nBlockAlign;
733 This->sInfo.dwScale = This->lpOutFormat->nBlockAlign;
734 This->sInfo.dwRate = This->lpOutFormat->nAvgBytesPerSec;
735 This->sInfo.dwQuality = (DWORD)ICQUALITY_DEFAULT;
736 SetRectEmpty(&This->sInfo.rcFrame);
738 /* convert positions ansd sizes to output format */
739 CONVERT_STREAM_to_THIS(This->sInfo.dwStart);
740 CONVERT_STREAM_to_THIS(This->sInfo.dwLength);
741 CONVERT_STREAM_to_THIS(This->sInfo.dwSuggestedBufferSize);