d3dxof: Uncomment some traces and use better name for some variables.
[wine/multimedia.git] / dlls / avifil32 / avifile.c
blobab60c3a30bacd93aecfa590cd16ed92aa7a03a3f
1 /*
2 * Copyright 1999 Marcus Meissner
3 * Copyright 2002-2003 Michael Günnewig
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20 /* TODO:
21 * - IAVIStreaming interface is missing for the IAVIStreamImpl
22 * - IAVIStream_fnFindSample: FIND_INDEX isn't supported.
23 * - IAVIStream_fnReadFormat: formatchanges aren't read in.
24 * - IAVIStream_fnDelete: a stub.
25 * - IAVIStream_fnSetInfo: a stub.
26 * - make thread safe
28 * KNOWN Bugs:
29 * - native version can hangup when reading a file generated with this DLL.
30 * When index is missing it works, but index seems to be okay.
33 #define COBJMACROS
34 #include <assert.h>
35 #include <stdarg.h>
37 #include "windef.h"
38 #include "winbase.h"
39 #include "wingdi.h"
40 #include "winuser.h"
41 #include "winnls.h"
42 #include "winerror.h"
43 #include "mmsystem.h"
44 #include "vfw.h"
46 #include "avifile_private.h"
47 #include "extrachunk.h"
49 #include "wine/unicode.h"
50 #include "wine/debug.h"
52 WINE_DEFAULT_DEBUG_CHANNEL(avifile);
54 #ifndef IDX_PER_BLOCK
55 #define IDX_PER_BLOCK 2730
56 #endif
58 static HRESULT WINAPI IAVIStream_fnQueryInterface(IAVIStream*iface,REFIID refiid,LPVOID *obj);
59 static ULONG WINAPI IAVIStream_fnAddRef(IAVIStream*iface);
60 static ULONG WINAPI IAVIStream_fnRelease(IAVIStream* iface);
61 static HRESULT WINAPI IAVIStream_fnCreate(IAVIStream*iface,LPARAM lParam1,LPARAM lParam2);
62 static HRESULT WINAPI IAVIStream_fnInfo(IAVIStream*iface,AVISTREAMINFOW *psi,LONG size);
63 static LONG WINAPI IAVIStream_fnFindSample(IAVIStream*iface,LONG pos,LONG flags);
64 static HRESULT WINAPI IAVIStream_fnReadFormat(IAVIStream*iface,LONG pos,LPVOID format,LONG *formatsize);
65 static HRESULT WINAPI IAVIStream_fnSetFormat(IAVIStream*iface,LONG pos,LPVOID format,LONG formatsize);
66 static HRESULT WINAPI IAVIStream_fnRead(IAVIStream*iface,LONG start,LONG samples,LPVOID buffer,LONG buffersize,LONG *bytesread,LONG *samplesread);
67 static HRESULT WINAPI IAVIStream_fnWrite(IAVIStream*iface,LONG start,LONG samples,LPVOID buffer,LONG buffersize,DWORD flags,LONG *sampwritten,LONG *byteswritten);
68 static HRESULT WINAPI IAVIStream_fnDelete(IAVIStream*iface,LONG start,LONG samples);
69 static HRESULT WINAPI IAVIStream_fnReadData(IAVIStream*iface,DWORD fcc,LPVOID lp,LONG *lpread);
70 static HRESULT WINAPI IAVIStream_fnWriteData(IAVIStream*iface,DWORD fcc,LPVOID lp,LONG size);
71 static HRESULT WINAPI IAVIStream_fnSetInfo(IAVIStream*iface,AVISTREAMINFOW*info,LONG infolen);
73 static const struct IAVIStreamVtbl iavist = {
74 IAVIStream_fnQueryInterface,
75 IAVIStream_fnAddRef,
76 IAVIStream_fnRelease,
77 IAVIStream_fnCreate,
78 IAVIStream_fnInfo,
79 IAVIStream_fnFindSample,
80 IAVIStream_fnReadFormat,
81 IAVIStream_fnSetFormat,
82 IAVIStream_fnRead,
83 IAVIStream_fnWrite,
84 IAVIStream_fnDelete,
85 IAVIStream_fnReadData,
86 IAVIStream_fnWriteData,
87 IAVIStream_fnSetInfo
90 typedef struct _IAVIFileImpl IAVIFileImpl;
92 typedef struct _IAVIStreamImpl {
93 /* IUnknown stuff */
94 const IAVIStreamVtbl *lpVtbl;
95 LONG ref;
97 /* IAVIStream stuff */
98 IAVIFileImpl *paf;
99 DWORD nStream; /* the n-th stream in file */
100 AVISTREAMINFOW sInfo;
102 LPVOID lpFormat;
103 DWORD cbFormat;
105 LPVOID lpHandlerData;
106 DWORD cbHandlerData;
108 EXTRACHUNKS extra;
110 LPDWORD lpBuffer;
111 DWORD cbBuffer; /* size of lpBuffer */
112 DWORD dwCurrentFrame; /* frame/block currently in lpBuffer */
114 LONG lLastFrame; /* last correct index in idxFrames */
115 AVIINDEXENTRY *idxFrames;
116 DWORD nIdxFrames; /* upper index limit of idxFrames */
117 AVIINDEXENTRY *idxFmtChanges;
118 DWORD nIdxFmtChanges; /* upper index limit of idxFmtChanges */
119 } IAVIStreamImpl;
121 struct _IAVIFileImpl {
122 IUnknown IUnknown_inner;
123 IAVIFile IAVIFile_iface;
124 IPersistFile IPersistFile_iface;
125 IUnknown *outer_unk;
126 LONG ref;
128 AVIFILEINFOW fInfo;
129 IAVIStreamImpl *ppStreams[MAX_AVISTREAMS];
131 EXTRACHUNKS fileextra;
133 DWORD dwMoviChunkPos; /* some stuff for saving ... */
134 DWORD dwIdxChunkPos;
135 DWORD dwNextFramePos;
136 DWORD dwInitialFrames;
138 MMCKINFO ckLastRecord;
139 AVIINDEXENTRY *idxRecords; /* won't be updated while loading */
140 DWORD nIdxRecords; /* current fill level */
141 DWORD cbIdxRecords; /* size of idxRecords */
143 /* IPersistFile stuff ... */
144 HMMIO hmmio;
145 LPWSTR szFileName;
146 UINT uMode;
147 BOOL fDirty;
150 static inline IAVIFileImpl *impl_from_IUnknown(IUnknown *iface)
152 return CONTAINING_RECORD(iface, IAVIFileImpl, IUnknown_inner);
155 static inline IAVIFileImpl *impl_from_IAVIFile(IAVIFile *iface)
157 return CONTAINING_RECORD(iface, IAVIFileImpl, IAVIFile_iface);
160 static inline IAVIFileImpl *impl_from_IPersistFile(IPersistFile *iface)
162 return CONTAINING_RECORD(iface, IAVIFileImpl, IPersistFile_iface);
165 /***********************************************************************/
167 static HRESULT AVIFILE_AddFrame(IAVIStreamImpl *This, DWORD ckid, DWORD size,
168 DWORD offset, DWORD flags);
169 static HRESULT AVIFILE_AddRecord(IAVIFileImpl *This);
170 static DWORD AVIFILE_ComputeMoviStart(IAVIFileImpl *This);
171 static void AVIFILE_ConstructAVIStream(IAVIFileImpl *paf, DWORD nr,
172 const AVISTREAMINFOW *asi);
173 static void AVIFILE_DestructAVIStream(IAVIStreamImpl *This);
174 static HRESULT AVIFILE_LoadFile(IAVIFileImpl *This);
175 static HRESULT AVIFILE_LoadIndex(const IAVIFileImpl *This, DWORD size, DWORD offset);
176 static HRESULT AVIFILE_ParseIndex(const IAVIFileImpl *This, AVIINDEXENTRY *lp,
177 LONG count, DWORD pos, BOOL *bAbsolute);
178 static HRESULT AVIFILE_ReadBlock(IAVIStreamImpl *This, DWORD start,
179 LPVOID buffer, DWORD size);
180 static void AVIFILE_SamplesToBlock(const IAVIStreamImpl *This, LPLONG pos,
181 LPLONG offset);
182 static HRESULT AVIFILE_SaveFile(IAVIFileImpl *This);
183 static HRESULT AVIFILE_SaveIndex(const IAVIFileImpl *This);
184 static ULONG AVIFILE_SearchStream(const IAVIFileImpl *This, DWORD fccType,
185 LONG lSkip);
186 static void AVIFILE_UpdateInfo(IAVIFileImpl *This);
187 static HRESULT AVIFILE_WriteBlock(IAVIStreamImpl *This, DWORD block,
188 FOURCC ckid, DWORD flags, LPCVOID buffer,
189 LONG size);
191 static HRESULT WINAPI IUnknown_fnQueryInterface(IUnknown *iface, REFIID riid, void **ppv)
193 IAVIFileImpl *This = impl_from_IUnknown(iface);
195 TRACE("(%p,%s,%p)\n", This, debugstr_guid(riid), ppv);
197 if (!ppv) {
198 WARN("invalid parameter\n");
199 return E_INVALIDARG;
201 *ppv = NULL;
203 if (IsEqualIID(riid, &IID_IUnknown))
204 *ppv = &This->IUnknown_inner;
205 else if (IsEqualIID(riid, &IID_IAVIFile))
206 *ppv = &This->IAVIFile_iface;
207 else if (IsEqualGUID(riid, &IID_IPersistFile))
208 *ppv = &This->IPersistFile_iface;
209 else {
210 WARN("unknown IID %s\n", debugstr_guid(riid));
211 return E_NOINTERFACE;
214 /* Violation of the COM aggregation ref counting rule */
215 IUnknown_AddRef(&This->IUnknown_inner);
216 return S_OK;
219 static ULONG WINAPI IUnknown_fnAddRef(IUnknown *iface)
221 IAVIFileImpl *This = impl_from_IUnknown(iface);
222 ULONG ref = InterlockedIncrement(&This->ref);
224 TRACE("(%p) ref=%d\n", This, ref);
226 return ref;
229 static ULONG WINAPI IUnknown_fnRelease(IUnknown *iface)
231 IAVIFileImpl *This = impl_from_IUnknown(iface);
232 ULONG ref = InterlockedDecrement(&This->ref);
233 UINT i;
235 TRACE("(%p) ref=%d\n", This, ref);
237 if (!ref) {
238 if (This->fDirty)
239 AVIFILE_SaveFile(This);
241 for (i = 0; i < This->fInfo.dwStreams; i++) {
242 if (This->ppStreams[i] != NULL) {
243 if (This->ppStreams[i]->ref != 0)
244 ERR(": someone has still %u reference to stream %u (%p)!\n",
245 This->ppStreams[i]->ref, i, This->ppStreams[i]);
246 AVIFILE_DestructAVIStream(This->ppStreams[i]);
247 HeapFree(GetProcessHeap(), 0, This->ppStreams[i]);
248 This->ppStreams[i] = NULL;
252 if (This->idxRecords != NULL) {
253 HeapFree(GetProcessHeap(), 0, This->idxRecords);
254 This->idxRecords = NULL;
255 This->nIdxRecords = 0;
258 if (This->fileextra.lp != NULL) {
259 HeapFree(GetProcessHeap(), 0, This->fileextra.lp);
260 This->fileextra.lp = NULL;
261 This->fileextra.cb = 0;
264 HeapFree(GetProcessHeap(), 0, This->szFileName);
265 This->szFileName = NULL;
267 if (This->hmmio != NULL) {
268 mmioClose(This->hmmio, 0);
269 This->hmmio = NULL;
272 HeapFree(GetProcessHeap(), 0, This);
274 return ref;
277 static const IUnknownVtbl unk_vtbl =
279 IUnknown_fnQueryInterface,
280 IUnknown_fnAddRef,
281 IUnknown_fnRelease
284 static HRESULT WINAPI IAVIFile_fnQueryInterface(IAVIFile *iface, REFIID riid, void **ppv)
286 IAVIFileImpl *This = impl_from_IAVIFile(iface);
288 return IUnknown_QueryInterface(This->outer_unk, riid, ppv);
291 static ULONG WINAPI IAVIFile_fnAddRef(IAVIFile *iface)
293 IAVIFileImpl *This = impl_from_IAVIFile(iface);
295 return IUnknown_AddRef(This->outer_unk);
298 static ULONG WINAPI IAVIFile_fnRelease(IAVIFile *iface)
300 IAVIFileImpl *This = impl_from_IAVIFile(iface);
302 return IUnknown_Release(This->outer_unk);
305 static HRESULT WINAPI IAVIFile_fnInfo(IAVIFile *iface, AVIFILEINFOW *afi, LONG size)
307 IAVIFileImpl *This = impl_from_IAVIFile(iface);
309 TRACE("(%p,%p,%d)\n",iface,afi,size);
311 if (afi == NULL)
312 return AVIERR_BADPARAM;
313 if (size < 0)
314 return AVIERR_BADSIZE;
316 AVIFILE_UpdateInfo(This);
318 memcpy(afi, &This->fInfo, min((DWORD)size, sizeof(This->fInfo)));
320 if ((DWORD)size < sizeof(This->fInfo))
321 return AVIERR_BUFFERTOOSMALL;
322 return AVIERR_OK;
325 static HRESULT WINAPI IAVIFile_fnGetStream(IAVIFile *iface, IAVIStream **avis, DWORD fccType,
326 LONG lParam)
328 IAVIFileImpl *This = impl_from_IAVIFile(iface);
329 ULONG nStream;
331 TRACE("(%p,%p,0x%08X,%d)\n", iface, avis, fccType, lParam);
333 if (avis == NULL || lParam < 0)
334 return AVIERR_BADPARAM;
336 nStream = AVIFILE_SearchStream(This, fccType, lParam);
338 /* Does the requested stream exist? */
339 if (nStream < This->fInfo.dwStreams &&
340 This->ppStreams[nStream] != NULL) {
341 *avis = (PAVISTREAM)This->ppStreams[nStream];
342 IAVIStream_AddRef(*avis);
344 return AVIERR_OK;
347 /* Sorry, but the specified stream doesn't exist */
348 return AVIERR_NODATA;
351 static HRESULT WINAPI IAVIFile_fnCreateStream(IAVIFile *iface, IAVIStream **avis,
352 AVISTREAMINFOW *asi)
354 IAVIFileImpl *This = impl_from_IAVIFile(iface);
355 DWORD n;
357 TRACE("(%p,%p,%p)\n", iface, avis, asi);
359 /* check parameters */
360 if (avis == NULL || asi == NULL)
361 return AVIERR_BADPARAM;
363 *avis = NULL;
365 /* Does the user have write permission? */
366 if ((This->uMode & MMIO_RWMODE) == 0)
367 return AVIERR_READONLY;
369 /* Can we add another stream? */
370 n = This->fInfo.dwStreams;
371 if (n >= MAX_AVISTREAMS || This->dwMoviChunkPos != 0) {
372 /* already reached max nr of streams
373 * or have already written frames to disk */
374 return AVIERR_UNSUPPORTED;
377 /* check AVISTREAMINFO for some really needed things */
378 if (asi->fccType == 0 || asi->dwScale == 0 || asi->dwRate == 0)
379 return AVIERR_BADFORMAT;
381 /* now it seems to be save to add the stream */
382 assert(This->ppStreams[n] == NULL);
383 This->ppStreams[n] = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
384 sizeof(IAVIStreamImpl));
385 if (This->ppStreams[n] == NULL)
386 return AVIERR_MEMORY;
388 /* initialize the new allocated stream */
389 AVIFILE_ConstructAVIStream(This, n, asi);
391 This->fInfo.dwStreams++;
392 This->fDirty = TRUE;
394 /* update our AVIFILEINFO structure */
395 AVIFILE_UpdateInfo(This);
397 /* return it */
398 *avis = (PAVISTREAM)This->ppStreams[n];
399 IAVIStream_AddRef(*avis);
401 return AVIERR_OK;
404 static HRESULT WINAPI IAVIFile_fnWriteData(IAVIFile *iface, DWORD ckid, void *lpData, LONG size)
406 IAVIFileImpl *This = impl_from_IAVIFile(iface);
408 TRACE("(%p,0x%08X,%p,%d)\n", iface, ckid, lpData, size);
410 /* check parameters */
411 if (lpData == NULL)
412 return AVIERR_BADPARAM;
413 if (size < 0)
414 return AVIERR_BADSIZE;
416 /* Do we have write permission? */
417 if ((This->uMode & MMIO_RWMODE) == 0)
418 return AVIERR_READONLY;
420 This->fDirty = TRUE;
422 return WriteExtraChunk(&This->fileextra, ckid, lpData, size);
425 static HRESULT WINAPI IAVIFile_fnReadData(IAVIFile *iface, DWORD ckid, void *lpData, LONG *size)
427 IAVIFileImpl *This = impl_from_IAVIFile(iface);
429 TRACE("(%p,0x%08X,%p,%p)\n", iface, ckid, lpData, size);
431 return ReadExtraChunk(&This->fileextra, ckid, lpData, size);
434 static HRESULT WINAPI IAVIFile_fnEndRecord(IAVIFile *iface)
436 IAVIFileImpl *This = impl_from_IAVIFile(iface);
438 TRACE("(%p)\n",iface);
440 if ((This->uMode & MMIO_RWMODE) == 0)
441 return AVIERR_READONLY;
443 This->fDirty = TRUE;
445 /* no frames written to any stream? -- compute start of 'movi'-chunk */
446 if (This->dwMoviChunkPos == 0)
447 AVIFILE_ComputeMoviStart(This);
449 This->fInfo.dwFlags |= AVIFILEINFO_ISINTERLEAVED;
451 /* already written frames to any stream, ... */
452 if (This->ckLastRecord.dwFlags & MMIO_DIRTY) {
453 /* close last record */
454 if (mmioAscend(This->hmmio, &This->ckLastRecord, 0) != 0)
455 return AVIERR_FILEWRITE;
457 AVIFILE_AddRecord(This);
459 if (This->fInfo.dwSuggestedBufferSize < This->ckLastRecord.cksize + 3 * sizeof(DWORD))
460 This->fInfo.dwSuggestedBufferSize = This->ckLastRecord.cksize + 3 * sizeof(DWORD);
463 /* write out a new record into file, but don't close it */
464 This->ckLastRecord.cksize = 0;
465 This->ckLastRecord.fccType = listtypeAVIRECORD;
466 if (mmioSeek(This->hmmio, This->dwNextFramePos, SEEK_SET) == -1)
467 return AVIERR_FILEWRITE;
468 if (mmioCreateChunk(This->hmmio, &This->ckLastRecord, MMIO_CREATELIST) != 0)
469 return AVIERR_FILEWRITE;
470 This->dwNextFramePos += 3 * sizeof(DWORD);
472 return AVIERR_OK;
475 static HRESULT WINAPI IAVIFile_fnDeleteStream(IAVIFile *iface, DWORD fccType, LONG lParam)
477 IAVIFileImpl *This = impl_from_IAVIFile(iface);
478 ULONG nStream;
480 TRACE("(%p,0x%08X,%d)\n", iface, fccType, lParam);
482 /* check parameter */
483 if (lParam < 0)
484 return AVIERR_BADPARAM;
486 /* Have user write permissions? */
487 if ((This->uMode & MMIO_RWMODE) == 0)
488 return AVIERR_READONLY;
490 nStream = AVIFILE_SearchStream(This, fccType, lParam);
492 /* Does the requested stream exist? */
493 if (nStream < This->fInfo.dwStreams &&
494 This->ppStreams[nStream] != NULL) {
495 /* ... so delete it now */
496 HeapFree(GetProcessHeap(), 0, This->ppStreams[nStream]);
498 if (This->fInfo.dwStreams - nStream > 0)
499 memcpy(This->ppStreams + nStream, This->ppStreams + nStream + 1,
500 (This->fInfo.dwStreams - nStream) * sizeof(IAVIStreamImpl*));
502 This->ppStreams[This->fInfo.dwStreams] = NULL;
503 This->fInfo.dwStreams--;
504 This->fDirty = TRUE;
506 /* This->fInfo will be updated further when asked for */
507 return AVIERR_OK;
508 } else
509 return AVIERR_NODATA;
512 static const struct IAVIFileVtbl avif_vt = {
513 IAVIFile_fnQueryInterface,
514 IAVIFile_fnAddRef,
515 IAVIFile_fnRelease,
516 IAVIFile_fnInfo,
517 IAVIFile_fnGetStream,
518 IAVIFile_fnCreateStream,
519 IAVIFile_fnWriteData,
520 IAVIFile_fnReadData,
521 IAVIFile_fnEndRecord,
522 IAVIFile_fnDeleteStream
526 static HRESULT WINAPI IPersistFile_fnQueryInterface(IPersistFile *iface, REFIID riid, void **ppv)
528 IAVIFileImpl *This = impl_from_IPersistFile(iface);
530 return IUnknown_QueryInterface(This->outer_unk, riid, ppv);
533 static ULONG WINAPI IPersistFile_fnAddRef(IPersistFile *iface)
535 IAVIFileImpl *This = impl_from_IPersistFile(iface);
537 return IUnknown_AddRef(This->outer_unk);
540 static ULONG WINAPI IPersistFile_fnRelease(IPersistFile *iface)
542 IAVIFileImpl *This = impl_from_IPersistFile(iface);
544 return IUnknown_Release(This->outer_unk);
547 static HRESULT WINAPI IPersistFile_fnGetClassID(IPersistFile *iface, LPCLSID pClassID)
549 TRACE("(%p,%p)\n", iface, pClassID);
551 if (pClassID == NULL)
552 return AVIERR_BADPARAM;
554 *pClassID = CLSID_AVIFile;
556 return AVIERR_OK;
559 static HRESULT WINAPI IPersistFile_fnIsDirty(IPersistFile *iface)
561 IAVIFileImpl *This = impl_from_IPersistFile(iface);
563 TRACE("(%p)\n", iface);
565 return (This->fDirty ? S_OK : S_FALSE);
568 static HRESULT WINAPI IPersistFile_fnLoad(IPersistFile *iface, LPCOLESTR pszFileName, DWORD dwMode)
570 IAVIFileImpl *This = impl_from_IPersistFile(iface);
571 int len;
573 TRACE("(%p,%s,0x%08X)\n", iface, debugstr_w(pszFileName), dwMode);
575 /* check parameter */
576 if (pszFileName == NULL)
577 return AVIERR_BADPARAM;
579 if (This->hmmio != NULL)
580 return AVIERR_ERROR; /* No reuse of this object for another file! */
582 /* remember mode and name */
583 This->uMode = dwMode;
585 len = lstrlenW(pszFileName) + 1;
586 This->szFileName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
587 if (This->szFileName == NULL)
588 return AVIERR_MEMORY;
589 lstrcpyW(This->szFileName, pszFileName);
591 /* try to open the file */
592 This->hmmio = mmioOpenW(This->szFileName, NULL, MMIO_ALLOCBUF | dwMode);
593 if (This->hmmio == NULL) {
594 /* mmioOpenW not in native DLLs of Win9x -- try mmioOpenA */
595 LPSTR szFileName;
597 len = WideCharToMultiByte(CP_ACP, 0, This->szFileName, -1, NULL, 0, NULL, NULL);
598 szFileName = HeapAlloc(GetProcessHeap(), 0, len * sizeof(CHAR));
599 if (szFileName == NULL)
600 return AVIERR_MEMORY;
602 WideCharToMultiByte(CP_ACP, 0, This->szFileName, -1, szFileName, len, NULL, NULL);
604 This->hmmio = mmioOpenA(szFileName, NULL, MMIO_ALLOCBUF | dwMode);
605 HeapFree(GetProcessHeap(), 0, szFileName);
606 if (This->hmmio == NULL)
607 return AVIERR_FILEOPEN;
610 /* should we create a new file? */
611 if (dwMode & OF_CREATE) {
612 memset(& This->fInfo, 0, sizeof(This->fInfo));
613 This->fInfo.dwFlags = AVIFILEINFO_HASINDEX | AVIFILEINFO_TRUSTCKTYPE;
615 return AVIERR_OK;
616 } else
617 return AVIFILE_LoadFile(This);
620 static HRESULT WINAPI IPersistFile_fnSave(IPersistFile *iface, LPCOLESTR pszFileName,
621 BOOL fRemember)
623 TRACE("(%p,%s,%d)\n", iface, debugstr_w(pszFileName), fRemember);
625 /* We write directly to disk, so nothing to do. */
627 return AVIERR_OK;
630 static HRESULT WINAPI IPersistFile_fnSaveCompleted(IPersistFile *iface, LPCOLESTR pszFileName)
632 TRACE("(%p,%s)\n", iface, debugstr_w(pszFileName));
634 /* We write directly to disk, so nothing to do. */
636 return AVIERR_OK;
639 static HRESULT WINAPI IPersistFile_fnGetCurFile(IPersistFile *iface, LPOLESTR *ppszFileName)
641 IAVIFileImpl *This = impl_from_IPersistFile(iface);
643 TRACE("(%p,%p)\n", iface, ppszFileName);
645 if (ppszFileName == NULL)
646 return AVIERR_BADPARAM;
648 *ppszFileName = NULL;
650 if (This->szFileName != NULL) {
651 int len = lstrlenW(This->szFileName) + 1;
653 *ppszFileName = CoTaskMemAlloc(len * sizeof(WCHAR));
654 if (*ppszFileName == NULL)
655 return AVIERR_MEMORY;
657 strcpyW(*ppszFileName, This->szFileName);
660 return AVIERR_OK;
663 static const struct IPersistFileVtbl pf_vt = {
664 IPersistFile_fnQueryInterface,
665 IPersistFile_fnAddRef,
666 IPersistFile_fnRelease,
667 IPersistFile_fnGetClassID,
668 IPersistFile_fnIsDirty,
669 IPersistFile_fnLoad,
670 IPersistFile_fnSave,
671 IPersistFile_fnSaveCompleted,
672 IPersistFile_fnGetCurFile
675 HRESULT AVIFILE_CreateAVIFile(IUnknown *pUnkOuter, REFIID riid, void **ppv)
677 IAVIFileImpl *obj;
678 HRESULT hr;
680 *ppv = NULL;
681 obj = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IAVIFileImpl));
682 if (!obj)
683 return AVIERR_MEMORY;
685 obj->IUnknown_inner.lpVtbl = &unk_vtbl;
686 obj->IAVIFile_iface.lpVtbl = &avif_vt;
687 obj->IPersistFile_iface.lpVtbl = &pf_vt;
688 obj->ref = 1;
689 if (pUnkOuter)
690 obj->outer_unk = pUnkOuter;
691 else
692 obj->outer_unk = &obj->IUnknown_inner;
694 hr = IUnknown_QueryInterface(&obj->IUnknown_inner, riid, ppv);
695 IUnknown_Release(&obj->IUnknown_inner);
697 return hr;
701 static HRESULT WINAPI IAVIStream_fnQueryInterface(IAVIStream *iface,
702 REFIID refiid, LPVOID *obj)
704 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
706 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(refiid), obj);
708 if (IsEqualGUID(&IID_IUnknown, refiid) ||
709 IsEqualGUID(&IID_IAVIStream, refiid)) {
710 *obj = This;
711 IAVIStream_AddRef(iface);
713 return S_OK;
715 /* FIXME: IAVIStreaming interface */
717 return OLE_E_ENUM_NOMORE;
720 static ULONG WINAPI IAVIStream_fnAddRef(IAVIStream *iface)
722 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
723 ULONG ref = InterlockedIncrement(&This->ref);
725 TRACE("(%p) -> %d\n", iface, ref);
727 /* also add ref to parent, so that it doesn't kill us */
728 if (This->paf != NULL)
729 IAVIFile_AddRef(&This->paf->IAVIFile_iface);
731 return ref;
734 static ULONG WINAPI IAVIStream_fnRelease(IAVIStream* iface)
736 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
737 ULONG ref = InterlockedDecrement(&This->ref);
739 TRACE("(%p) -> %d\n", iface, ref);
741 if (This->paf != NULL)
742 IAVIFile_Release(&This->paf->IAVIFile_iface);
744 return ref;
747 static HRESULT WINAPI IAVIStream_fnCreate(IAVIStream *iface, LPARAM lParam1,
748 LPARAM lParam2)
750 TRACE("(%p,0x%08lX,0x%08lX)\n", iface, lParam1, lParam2);
752 /* This IAVIStream interface needs an AVIFile */
753 return AVIERR_UNSUPPORTED;
756 static HRESULT WINAPI IAVIStream_fnInfo(IAVIStream *iface,LPAVISTREAMINFOW psi,
757 LONG size)
759 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
761 TRACE("(%p,%p,%d)\n", iface, psi, size);
763 if (psi == NULL)
764 return AVIERR_BADPARAM;
765 if (size < 0)
766 return AVIERR_BADSIZE;
768 memcpy(psi, &This->sInfo, min((DWORD)size, sizeof(This->sInfo)));
770 if ((DWORD)size < sizeof(This->sInfo))
771 return AVIERR_BUFFERTOOSMALL;
772 return AVIERR_OK;
775 static LONG WINAPI IAVIStream_fnFindSample(IAVIStream *iface, LONG pos,
776 LONG flags)
778 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
780 LONG offset = 0;
782 TRACE("(%p,%d,0x%08X)\n",iface,pos,flags);
784 if (flags & FIND_FROM_START) {
785 pos = This->sInfo.dwStart;
786 flags &= ~(FIND_FROM_START|FIND_PREV);
787 flags |= FIND_NEXT;
790 if (This->sInfo.dwSampleSize != 0) {
791 /* convert samples into block number with offset */
792 AVIFILE_SamplesToBlock(This, &pos, &offset);
795 if (flags & FIND_TYPE) {
796 if (flags & FIND_KEY) {
797 while (0 <= pos && pos <= This->lLastFrame) {
798 if (This->idxFrames[pos].dwFlags & AVIIF_KEYFRAME)
799 goto RETURN_FOUND;
801 if (flags & FIND_NEXT)
802 pos++;
803 else
804 pos--;
806 } else if (flags & FIND_ANY) {
807 while (0 <= pos && pos <= This->lLastFrame) {
808 if (This->idxFrames[pos].dwChunkLength > 0)
809 goto RETURN_FOUND;
811 if (flags & FIND_NEXT)
812 pos++;
813 else
814 pos--;
817 } else if ((flags & FIND_FORMAT) && This->idxFmtChanges != NULL &&
818 This->sInfo.fccType == streamtypeVIDEO) {
819 if (flags & FIND_NEXT) {
820 ULONG n;
822 for (n = 0; n < This->sInfo.dwFormatChangeCount; n++)
823 if (This->idxFmtChanges[n].ckid >= pos) {
824 pos = This->idxFmtChanges[n].ckid;
825 goto RETURN_FOUND;
827 } else {
828 LONG n;
830 for (n = (LONG)This->sInfo.dwFormatChangeCount; n >= 0; n--) {
831 if (This->idxFmtChanges[n].ckid <= pos) {
832 pos = This->idxFmtChanges[n].ckid;
833 goto RETURN_FOUND;
837 if (pos > (LONG)This->sInfo.dwStart)
838 return 0; /* format changes always for first frame */
842 return -1;
845 RETURN_FOUND:
846 if (pos < (LONG)This->sInfo.dwStart)
847 return -1;
849 switch (flags & FIND_RET) {
850 case FIND_LENGTH:
851 /* physical size */
852 pos = This->idxFrames[pos].dwChunkLength;
853 break;
854 case FIND_OFFSET:
855 /* physical position */
856 pos = This->idxFrames[pos].dwChunkOffset + 2 * sizeof(DWORD)
857 + offset * This->sInfo.dwSampleSize;
858 break;
859 case FIND_SIZE:
860 /* logical size */
861 if (This->sInfo.dwSampleSize)
862 pos = This->sInfo.dwSampleSize;
863 else
864 pos = 1;
865 break;
866 case FIND_INDEX:
867 FIXME(": FIND_INDEX flag is not supported!\n");
868 /* This is an index in the index-table on disc. */
869 break;
870 }; /* else logical position */
872 return pos;
875 static HRESULT WINAPI IAVIStream_fnReadFormat(IAVIStream *iface, LONG pos,
876 LPVOID format, LONG *formatsize)
878 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
880 TRACE("(%p,%d,%p,%p)\n", iface, pos, format, formatsize);
882 if (formatsize == NULL)
883 return AVIERR_BADPARAM;
885 /* only interested in needed buffersize? */
886 if (format == NULL || *formatsize <= 0) {
887 *formatsize = This->cbFormat;
889 return AVIERR_OK;
892 /* copy initial format (only as much as will fit) */
893 memcpy(format, This->lpFormat, min(*(DWORD*)formatsize, This->cbFormat));
894 if (*(DWORD*)formatsize < This->cbFormat) {
895 *formatsize = This->cbFormat;
896 return AVIERR_BUFFERTOOSMALL;
899 /* Could format change? When yes will it change? */
900 if ((This->sInfo.dwFlags & AVISTREAMINFO_FORMATCHANGES) &&
901 pos > This->sInfo.dwStart) {
902 LONG lLastFmt;
904 lLastFmt = IAVIStream_fnFindSample(iface, pos, FIND_FORMAT|FIND_PREV);
905 if (lLastFmt > 0) {
906 FIXME(": need to read formatchange for %d -- unimplemented!\n",lLastFmt);
910 *formatsize = This->cbFormat;
911 return AVIERR_OK;
914 static HRESULT WINAPI IAVIStream_fnSetFormat(IAVIStream *iface, LONG pos,
915 LPVOID format, LONG formatsize)
917 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
919 LPBITMAPINFOHEADER lpbiNew = format;
921 TRACE("(%p,%d,%p,%d)\n", iface, pos, format, formatsize);
923 /* check parameters */
924 if (format == NULL || formatsize <= 0)
925 return AVIERR_BADPARAM;
927 /* Do we have write permission? */
928 if ((This->paf->uMode & MMIO_RWMODE) == 0)
929 return AVIERR_READONLY;
931 /* can only set format before frame is written! */
932 if (This->lLastFrame > pos)
933 return AVIERR_UNSUPPORTED;
935 /* initial format or a formatchange? */
936 if (This->lpFormat == NULL) {
937 /* initial format */
938 if (This->paf->dwMoviChunkPos != 0)
939 return AVIERR_ERROR; /* user has used API in wrong sequence! */
941 This->lpFormat = HeapAlloc(GetProcessHeap(), 0, formatsize);
942 if (This->lpFormat == NULL)
943 return AVIERR_MEMORY;
944 This->cbFormat = formatsize;
946 memcpy(This->lpFormat, format, formatsize);
948 /* update some infos about stream */
949 if (This->sInfo.fccType == streamtypeVIDEO) {
950 LONG lDim;
952 lDim = This->sInfo.rcFrame.right - This->sInfo.rcFrame.left;
953 if (lDim < lpbiNew->biWidth)
954 This->sInfo.rcFrame.right = This->sInfo.rcFrame.left + lpbiNew->biWidth;
955 lDim = This->sInfo.rcFrame.bottom - This->sInfo.rcFrame.top;
956 if (lDim < lpbiNew->biHeight)
957 This->sInfo.rcFrame.bottom = This->sInfo.rcFrame.top + lpbiNew->biHeight;
958 } else if (This->sInfo.fccType == streamtypeAUDIO)
959 This->sInfo.dwSampleSize = ((LPWAVEFORMATEX)This->lpFormat)->nBlockAlign;
961 return AVIERR_OK;
962 } else {
963 MMCKINFO ck;
964 LPBITMAPINFOHEADER lpbiOld = This->lpFormat;
965 RGBQUAD *rgbNew = (RGBQUAD*)((LPBYTE)lpbiNew + lpbiNew->biSize);
966 AVIPALCHANGE *lppc = NULL;
967 UINT n;
969 /* perhaps format change, check it ... */
970 if (This->cbFormat != formatsize)
971 return AVIERR_UNSUPPORTED;
973 /* no format change, only the initial one */
974 if (memcmp(This->lpFormat, format, formatsize) == 0)
975 return AVIERR_OK;
977 /* check that's only the palette, which changes */
978 if (lpbiOld->biSize != lpbiNew->biSize ||
979 lpbiOld->biWidth != lpbiNew->biWidth ||
980 lpbiOld->biHeight != lpbiNew->biHeight ||
981 lpbiOld->biPlanes != lpbiNew->biPlanes ||
982 lpbiOld->biBitCount != lpbiNew->biBitCount ||
983 lpbiOld->biCompression != lpbiNew->biCompression ||
984 lpbiOld->biClrUsed != lpbiNew->biClrUsed)
985 return AVIERR_UNSUPPORTED;
987 This->sInfo.dwFlags |= AVISTREAMINFO_FORMATCHANGES;
989 /* simply say all colors have changed */
990 ck.ckid = MAKEAVICKID(cktypePALchange, This->nStream);
991 ck.cksize = 2 * sizeof(WORD) + lpbiOld->biClrUsed * sizeof(PALETTEENTRY);
992 lppc = HeapAlloc(GetProcessHeap(), 0, ck.cksize);
993 if (lppc == NULL)
994 return AVIERR_MEMORY;
996 lppc->bFirstEntry = 0;
997 lppc->bNumEntries = (lpbiOld->biClrUsed < 256 ? lpbiOld->biClrUsed : 0);
998 lppc->wFlags = 0;
999 for (n = 0; n < lpbiOld->biClrUsed; n++) {
1000 lppc->peNew[n].peRed = rgbNew[n].rgbRed;
1001 lppc->peNew[n].peGreen = rgbNew[n].rgbGreen;
1002 lppc->peNew[n].peBlue = rgbNew[n].rgbBlue;
1003 lppc->peNew[n].peFlags = 0;
1006 if (mmioSeek(This->paf->hmmio, This->paf->dwNextFramePos, SEEK_SET) == -1 ||
1007 mmioCreateChunk(This->paf->hmmio, &ck, 0) != S_OK ||
1008 mmioWrite(This->paf->hmmio, (HPSTR)lppc, ck.cksize) != ck.cksize ||
1009 mmioAscend(This->paf->hmmio, &ck, 0) != S_OK)
1011 HeapFree(GetProcessHeap(), 0, lppc);
1012 return AVIERR_FILEWRITE;
1015 This->paf->dwNextFramePos += ck.cksize + 2 * sizeof(DWORD);
1017 HeapFree(GetProcessHeap(), 0, lppc);
1019 return AVIFILE_AddFrame(This, cktypePALchange, n, ck.dwDataOffset, 0);
1023 static HRESULT WINAPI IAVIStream_fnRead(IAVIStream *iface, LONG start,
1024 LONG samples, LPVOID buffer,
1025 LONG buffersize, LPLONG bytesread,
1026 LPLONG samplesread)
1028 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
1030 DWORD size;
1031 HRESULT hr;
1033 TRACE("(%p,%d,%d,%p,%d,%p,%p)\n", iface, start, samples, buffer,
1034 buffersize, bytesread, samplesread);
1036 /* clear return parameters if given */
1037 if (bytesread != NULL)
1038 *bytesread = 0;
1039 if (samplesread != NULL)
1040 *samplesread = 0;
1042 /* check parameters */
1043 if ((LONG)This->sInfo.dwStart > start)
1044 return AVIERR_NODATA; /* couldn't read before start of stream */
1045 if (This->sInfo.dwStart + This->sInfo.dwLength < (DWORD)start)
1046 return AVIERR_NODATA; /* start is past end of stream */
1048 /* should we read as much as possible? */
1049 if (samples == -1) {
1050 /* User should know how much we have read */
1051 if (bytesread == NULL && samplesread == NULL)
1052 return AVIERR_BADPARAM;
1054 if (This->sInfo.dwSampleSize != 0)
1055 samples = buffersize / This->sInfo.dwSampleSize;
1056 else
1057 samples = 1;
1060 /* limit to end of stream */
1061 if ((LONG)This->sInfo.dwLength < samples)
1062 samples = This->sInfo.dwLength;
1063 if ((start - This->sInfo.dwStart) > (This->sInfo.dwLength - samples))
1064 samples = This->sInfo.dwLength - (start - This->sInfo.dwStart);
1066 /* nothing to read? Then leave ... */
1067 if (samples == 0)
1068 return AVIERR_OK;
1070 if (This->sInfo.dwSampleSize != 0) {
1071 /* fixed samplesize -- we can read over frame/block boundaries */
1072 LONG block = start;
1073 LONG offset = 0;
1075 if (!buffer)
1077 if (bytesread)
1078 *bytesread = samples*This->sInfo.dwSampleSize;
1079 if (samplesread)
1080 *samplesread = samples;
1081 return AVIERR_OK;
1084 /* convert start sample to block,offset pair */
1085 AVIFILE_SamplesToBlock(This, &block, &offset);
1087 /* convert samples to bytes */
1088 samples *= This->sInfo.dwSampleSize;
1090 while (samples > 0 && buffersize > 0) {
1091 LONG blocksize;
1092 if (block != This->dwCurrentFrame) {
1093 hr = AVIFILE_ReadBlock(This, block, NULL, 0);
1094 if (FAILED(hr))
1095 return hr;
1098 size = min((DWORD)samples, (DWORD)buffersize);
1099 blocksize = This->lpBuffer[1];
1100 TRACE("blocksize = %u\n",blocksize);
1101 size = min(size, blocksize - offset);
1102 memcpy(buffer, ((BYTE*)&This->lpBuffer[2]) + offset, size);
1104 block++;
1105 offset = 0;
1106 buffer = ((LPBYTE)buffer)+size;
1107 samples -= size;
1108 buffersize -= size;
1110 /* fill out return parameters if given */
1111 if (bytesread != NULL)
1112 *bytesread += size;
1113 if (samplesread != NULL)
1114 *samplesread += size / This->sInfo.dwSampleSize;
1117 if (samples == 0)
1118 return AVIERR_OK;
1119 else
1120 return AVIERR_BUFFERTOOSMALL;
1121 } else {
1122 /* variable samplesize -- we can only read one full frame/block */
1123 if (samples > 1)
1124 samples = 1;
1126 assert(start <= This->lLastFrame);
1127 size = This->idxFrames[start].dwChunkLength;
1128 if (buffer != NULL && buffersize >= size) {
1129 hr = AVIFILE_ReadBlock(This, start, buffer, size);
1130 if (FAILED(hr))
1131 return hr;
1132 } else if (buffer != NULL)
1133 return AVIERR_BUFFERTOOSMALL;
1135 /* fill out return parameters if given */
1136 if (bytesread != NULL)
1137 *bytesread = size;
1138 if (samplesread != NULL)
1139 *samplesread = samples;
1141 return AVIERR_OK;
1145 static HRESULT WINAPI IAVIStream_fnWrite(IAVIStream *iface, LONG start,
1146 LONG samples, LPVOID buffer,
1147 LONG buffersize, DWORD flags,
1148 LPLONG sampwritten,
1149 LPLONG byteswritten)
1151 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
1153 FOURCC ckid;
1154 HRESULT hr;
1156 TRACE("(%p,%d,%d,%p,%d,0x%08X,%p,%p)\n", iface, start, samples,
1157 buffer, buffersize, flags, sampwritten, byteswritten);
1159 /* clear return parameters if given */
1160 if (sampwritten != NULL)
1161 *sampwritten = 0;
1162 if (byteswritten != NULL)
1163 *byteswritten = 0;
1165 /* check parameters */
1166 if (buffer == NULL && (buffersize > 0 || samples > 0))
1167 return AVIERR_BADPARAM;
1169 /* Have we write permission? */
1170 if ((This->paf->uMode & MMIO_RWMODE) == 0)
1171 return AVIERR_READONLY;
1173 switch (This->sInfo.fccType) {
1174 case streamtypeAUDIO:
1175 ckid = MAKEAVICKID(cktypeWAVEbytes, This->nStream);
1176 break;
1177 default:
1178 if ((flags & AVIIF_KEYFRAME) && buffersize != 0)
1179 ckid = MAKEAVICKID(cktypeDIBbits, This->nStream);
1180 else
1181 ckid = MAKEAVICKID(cktypeDIBcompressed, This->nStream);
1182 break;
1185 /* append to end of stream? */
1186 if (start == -1) {
1187 if (This->lLastFrame == -1)
1188 start = This->sInfo.dwStart;
1189 else
1190 start = This->sInfo.dwLength;
1191 } else if (This->lLastFrame == -1)
1192 This->sInfo.dwStart = start;
1194 if (This->sInfo.dwSampleSize != 0) {
1195 /* fixed sample size -- audio like */
1196 if (samples * This->sInfo.dwSampleSize != buffersize)
1197 return AVIERR_BADPARAM;
1199 /* Couldn't skip audio-like data -- User must supply appropriate silence */
1200 if (This->sInfo.dwLength != start)
1201 return AVIERR_UNSUPPORTED;
1203 /* Convert position to frame/block */
1204 start = This->lLastFrame + 1;
1206 if ((This->paf->fInfo.dwFlags & AVIFILEINFO_ISINTERLEAVED) == 0) {
1207 FIXME(": not interleaved, could collect audio data!\n");
1209 } else {
1210 /* variable sample size -- video like */
1211 if (samples > 1)
1212 return AVIERR_UNSUPPORTED;
1214 /* must we fill up with empty frames? */
1215 if (This->lLastFrame != -1) {
1216 FOURCC ckid2 = MAKEAVICKID(cktypeDIBcompressed, This->nStream);
1218 while (start > This->lLastFrame + 1) {
1219 hr = AVIFILE_WriteBlock(This, This->lLastFrame + 1, ckid2, 0, NULL, 0);
1220 if (FAILED(hr))
1221 return hr;
1226 /* write the block now */
1227 hr = AVIFILE_WriteBlock(This, start, ckid, flags, buffer, buffersize);
1228 if (SUCCEEDED(hr)) {
1229 /* fill out return parameters if given */
1230 if (sampwritten != NULL)
1231 *sampwritten = samples;
1232 if (byteswritten != NULL)
1233 *byteswritten = buffersize;
1236 return hr;
1239 static HRESULT WINAPI IAVIStream_fnDelete(IAVIStream *iface, LONG start,
1240 LONG samples)
1242 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
1244 FIXME("(%p,%d,%d): stub\n", iface, start, samples);
1246 /* check parameters */
1247 if (start < 0 || samples < 0)
1248 return AVIERR_BADPARAM;
1250 /* Delete before start of stream? */
1251 if (start + samples < This->sInfo.dwStart)
1252 return AVIERR_OK;
1254 /* Delete after end of stream? */
1255 if (start > This->sInfo.dwLength)
1256 return AVIERR_OK;
1258 /* For the rest we need write permissions */
1259 if ((This->paf->uMode & MMIO_RWMODE) == 0)
1260 return AVIERR_READONLY;
1262 /* 1. overwrite the data with JUNK
1264 * if ISINTERLEAVED {
1265 * 2. concat all neighboured JUNK-blocks in this record to one
1266 * 3. if this record only contains JUNK and is at end set dwNextFramePos
1267 * to start of this record, repeat this.
1268 * } else {
1269 * 2. concat all neighboured JUNK-blocks.
1270 * 3. if the JUNK block is at the end, then set dwNextFramePos to
1271 * start of this block.
1275 return AVIERR_UNSUPPORTED;
1278 static HRESULT WINAPI IAVIStream_fnReadData(IAVIStream *iface, DWORD fcc,
1279 LPVOID lp, LPLONG lpread)
1281 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
1283 TRACE("(%p,0x%08X,%p,%p)\n", iface, fcc, lp, lpread);
1285 if (fcc == ckidSTREAMHANDLERDATA) {
1286 if (This->lpHandlerData != NULL && This->cbHandlerData > 0) {
1287 if (lp == NULL || *lpread <= 0) {
1288 *lpread = This->cbHandlerData;
1289 return AVIERR_OK;
1292 memcpy(lp, This->lpHandlerData, min(This->cbHandlerData, *lpread));
1293 if (*lpread < This->cbHandlerData)
1294 return AVIERR_BUFFERTOOSMALL;
1295 return AVIERR_OK;
1296 } else
1297 return AVIERR_NODATA;
1298 } else
1299 return ReadExtraChunk(&This->extra, fcc, lp, lpread);
1302 static HRESULT WINAPI IAVIStream_fnWriteData(IAVIStream *iface, DWORD fcc,
1303 LPVOID lp, LONG size)
1305 IAVIStreamImpl *This = (IAVIStreamImpl *)iface;
1307 TRACE("(%p,0x%08x,%p,%d)\n", iface, fcc, lp, size);
1309 /* check parameters */
1310 if (lp == NULL)
1311 return AVIERR_BADPARAM;
1312 if (size <= 0)
1313 return AVIERR_BADSIZE;
1315 /* need write permission */
1316 if ((This->paf->uMode & MMIO_RWMODE) == 0)
1317 return AVIERR_READONLY;
1319 /* already written something to this file? */
1320 if (This->paf->dwMoviChunkPos != 0) {
1321 /* the data will be inserted before the 'movi' chunk, so check for
1322 * enough space */
1323 DWORD dwPos = AVIFILE_ComputeMoviStart(This->paf);
1325 /* ckid,size => 2 * sizeof(DWORD) */
1326 dwPos += 2 * sizeof(DWORD) + size;
1327 if (dwPos >= This->paf->dwMoviChunkPos - 2 * sizeof(DWORD))
1328 return AVIERR_UNSUPPORTED; /* not enough space left */
1331 This->paf->fDirty = TRUE;
1333 if (fcc == ckidSTREAMHANDLERDATA) {
1334 if (This->lpHandlerData != NULL) {
1335 FIXME(": handler data already set -- overwirte?\n");
1336 return AVIERR_UNSUPPORTED;
1339 This->lpHandlerData = HeapAlloc(GetProcessHeap(), 0, size);
1340 if (This->lpHandlerData == NULL)
1341 return AVIERR_MEMORY;
1342 This->cbHandlerData = size;
1343 memcpy(This->lpHandlerData, lp, size);
1345 return AVIERR_OK;
1346 } else
1347 return WriteExtraChunk(&This->extra, fcc, lp, size);
1350 static HRESULT WINAPI IAVIStream_fnSetInfo(IAVIStream *iface,
1351 LPAVISTREAMINFOW info, LONG infolen)
1353 FIXME("(%p,%p,%d): stub\n", iface, info, infolen);
1355 return E_FAIL;
1358 /***********************************************************************/
1360 static HRESULT AVIFILE_AddFrame(IAVIStreamImpl *This, DWORD ckid, DWORD size, DWORD offset, DWORD flags)
1362 UINT n;
1364 /* pre-conditions */
1365 assert(This != NULL);
1367 switch (TWOCCFromFOURCC(ckid)) {
1368 case cktypeDIBbits:
1369 if (This->paf->fInfo.dwFlags & AVIFILEINFO_TRUSTCKTYPE)
1370 flags |= AVIIF_KEYFRAME;
1371 break;
1372 case cktypeDIBcompressed:
1373 if (This->paf->fInfo.dwFlags & AVIFILEINFO_TRUSTCKTYPE)
1374 flags &= ~AVIIF_KEYFRAME;
1375 break;
1376 case cktypePALchange:
1377 if (This->sInfo.fccType != streamtypeVIDEO) {
1378 ERR(": found palette change in non-video stream!\n");
1379 return AVIERR_BADFORMAT;
1382 if (This->idxFmtChanges == NULL || This->nIdxFmtChanges <= This->sInfo.dwFormatChangeCount) {
1383 DWORD new_count = This->nIdxFmtChanges + 16;
1384 void *new_buffer;
1386 if (This->idxFmtChanges == NULL) {
1387 This->idxFmtChanges =
1388 HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, new_count * sizeof(AVIINDEXENTRY));
1389 if (!This->idxFmtChanges) return AVIERR_MEMORY;
1390 } else {
1391 new_buffer = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, This->idxFmtChanges,
1392 new_count * sizeof(AVIINDEXENTRY));
1393 if (!new_buffer) return AVIERR_MEMORY;
1394 This->idxFmtChanges = new_buffer;
1396 This->nIdxFmtChanges = new_count;
1399 This->sInfo.dwFlags |= AVISTREAMINFO_FORMATCHANGES;
1400 n = ++This->sInfo.dwFormatChangeCount;
1401 This->idxFmtChanges[n].ckid = This->lLastFrame;
1402 This->idxFmtChanges[n].dwFlags = 0;
1403 This->idxFmtChanges[n].dwChunkOffset = offset;
1404 This->idxFmtChanges[n].dwChunkLength = size;
1406 return AVIERR_OK;
1407 case cktypeWAVEbytes:
1408 if (This->paf->fInfo.dwFlags & AVIFILEINFO_TRUSTCKTYPE)
1409 flags |= AVIIF_KEYFRAME;
1410 break;
1411 default:
1412 WARN(": unknown TWOCC 0x%04X found\n", TWOCCFromFOURCC(ckid));
1413 break;
1416 /* first frame is always a keyframe */
1417 if (This->lLastFrame == -1)
1418 flags |= AVIIF_KEYFRAME;
1420 if (This->sInfo.dwSuggestedBufferSize < size)
1421 This->sInfo.dwSuggestedBufferSize = size;
1423 /* get memory for index */
1424 if (This->idxFrames == NULL || This->lLastFrame + 1 >= This->nIdxFrames) {
1425 This->nIdxFrames += 512;
1426 if (This->idxFrames == NULL)
1427 This->idxFrames = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, This->nIdxFrames * sizeof(AVIINDEXENTRY));
1428 else
1429 This->idxFrames = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, This->idxFrames,
1430 This->nIdxFrames * sizeof(AVIINDEXENTRY));
1431 if (This->idxFrames == NULL)
1432 return AVIERR_MEMORY;
1435 This->lLastFrame++;
1436 This->idxFrames[This->lLastFrame].ckid = ckid;
1437 This->idxFrames[This->lLastFrame].dwFlags = flags;
1438 This->idxFrames[This->lLastFrame].dwChunkOffset = offset;
1439 This->idxFrames[This->lLastFrame].dwChunkLength = size;
1441 /* update AVISTREAMINFO structure if necessary */
1442 if (This->sInfo.dwLength <= This->lLastFrame)
1443 This->sInfo.dwLength = This->lLastFrame + 1;
1445 return AVIERR_OK;
1448 static HRESULT AVIFILE_AddRecord(IAVIFileImpl *This)
1450 /* pre-conditions */
1451 assert(This != NULL && This->ppStreams[0] != NULL);
1453 if (This->idxRecords == NULL || This->cbIdxRecords == 0) {
1454 This->cbIdxRecords += 1024 * sizeof(AVIINDEXENTRY);
1455 This->idxRecords = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, This->cbIdxRecords);
1456 if (This->idxRecords == NULL)
1457 return AVIERR_MEMORY;
1460 assert(This->nIdxRecords < This->cbIdxRecords/sizeof(AVIINDEXENTRY));
1462 This->idxRecords[This->nIdxRecords].ckid = listtypeAVIRECORD;
1463 This->idxRecords[This->nIdxRecords].dwFlags = AVIIF_LIST;
1464 This->idxRecords[This->nIdxRecords].dwChunkOffset =
1465 This->ckLastRecord.dwDataOffset - 2 * sizeof(DWORD);
1466 This->idxRecords[This->nIdxRecords].dwChunkLength =
1467 This->ckLastRecord.cksize;
1468 This->nIdxRecords++;
1470 return AVIERR_OK;
1473 static DWORD AVIFILE_ComputeMoviStart(IAVIFileImpl *This)
1475 DWORD dwPos;
1476 DWORD nStream;
1478 /* RIFF,hdrl,movi,avih => (3 * 3 + 2) * sizeof(DWORD) = 11 * sizeof(DWORD) */
1479 dwPos = 11 * sizeof(DWORD) + sizeof(MainAVIHeader);
1481 for (nStream = 0; nStream < This->fInfo.dwStreams; nStream++) {
1482 IAVIStreamImpl *pStream = This->ppStreams[nStream];
1484 /* strl,strh,strf => (3 + 2 * 2) * sizeof(DWORD) = 7 * sizeof(DWORD) */
1485 dwPos += 7 * sizeof(DWORD) + sizeof(AVIStreamHeader);
1486 dwPos += ((pStream->cbFormat + 1) & ~1U);
1487 if (pStream->lpHandlerData != NULL && pStream->cbHandlerData > 0)
1488 dwPos += 2 * sizeof(DWORD) + ((pStream->cbHandlerData + 1) & ~1U);
1489 if (lstrlenW(pStream->sInfo.szName) > 0)
1490 dwPos += 2 * sizeof(DWORD) + ((lstrlenW(pStream->sInfo.szName) + 1) & ~1U);
1493 if (This->dwMoviChunkPos == 0) {
1494 This->dwNextFramePos = dwPos;
1496 /* pad to multiple of AVI_HEADERSIZE only if we are more than 8 bytes away from it */
1497 if (((dwPos + AVI_HEADERSIZE) & ~(AVI_HEADERSIZE - 1)) - dwPos > 2 * sizeof(DWORD))
1498 This->dwNextFramePos = (dwPos + AVI_HEADERSIZE) & ~(AVI_HEADERSIZE - 1);
1500 This->dwMoviChunkPos = This->dwNextFramePos - sizeof(DWORD);
1503 return dwPos;
1506 static void AVIFILE_ConstructAVIStream(IAVIFileImpl *paf, DWORD nr, const AVISTREAMINFOW *asi)
1508 IAVIStreamImpl *pstream;
1510 /* pre-conditions */
1511 assert(paf != NULL);
1512 assert(nr < MAX_AVISTREAMS);
1513 assert(paf->ppStreams[nr] != NULL);
1515 pstream = paf->ppStreams[nr];
1517 pstream->lpVtbl = &iavist;
1518 pstream->ref = 0;
1519 pstream->paf = paf;
1520 pstream->nStream = nr;
1521 pstream->dwCurrentFrame = (DWORD)-1;
1522 pstream->lLastFrame = -1;
1524 if (asi != NULL) {
1525 memcpy(&pstream->sInfo, asi, sizeof(pstream->sInfo));
1527 if (asi->dwLength > 0) {
1528 /* pre-allocate mem for frame-index structure */
1529 pstream->idxFrames =
1530 HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, asi->dwLength * sizeof(AVIINDEXENTRY));
1531 if (pstream->idxFrames != NULL)
1532 pstream->nIdxFrames = asi->dwLength;
1534 if (asi->dwFormatChangeCount > 0) {
1535 /* pre-allocate mem for formatchange-index structure */
1536 pstream->idxFmtChanges =
1537 HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, asi->dwFormatChangeCount * sizeof(AVIINDEXENTRY));
1538 if (pstream->idxFmtChanges != NULL)
1539 pstream->nIdxFmtChanges = asi->dwFormatChangeCount;
1542 /* These values will be computed */
1543 pstream->sInfo.dwLength = 0;
1544 pstream->sInfo.dwSuggestedBufferSize = 0;
1545 pstream->sInfo.dwFormatChangeCount = 0;
1546 pstream->sInfo.dwEditCount = 1;
1547 if (pstream->sInfo.dwSampleSize > 0)
1548 SetRectEmpty(&pstream->sInfo.rcFrame);
1551 pstream->sInfo.dwCaps = AVIFILECAPS_CANREAD|AVIFILECAPS_CANWRITE;
1554 static void AVIFILE_DestructAVIStream(IAVIStreamImpl *This)
1556 /* pre-conditions */
1557 assert(This != NULL);
1559 This->dwCurrentFrame = (DWORD)-1;
1560 This->lLastFrame = -1;
1561 This->paf = NULL;
1562 if (This->idxFrames != NULL) {
1563 HeapFree(GetProcessHeap(), 0, This->idxFrames);
1564 This->idxFrames = NULL;
1565 This->nIdxFrames = 0;
1567 HeapFree(GetProcessHeap(), 0, This->idxFmtChanges);
1568 This->idxFmtChanges = NULL;
1569 if (This->lpBuffer != NULL) {
1570 HeapFree(GetProcessHeap(), 0, This->lpBuffer);
1571 This->lpBuffer = NULL;
1572 This->cbBuffer = 0;
1574 if (This->lpHandlerData != NULL) {
1575 HeapFree(GetProcessHeap(), 0, This->lpHandlerData);
1576 This->lpHandlerData = NULL;
1577 This->cbHandlerData = 0;
1579 if (This->extra.lp != NULL) {
1580 HeapFree(GetProcessHeap(), 0, This->extra.lp);
1581 This->extra.lp = NULL;
1582 This->extra.cb = 0;
1584 if (This->lpFormat != NULL) {
1585 HeapFree(GetProcessHeap(), 0, This->lpFormat);
1586 This->lpFormat = NULL;
1587 This->cbFormat = 0;
1591 static HRESULT AVIFILE_LoadFile(IAVIFileImpl *This)
1593 MainAVIHeader MainAVIHdr;
1594 MMCKINFO ckRIFF;
1595 MMCKINFO ckLIST1;
1596 MMCKINFO ckLIST2;
1597 MMCKINFO ck;
1598 IAVIStreamImpl *pStream;
1599 DWORD nStream;
1600 HRESULT hr;
1602 if (This->hmmio == NULL)
1603 return AVIERR_FILEOPEN;
1605 /* initialize stream ptr's */
1606 memset(This->ppStreams, 0, sizeof(This->ppStreams));
1608 /* try to get "RIFF" chunk -- must not be at beginning of file! */
1609 ckRIFF.fccType = formtypeAVI;
1610 if (mmioDescend(This->hmmio, &ckRIFF, NULL, MMIO_FINDRIFF) != S_OK) {
1611 ERR(": not an AVI!\n");
1612 return AVIERR_FILEREAD;
1615 /* get "LIST" "hdrl" */
1616 ckLIST1.fccType = listtypeAVIHEADER;
1617 hr = FindChunkAndKeepExtras(&This->fileextra, This->hmmio, &ckLIST1, &ckRIFF, MMIO_FINDLIST);
1618 if (FAILED(hr))
1619 return hr;
1621 /* get "avih" chunk */
1622 ck.ckid = ckidAVIMAINHDR;
1623 hr = FindChunkAndKeepExtras(&This->fileextra, This->hmmio, &ck, &ckLIST1, MMIO_FINDCHUNK);
1624 if (FAILED(hr))
1625 return hr;
1627 if (ck.cksize != sizeof(MainAVIHdr)) {
1628 ERR(": invalid size of %d for MainAVIHeader!\n", ck.cksize);
1629 return AVIERR_BADFORMAT;
1631 if (mmioRead(This->hmmio, (HPSTR)&MainAVIHdr, ck.cksize) != ck.cksize)
1632 return AVIERR_FILEREAD;
1634 /* check for MAX_AVISTREAMS limit */
1635 if (MainAVIHdr.dwStreams > MAX_AVISTREAMS) {
1636 WARN("file contains %u streams, but only supports %d -- change MAX_AVISTREAMS!\n", MainAVIHdr.dwStreams, MAX_AVISTREAMS);
1637 return AVIERR_UNSUPPORTED;
1640 /* adjust permissions if copyrighted material in file */
1641 if (MainAVIHdr.dwFlags & AVIFILEINFO_COPYRIGHTED) {
1642 This->uMode &= ~MMIO_RWMODE;
1643 This->uMode |= MMIO_READ;
1646 /* convert MainAVIHeader into AVIFILINFOW */
1647 memset(&This->fInfo, 0, sizeof(This->fInfo));
1648 This->fInfo.dwRate = MainAVIHdr.dwMicroSecPerFrame;
1649 This->fInfo.dwScale = 1000000;
1650 This->fInfo.dwMaxBytesPerSec = MainAVIHdr.dwMaxBytesPerSec;
1651 This->fInfo.dwFlags = MainAVIHdr.dwFlags;
1652 This->fInfo.dwCaps = AVIFILECAPS_CANREAD|AVIFILECAPS_CANWRITE;
1653 This->fInfo.dwLength = MainAVIHdr.dwTotalFrames;
1654 This->fInfo.dwStreams = MainAVIHdr.dwStreams;
1655 This->fInfo.dwSuggestedBufferSize = 0;
1656 This->fInfo.dwWidth = MainAVIHdr.dwWidth;
1657 This->fInfo.dwHeight = MainAVIHdr.dwHeight;
1658 LoadStringW(AVIFILE_hModule, IDS_AVIFILETYPE, This->fInfo.szFileType,
1659 sizeof(This->fInfo.szFileType)/sizeof(This->fInfo.szFileType[0]));
1661 /* go back to into header list */
1662 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
1663 return AVIERR_FILEREAD;
1665 /* foreach stream exists a "LIST","strl" chunk */
1666 for (nStream = 0; nStream < This->fInfo.dwStreams; nStream++) {
1667 /* get next nested chunk in this "LIST","strl" */
1668 if (mmioDescend(This->hmmio, &ckLIST2, &ckLIST1, 0) != S_OK)
1669 return AVIERR_FILEREAD;
1671 /* nested chunk must be of type "LIST","strl" -- when not normally JUNK */
1672 if (ckLIST2.ckid == FOURCC_LIST &&
1673 ckLIST2.fccType == listtypeSTREAMHEADER) {
1674 pStream = This->ppStreams[nStream] =
1675 HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IAVIStreamImpl));
1676 if (pStream == NULL)
1677 return AVIERR_MEMORY;
1678 AVIFILE_ConstructAVIStream(This, nStream, NULL);
1680 ck.ckid = 0;
1681 while (mmioDescend(This->hmmio, &ck, &ckLIST2, 0) == S_OK) {
1682 switch (ck.ckid) {
1683 case ckidSTREAMHANDLERDATA:
1684 if (pStream->lpHandlerData != NULL)
1685 return AVIERR_BADFORMAT;
1686 pStream->lpHandlerData = HeapAlloc(GetProcessHeap(), 0, ck.cksize);
1687 if (pStream->lpHandlerData == NULL)
1688 return AVIERR_MEMORY;
1689 pStream->cbHandlerData = ck.cksize;
1691 if (mmioRead(This->hmmio, pStream->lpHandlerData, ck.cksize) != ck.cksize)
1692 return AVIERR_FILEREAD;
1693 break;
1694 case ckidSTREAMFORMAT:
1695 if (pStream->lpFormat != NULL)
1696 return AVIERR_BADFORMAT;
1697 if (ck.cksize == 0)
1698 break;
1700 pStream->lpFormat = HeapAlloc(GetProcessHeap(), 0, ck.cksize);
1701 if (pStream->lpFormat == NULL)
1702 return AVIERR_MEMORY;
1703 pStream->cbFormat = ck.cksize;
1705 if (mmioRead(This->hmmio, pStream->lpFormat, ck.cksize) != ck.cksize)
1706 return AVIERR_FILEREAD;
1708 if (pStream->sInfo.fccType == streamtypeVIDEO) {
1709 LPBITMAPINFOHEADER lpbi = pStream->lpFormat;
1711 /* some corrections to the video format */
1712 if (lpbi->biClrUsed == 0 && lpbi->biBitCount <= 8)
1713 lpbi->biClrUsed = 1u << lpbi->biBitCount;
1714 if (lpbi->biCompression == BI_RGB && lpbi->biSizeImage == 0)
1715 lpbi->biSizeImage = DIBWIDTHBYTES(*lpbi) * lpbi->biHeight;
1716 if (lpbi->biCompression != BI_RGB && lpbi->biBitCount == 8) {
1717 if (pStream->sInfo.fccHandler == mmioFOURCC('R','L','E','0') ||
1718 pStream->sInfo.fccHandler == mmioFOURCC('R','L','E',' '))
1719 lpbi->biCompression = BI_RLE8;
1721 if (lpbi->biCompression == BI_RGB &&
1722 (pStream->sInfo.fccHandler == 0 ||
1723 pStream->sInfo.fccHandler == mmioFOURCC('N','O','N','E')))
1724 pStream->sInfo.fccHandler = comptypeDIB;
1726 /* init rcFrame if it's empty */
1727 if (IsRectEmpty(&pStream->sInfo.rcFrame))
1728 SetRect(&pStream->sInfo.rcFrame, 0, 0, lpbi->biWidth, lpbi->biHeight);
1730 break;
1731 case ckidSTREAMHEADER:
1733 static const WCHAR streamTypeFmt[] = {'%','4','.','4','h','s',0};
1734 static const WCHAR streamNameFmt[] = {'%','s',' ','%','s',' ','#','%','d',0};
1736 AVIStreamHeader streamHdr;
1737 WCHAR szType[25];
1738 UINT count;
1739 LONG n = ck.cksize;
1741 if (ck.cksize > sizeof(streamHdr))
1742 n = sizeof(streamHdr);
1744 if (mmioRead(This->hmmio, (HPSTR)&streamHdr, n) != n)
1745 return AVIERR_FILEREAD;
1747 pStream->sInfo.fccType = streamHdr.fccType;
1748 pStream->sInfo.fccHandler = streamHdr.fccHandler;
1749 pStream->sInfo.dwFlags = streamHdr.dwFlags;
1750 pStream->sInfo.wPriority = streamHdr.wPriority;
1751 pStream->sInfo.wLanguage = streamHdr.wLanguage;
1752 pStream->sInfo.dwInitialFrames = streamHdr.dwInitialFrames;
1753 pStream->sInfo.dwScale = streamHdr.dwScale;
1754 pStream->sInfo.dwRate = streamHdr.dwRate;
1755 pStream->sInfo.dwStart = streamHdr.dwStart;
1756 pStream->sInfo.dwLength = streamHdr.dwLength;
1757 pStream->sInfo.dwSuggestedBufferSize = 0;
1758 pStream->sInfo.dwQuality = streamHdr.dwQuality;
1759 pStream->sInfo.dwSampleSize = streamHdr.dwSampleSize;
1760 pStream->sInfo.rcFrame.left = streamHdr.rcFrame.left;
1761 pStream->sInfo.rcFrame.top = streamHdr.rcFrame.top;
1762 pStream->sInfo.rcFrame.right = streamHdr.rcFrame.right;
1763 pStream->sInfo.rcFrame.bottom = streamHdr.rcFrame.bottom;
1764 pStream->sInfo.dwEditCount = 0;
1765 pStream->sInfo.dwFormatChangeCount = 0;
1767 /* generate description for stream like "filename.avi Type #n" */
1768 if (streamHdr.fccType == streamtypeVIDEO)
1769 LoadStringW(AVIFILE_hModule, IDS_VIDEO, szType, sizeof(szType)/sizeof(szType[0]));
1770 else if (streamHdr.fccType == streamtypeAUDIO)
1771 LoadStringW(AVIFILE_hModule, IDS_AUDIO, szType, sizeof(szType)/sizeof(szType[0]));
1772 else
1773 wsprintfW(szType, streamTypeFmt, (char*)&streamHdr.fccType);
1775 /* get count of this streamtype up to this stream */
1776 count = 0;
1777 for (n = nStream; 0 <= n; n--) {
1778 if (This->ppStreams[n]->sInfo.fccHandler == streamHdr.fccType)
1779 count++;
1782 memset(pStream->sInfo.szName, 0, sizeof(pStream->sInfo.szName));
1784 /* FIXME: avoid overflow -- better use wsnprintfW, which doesn't exists ! */
1785 wsprintfW(pStream->sInfo.szName, streamNameFmt,
1786 AVIFILE_BasenameW(This->szFileName), szType, count);
1788 break;
1789 case ckidSTREAMNAME:
1790 { /* streamname will be saved as ASCII string */
1791 LPSTR str = HeapAlloc(GetProcessHeap(), 0, ck.cksize);
1792 if (str == NULL)
1793 return AVIERR_MEMORY;
1795 if (mmioRead(This->hmmio, str, ck.cksize) != ck.cksize)
1797 HeapFree(GetProcessHeap(), 0, str);
1798 return AVIERR_FILEREAD;
1801 MultiByteToWideChar(CP_ACP, 0, str, -1, pStream->sInfo.szName,
1802 sizeof(pStream->sInfo.szName)/sizeof(pStream->sInfo.szName[0]));
1804 HeapFree(GetProcessHeap(), 0, str);
1806 break;
1807 case ckidAVIPADDING:
1808 case mmioFOURCC('p','a','d','d'):
1809 break;
1810 default:
1811 WARN(": found extra chunk 0x%08X\n", ck.ckid);
1812 hr = ReadChunkIntoExtra(&pStream->extra, This->hmmio, &ck);
1813 if (FAILED(hr))
1814 return hr;
1816 if (pStream->lpFormat != NULL && pStream->sInfo.fccType == streamtypeAUDIO)
1818 WAVEFORMATEX *wfx = pStream->lpFormat; /* wfx->nBlockAlign = wfx->nChannels * wfx->wBitsPerSample / 8; could be added */
1819 pStream->sInfo.dwSampleSize = wfx->nBlockAlign; /* to deal with corrupt wfx->nBlockAlign but Windows doesn't do this */
1820 TRACE("Block size reset to %u, chan=%u bpp=%u\n", wfx->nBlockAlign, wfx->nChannels, wfx->wBitsPerSample);
1821 pStream->sInfo.dwScale = 1;
1822 pStream->sInfo.dwRate = wfx->nSamplesPerSec;
1824 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
1825 return AVIERR_FILEREAD;
1827 } else {
1828 /* nested chunks in "LIST","hdrl" which are not of type "LIST","strl" */
1829 hr = ReadChunkIntoExtra(&This->fileextra, This->hmmio, &ckLIST2);
1830 if (FAILED(hr))
1831 return hr;
1833 if (mmioAscend(This->hmmio, &ckLIST2, 0) != S_OK)
1834 return AVIERR_FILEREAD;
1837 /* read any extra headers in "LIST","hdrl" */
1838 FindChunkAndKeepExtras(&This->fileextra, This->hmmio, &ck, &ckLIST1, 0);
1839 if (mmioAscend(This->hmmio, &ckLIST1, 0) != S_OK)
1840 return AVIERR_FILEREAD;
1842 /* search "LIST","movi" chunk in "RIFF","AVI " */
1843 ckLIST1.fccType = listtypeAVIMOVIE;
1844 hr = FindChunkAndKeepExtras(&This->fileextra, This->hmmio, &ckLIST1, &ckRIFF,
1845 MMIO_FINDLIST);
1846 if (FAILED(hr))
1847 return hr;
1849 This->dwMoviChunkPos = ckLIST1.dwDataOffset;
1850 This->dwIdxChunkPos = ckLIST1.cksize + ckLIST1.dwDataOffset;
1851 if (mmioAscend(This->hmmio, &ckLIST1, 0) != S_OK)
1852 return AVIERR_FILEREAD;
1854 /* try to find an index */
1855 ck.ckid = ckidAVINEWINDEX;
1856 hr = FindChunkAndKeepExtras(&This->fileextra, This->hmmio,
1857 &ck, &ckRIFF, MMIO_FINDCHUNK);
1858 if (SUCCEEDED(hr) && ck.cksize > 0) {
1859 if (FAILED(AVIFILE_LoadIndex(This, ck.cksize, ckLIST1.dwDataOffset)))
1860 This->fInfo.dwFlags &= ~AVIFILEINFO_HASINDEX;
1863 /* when we haven't found an index or it's bad, then build one
1864 * by parsing 'movi' chunk */
1865 if ((This->fInfo.dwFlags & AVIFILEINFO_HASINDEX) == 0) {
1866 for (nStream = 0; nStream < This->fInfo.dwStreams; nStream++)
1867 This->ppStreams[nStream]->lLastFrame = -1;
1869 if (mmioSeek(This->hmmio, ckLIST1.dwDataOffset + sizeof(DWORD), SEEK_SET) == -1) {
1870 ERR(": Oops, can't seek back to 'movi' chunk!\n");
1871 return AVIERR_FILEREAD;
1874 /* seek through the 'movi' list until end */
1875 while (mmioDescend(This->hmmio, &ck, &ckLIST1, 0) == S_OK) {
1876 if (ck.ckid != FOURCC_LIST) {
1877 if (mmioAscend(This->hmmio, &ck, 0) == S_OK) {
1878 nStream = StreamFromFOURCC(ck.ckid);
1880 if (nStream > This->fInfo.dwStreams)
1881 return AVIERR_BADFORMAT;
1883 AVIFILE_AddFrame(This->ppStreams[nStream], ck.ckid, ck.cksize,
1884 ck.dwDataOffset - 2 * sizeof(DWORD), 0);
1885 } else {
1886 nStream = StreamFromFOURCC(ck.ckid);
1887 WARN(": file seems to be truncated!\n");
1888 if (nStream <= This->fInfo.dwStreams &&
1889 This->ppStreams[nStream]->sInfo.dwSampleSize > 0) {
1890 ck.cksize = mmioSeek(This->hmmio, 0, SEEK_END);
1891 if (ck.cksize != -1) {
1892 ck.cksize -= ck.dwDataOffset;
1893 ck.cksize &= ~(This->ppStreams[nStream]->sInfo.dwSampleSize - 1);
1895 AVIFILE_AddFrame(This->ppStreams[nStream], ck.ckid, ck.cksize,
1896 ck.dwDataOffset - 2 * sizeof(DWORD), 0);
1904 for (nStream = 0; nStream < This->fInfo.dwStreams; nStream++)
1906 DWORD sugbuf = This->ppStreams[nStream]->sInfo.dwSuggestedBufferSize;
1907 if (This->fInfo.dwSuggestedBufferSize < sugbuf)
1908 This->fInfo.dwSuggestedBufferSize = sugbuf;
1911 /* find other chunks */
1912 FindChunkAndKeepExtras(&This->fileextra, This->hmmio, &ck, &ckRIFF, 0);
1914 return AVIERR_OK;
1917 static HRESULT AVIFILE_LoadIndex(const IAVIFileImpl *This, DWORD size, DWORD offset)
1919 AVIINDEXENTRY *lp;
1920 DWORD pos, n;
1921 HRESULT hr = AVIERR_OK;
1922 BOOL bAbsolute = TRUE;
1924 lp = HeapAlloc(GetProcessHeap(), 0, IDX_PER_BLOCK * sizeof(AVIINDEXENTRY));
1925 if (lp == NULL)
1926 return AVIERR_MEMORY;
1928 /* adjust limits for index tables, so that inserting will be faster */
1929 for (n = 0; n < This->fInfo.dwStreams; n++) {
1930 IAVIStreamImpl *pStream = This->ppStreams[n];
1932 pStream->lLastFrame = -1;
1934 if (pStream->idxFrames != NULL) {
1935 HeapFree(GetProcessHeap(), 0, pStream->idxFrames);
1936 pStream->idxFrames = NULL;
1937 pStream->nIdxFrames = 0;
1940 if (pStream->sInfo.dwSampleSize != 0) {
1941 if (n > 0 && This->fInfo.dwFlags & AVIFILEINFO_ISINTERLEAVED) {
1942 pStream->nIdxFrames = This->ppStreams[0]->nIdxFrames;
1943 } else if (pStream->sInfo.dwSuggestedBufferSize) {
1944 pStream->nIdxFrames =
1945 pStream->sInfo.dwLength / pStream->sInfo.dwSuggestedBufferSize;
1947 } else
1948 pStream->nIdxFrames = pStream->sInfo.dwLength;
1950 pStream->idxFrames =
1951 HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, pStream->nIdxFrames * sizeof(AVIINDEXENTRY));
1952 if (pStream->idxFrames == NULL && pStream->nIdxFrames > 0) {
1953 pStream->nIdxFrames = 0;
1954 HeapFree(GetProcessHeap(), 0, lp);
1955 return AVIERR_MEMORY;
1959 pos = (DWORD)-1;
1960 while (size != 0) {
1961 LONG read = min(IDX_PER_BLOCK * sizeof(AVIINDEXENTRY), size);
1963 if (mmioRead(This->hmmio, (HPSTR)lp, read) != read) {
1964 hr = AVIERR_FILEREAD;
1965 break;
1967 size -= read;
1969 if (pos == (DWORD)-1)
1970 pos = offset - lp->dwChunkOffset + sizeof(DWORD);
1972 AVIFILE_ParseIndex(This, lp, read / sizeof(AVIINDEXENTRY),
1973 pos, &bAbsolute);
1976 HeapFree(GetProcessHeap(), 0, lp);
1978 /* checking ... */
1979 for (n = 0; n < This->fInfo.dwStreams; n++) {
1980 IAVIStreamImpl *pStream = This->ppStreams[n];
1982 if (pStream->sInfo.dwSampleSize == 0 &&
1983 pStream->sInfo.dwLength != pStream->lLastFrame+1)
1984 ERR("stream %u length mismatch: dwLength=%u found=%d\n",
1985 n, pStream->sInfo.dwLength, pStream->lLastFrame);
1988 return hr;
1991 static HRESULT AVIFILE_ParseIndex(const IAVIFileImpl *This, AVIINDEXENTRY *lp,
1992 LONG count, DWORD pos, BOOL *bAbsolute)
1994 if (lp == NULL)
1995 return AVIERR_BADPARAM;
1997 for (; count > 0; count--, lp++) {
1998 WORD nStream = StreamFromFOURCC(lp->ckid);
2000 if (lp->ckid == listtypeAVIRECORD || nStream == 0x7F)
2001 continue; /* skip these */
2003 if (nStream > This->fInfo.dwStreams)
2004 return AVIERR_BADFORMAT;
2006 if (*bAbsolute && lp->dwChunkOffset < This->dwMoviChunkPos)
2007 *bAbsolute = FALSE;
2009 if (*bAbsolute)
2010 lp->dwChunkOffset += sizeof(DWORD);
2011 else
2012 lp->dwChunkOffset += pos;
2014 if (FAILED(AVIFILE_AddFrame(This->ppStreams[nStream], lp->ckid, lp->dwChunkLength, lp->dwChunkOffset, lp->dwFlags)))
2015 return AVIERR_MEMORY;
2018 return AVIERR_OK;
2021 static HRESULT AVIFILE_ReadBlock(IAVIStreamImpl *This, DWORD pos,
2022 LPVOID buffer, DWORD size)
2024 /* pre-conditions */
2025 assert(This != NULL);
2026 assert(This->paf != NULL);
2027 assert(This->paf->hmmio != NULL);
2028 assert(This->sInfo.dwStart <= pos && pos < This->sInfo.dwLength);
2029 assert(pos <= This->lLastFrame);
2031 /* should we read as much as block gives us? */
2032 if (size == 0 || size > This->idxFrames[pos].dwChunkLength)
2033 size = This->idxFrames[pos].dwChunkLength;
2035 /* read into out own buffer or given one? */
2036 if (buffer == NULL) {
2037 /* we also read the chunk */
2038 size += 2 * sizeof(DWORD);
2040 /* check that buffer is big enough -- don't trust dwSuggestedBufferSize */
2041 if (This->lpBuffer == NULL || This->cbBuffer < size) {
2042 DWORD maxSize = max(size, This->sInfo.dwSuggestedBufferSize);
2044 if (This->lpBuffer == NULL) {
2045 This->lpBuffer = HeapAlloc(GetProcessHeap(), 0, maxSize);
2046 if (!This->lpBuffer) return AVIERR_MEMORY;
2047 } else {
2048 void *new_buffer = HeapReAlloc(GetProcessHeap(), 0, This->lpBuffer, maxSize);
2049 if (!new_buffer) return AVIERR_MEMORY;
2050 This->lpBuffer = new_buffer;
2052 This->cbBuffer = maxSize;
2055 /* now read the complete chunk into our buffer */
2056 if (mmioSeek(This->paf->hmmio, This->idxFrames[pos].dwChunkOffset, SEEK_SET) == -1)
2057 return AVIERR_FILEREAD;
2058 if (mmioRead(This->paf->hmmio, (HPSTR)This->lpBuffer, size) != size)
2059 return AVIERR_FILEREAD;
2061 /* check if it was the correct block which we have read */
2062 if (This->lpBuffer[0] != This->idxFrames[pos].ckid ||
2063 This->lpBuffer[1] != This->idxFrames[pos].dwChunkLength) {
2064 ERR(": block %d not found at 0x%08X\n", pos, This->idxFrames[pos].dwChunkOffset);
2065 ERR(": Index says: '%4.4s'(0x%08X) size 0x%08X\n",
2066 (char*)&This->idxFrames[pos].ckid, This->idxFrames[pos].ckid,
2067 This->idxFrames[pos].dwChunkLength);
2068 ERR(": Data says: '%4.4s'(0x%08X) size 0x%08X\n",
2069 (char*)&This->lpBuffer[0], This->lpBuffer[0], This->lpBuffer[1]);
2070 return AVIERR_FILEREAD;
2072 } else {
2073 if (mmioSeek(This->paf->hmmio, This->idxFrames[pos].dwChunkOffset + 2 * sizeof(DWORD), SEEK_SET) == -1)
2074 return AVIERR_FILEREAD;
2075 if (mmioRead(This->paf->hmmio, buffer, size) != size)
2076 return AVIERR_FILEREAD;
2079 return AVIERR_OK;
2082 static void AVIFILE_SamplesToBlock(const IAVIStreamImpl *This, LPLONG pos, LPLONG offset)
2084 LONG block;
2086 /* pre-conditions */
2087 assert(This != NULL);
2088 assert(pos != NULL);
2089 assert(offset != NULL);
2090 assert(This->sInfo.dwSampleSize != 0);
2091 assert(*pos >= This->sInfo.dwStart);
2093 /* convert start sample to start bytes */
2094 (*offset) = (*pos) - This->sInfo.dwStart;
2095 (*offset) *= This->sInfo.dwSampleSize;
2097 /* convert bytes to block number */
2098 for (block = 0; block <= This->lLastFrame; block++) {
2099 if (This->idxFrames[block].dwChunkLength <= *offset)
2100 (*offset) -= This->idxFrames[block].dwChunkLength;
2101 else
2102 break;
2105 *pos = block;
2108 static HRESULT AVIFILE_SaveFile(IAVIFileImpl *This)
2110 MainAVIHeader MainAVIHdr;
2111 IAVIStreamImpl* pStream;
2112 MMCKINFO ckRIFF;
2113 MMCKINFO ckLIST1;
2114 MMCKINFO ckLIST2;
2115 MMCKINFO ck;
2116 DWORD nStream;
2117 DWORD dwPos;
2118 HRESULT hr;
2120 /* initialize some things */
2121 if (This->dwMoviChunkPos == 0)
2122 AVIFILE_ComputeMoviStart(This);
2124 /* written one record to much? */
2125 if (This->ckLastRecord.dwFlags & MMIO_DIRTY) {
2126 This->dwNextFramePos -= 3 * sizeof(DWORD);
2127 if (This->nIdxRecords > 0)
2128 This->nIdxRecords--;
2131 AVIFILE_UpdateInfo(This);
2133 assert(This->fInfo.dwScale != 0);
2135 memset(&MainAVIHdr, 0, sizeof(MainAVIHdr));
2136 MainAVIHdr.dwMicroSecPerFrame = MulDiv(This->fInfo.dwRate, 1000000,
2137 This->fInfo.dwScale);
2138 MainAVIHdr.dwMaxBytesPerSec = This->fInfo.dwMaxBytesPerSec;
2139 MainAVIHdr.dwPaddingGranularity = AVI_HEADERSIZE;
2140 MainAVIHdr.dwFlags = This->fInfo.dwFlags;
2141 MainAVIHdr.dwTotalFrames = This->fInfo.dwLength;
2142 MainAVIHdr.dwInitialFrames = 0;
2143 MainAVIHdr.dwStreams = This->fInfo.dwStreams;
2144 MainAVIHdr.dwSuggestedBufferSize = This->fInfo.dwSuggestedBufferSize;
2145 MainAVIHdr.dwWidth = This->fInfo.dwWidth;
2146 MainAVIHdr.dwHeight = This->fInfo.dwHeight;
2147 MainAVIHdr.dwInitialFrames = This->dwInitialFrames;
2149 /* now begin writing ... */
2150 mmioSeek(This->hmmio, 0, SEEK_SET);
2152 /* RIFF chunk */
2153 ckRIFF.cksize = 0;
2154 ckRIFF.fccType = formtypeAVI;
2155 if (mmioCreateChunk(This->hmmio, &ckRIFF, MMIO_CREATERIFF) != S_OK)
2156 return AVIERR_FILEWRITE;
2158 /* AVI headerlist */
2159 ckLIST1.cksize = 0;
2160 ckLIST1.fccType = listtypeAVIHEADER;
2161 if (mmioCreateChunk(This->hmmio, &ckLIST1, MMIO_CREATELIST) != S_OK)
2162 return AVIERR_FILEWRITE;
2164 /* MainAVIHeader */
2165 ck.ckid = ckidAVIMAINHDR;
2166 ck.cksize = sizeof(MainAVIHdr);
2167 ck.fccType = 0;
2168 if (mmioCreateChunk(This->hmmio, &ck, 0) != S_OK)
2169 return AVIERR_FILEWRITE;
2170 if (mmioWrite(This->hmmio, (HPSTR)&MainAVIHdr, ck.cksize) != ck.cksize)
2171 return AVIERR_FILEWRITE;
2172 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
2173 return AVIERR_FILEWRITE;
2175 /* write the headers of each stream into a separate streamheader list */
2176 for (nStream = 0; nStream < This->fInfo.dwStreams; nStream++) {
2177 AVIStreamHeader strHdr;
2179 pStream = This->ppStreams[nStream];
2181 /* begin the new streamheader list */
2182 ckLIST2.cksize = 0;
2183 ckLIST2.fccType = listtypeSTREAMHEADER;
2184 if (mmioCreateChunk(This->hmmio, &ckLIST2, MMIO_CREATELIST) != S_OK)
2185 return AVIERR_FILEWRITE;
2187 /* create an AVIStreamHeader from the AVSTREAMINFO */
2188 strHdr.fccType = pStream->sInfo.fccType;
2189 strHdr.fccHandler = pStream->sInfo.fccHandler;
2190 strHdr.dwFlags = pStream->sInfo.dwFlags;
2191 strHdr.wPriority = pStream->sInfo.wPriority;
2192 strHdr.wLanguage = pStream->sInfo.wLanguage;
2193 strHdr.dwInitialFrames = pStream->sInfo.dwInitialFrames;
2194 strHdr.dwScale = pStream->sInfo.dwScale;
2195 strHdr.dwRate = pStream->sInfo.dwRate;
2196 strHdr.dwStart = pStream->sInfo.dwStart;
2197 strHdr.dwLength = pStream->sInfo.dwLength;
2198 strHdr.dwSuggestedBufferSize = pStream->sInfo.dwSuggestedBufferSize;
2199 strHdr.dwQuality = pStream->sInfo.dwQuality;
2200 strHdr.dwSampleSize = pStream->sInfo.dwSampleSize;
2201 strHdr.rcFrame.left = pStream->sInfo.rcFrame.left;
2202 strHdr.rcFrame.top = pStream->sInfo.rcFrame.top;
2203 strHdr.rcFrame.right = pStream->sInfo.rcFrame.right;
2204 strHdr.rcFrame.bottom = pStream->sInfo.rcFrame.bottom;
2206 /* now write the AVIStreamHeader */
2207 ck.ckid = ckidSTREAMHEADER;
2208 ck.cksize = sizeof(strHdr);
2209 if (mmioCreateChunk(This->hmmio, &ck, 0) != S_OK)
2210 return AVIERR_FILEWRITE;
2211 if (mmioWrite(This->hmmio, (HPSTR)&strHdr, ck.cksize) != ck.cksize)
2212 return AVIERR_FILEWRITE;
2213 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
2214 return AVIERR_FILEWRITE;
2216 /* ... the hopefully ever present streamformat ... */
2217 ck.ckid = ckidSTREAMFORMAT;
2218 ck.cksize = pStream->cbFormat;
2219 if (mmioCreateChunk(This->hmmio, &ck, 0) != S_OK)
2220 return AVIERR_FILEWRITE;
2221 if (pStream->lpFormat != NULL && ck.cksize > 0) {
2222 if (mmioWrite(This->hmmio, pStream->lpFormat, ck.cksize) != ck.cksize)
2223 return AVIERR_FILEWRITE;
2225 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
2226 return AVIERR_FILEWRITE;
2228 /* ... some optional existing handler data ... */
2229 if (pStream->lpHandlerData != NULL && pStream->cbHandlerData > 0) {
2230 ck.ckid = ckidSTREAMHANDLERDATA;
2231 ck.cksize = pStream->cbHandlerData;
2232 if (mmioCreateChunk(This->hmmio, &ck, 0) != S_OK)
2233 return AVIERR_FILEWRITE;
2234 if (mmioWrite(This->hmmio, pStream->lpHandlerData, ck.cksize) != ck.cksize)
2235 return AVIERR_FILEWRITE;
2236 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
2237 return AVIERR_FILEWRITE;
2240 /* ... some optional additional extra chunk for this stream ... */
2241 if (pStream->extra.lp != NULL && pStream->extra.cb > 0) {
2242 /* the chunk header(s) are already in the structure */
2243 if (mmioWrite(This->hmmio, pStream->extra.lp, pStream->extra.cb) != pStream->extra.cb)
2244 return AVIERR_FILEWRITE;
2247 /* ... an optional name for this stream ... */
2248 if (lstrlenW(pStream->sInfo.szName) > 0) {
2249 LPSTR str;
2251 ck.ckid = ckidSTREAMNAME;
2252 ck.cksize = lstrlenW(pStream->sInfo.szName) + 1;
2253 if (ck.cksize & 1) /* align */
2254 ck.cksize++;
2255 if (mmioCreateChunk(This->hmmio, &ck, 0) != S_OK)
2256 return AVIERR_FILEWRITE;
2258 /* the streamname must be saved in ASCII not Unicode */
2259 str = HeapAlloc(GetProcessHeap(), 0, ck.cksize);
2260 if (str == NULL)
2261 return AVIERR_MEMORY;
2262 WideCharToMultiByte(CP_ACP, 0, pStream->sInfo.szName, -1, str,
2263 ck.cksize, NULL, NULL);
2265 if (mmioWrite(This->hmmio, str, ck.cksize) != ck.cksize) {
2266 HeapFree(GetProcessHeap(), 0, str);
2267 return AVIERR_FILEWRITE;
2270 HeapFree(GetProcessHeap(), 0, str);
2271 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
2272 return AVIERR_FILEWRITE;
2275 /* close streamheader list for this stream */
2276 if (mmioAscend(This->hmmio, &ckLIST2, 0) != S_OK)
2277 return AVIERR_FILEWRITE;
2278 } /* for (0 <= nStream < MainAVIHdr.dwStreams) */
2280 /* close the aviheader list */
2281 if (mmioAscend(This->hmmio, &ckLIST1, 0) != S_OK)
2282 return AVIERR_FILEWRITE;
2284 /* check for padding to pre-guessed 'movi'-chunk position */
2285 dwPos = ckLIST1.dwDataOffset + ckLIST1.cksize;
2286 if (This->dwMoviChunkPos - 2 * sizeof(DWORD) > dwPos) {
2287 ck.ckid = ckidAVIPADDING;
2288 ck.cksize = This->dwMoviChunkPos - dwPos - 4 * sizeof(DWORD);
2289 assert((LONG)ck.cksize >= 0);
2291 if (mmioCreateChunk(This->hmmio, &ck, 0) != S_OK)
2292 return AVIERR_FILEWRITE;
2293 if (mmioSeek(This->hmmio, ck.cksize, SEEK_CUR) == -1)
2294 return AVIERR_FILEWRITE;
2295 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
2296 return AVIERR_FILEWRITE;
2299 /* now write the 'movi' chunk */
2300 mmioSeek(This->hmmio, This->dwMoviChunkPos - 2 * sizeof(DWORD), SEEK_SET);
2301 ckLIST1.cksize = 0;
2302 ckLIST1.fccType = listtypeAVIMOVIE;
2303 if (mmioCreateChunk(This->hmmio, &ckLIST1, MMIO_CREATELIST) != S_OK)
2304 return AVIERR_FILEWRITE;
2305 if (mmioSeek(This->hmmio, This->dwNextFramePos, SEEK_SET) == -1)
2306 return AVIERR_FILEWRITE;
2307 if (mmioAscend(This->hmmio, &ckLIST1, 0) != S_OK)
2308 return AVIERR_FILEWRITE;
2310 /* write 'idx1' chunk */
2311 hr = AVIFILE_SaveIndex(This);
2312 if (FAILED(hr))
2313 return hr;
2315 /* write optional extra file chunks */
2316 if (This->fileextra.lp != NULL && This->fileextra.cb > 0) {
2317 /* as for the streams, are the chunk header(s) in the structure */
2318 if (mmioWrite(This->hmmio, This->fileextra.lp, This->fileextra.cb) != This->fileextra.cb)
2319 return AVIERR_FILEWRITE;
2322 /* close RIFF chunk */
2323 if (mmioAscend(This->hmmio, &ckRIFF, 0) != S_OK)
2324 return AVIERR_FILEWRITE;
2326 /* add some JUNK at end for bad parsers */
2327 memset(&ckRIFF, 0, sizeof(ckRIFF));
2328 mmioWrite(This->hmmio, (HPSTR)&ckRIFF, sizeof(ckRIFF));
2329 mmioFlush(This->hmmio, 0);
2331 return AVIERR_OK;
2334 static HRESULT AVIFILE_SaveIndex(const IAVIFileImpl *This)
2336 IAVIStreamImpl *pStream;
2337 AVIINDEXENTRY idx;
2338 MMCKINFO ck;
2339 DWORD nStream;
2340 LONG n;
2342 ck.ckid = ckidAVINEWINDEX;
2343 ck.cksize = 0;
2344 if (mmioCreateChunk(This->hmmio, &ck, 0) != S_OK)
2345 return AVIERR_FILEWRITE;
2347 if (This->fInfo.dwFlags & AVIFILEINFO_ISINTERLEAVED) {
2348 /* is interleaved -- write block of corresponding frames */
2349 LONG lInitialFrames = 0;
2350 LONG stepsize;
2351 LONG i;
2353 if (This->ppStreams[0]->sInfo.dwSampleSize == 0)
2354 stepsize = 1;
2355 else
2356 stepsize = AVIStreamTimeToSample((PAVISTREAM)This->ppStreams[0], 1000000);
2358 assert(stepsize > 0);
2360 for (nStream = 0; nStream < This->fInfo.dwStreams; nStream++) {
2361 if (lInitialFrames < This->ppStreams[nStream]->sInfo.dwInitialFrames)
2362 lInitialFrames = This->ppStreams[nStream]->sInfo.dwInitialFrames;
2365 for (i = -lInitialFrames; i < (LONG)This->fInfo.dwLength - lInitialFrames;
2366 i += stepsize) {
2367 DWORD nFrame = lInitialFrames + i;
2369 assert(nFrame < This->nIdxRecords);
2371 idx.ckid = listtypeAVIRECORD;
2372 idx.dwFlags = AVIIF_LIST;
2373 idx.dwChunkLength = This->idxRecords[nFrame].dwChunkLength;
2374 idx.dwChunkOffset = This->idxRecords[nFrame].dwChunkOffset
2375 - This->dwMoviChunkPos;
2376 if (mmioWrite(This->hmmio, (HPSTR)&idx, sizeof(idx)) != sizeof(idx))
2377 return AVIERR_FILEWRITE;
2379 for (nStream = 0; nStream < This->fInfo.dwStreams; nStream++) {
2380 pStream = This->ppStreams[nStream];
2382 /* heave we reached start of this stream? */
2383 if (-(LONG)pStream->sInfo.dwInitialFrames > i)
2384 continue;
2386 if (pStream->sInfo.dwInitialFrames < lInitialFrames)
2387 nFrame -= (lInitialFrames - pStream->sInfo.dwInitialFrames);
2389 /* reached end of this stream? */
2390 if (pStream->lLastFrame <= nFrame)
2391 continue;
2393 if ((pStream->sInfo.dwFlags & AVISTREAMINFO_FORMATCHANGES) &&
2394 pStream->sInfo.dwFormatChangeCount != 0 &&
2395 pStream->idxFmtChanges != NULL) {
2396 DWORD pos;
2398 for (pos = 0; pos < pStream->sInfo.dwFormatChangeCount; pos++) {
2399 if (pStream->idxFmtChanges[pos].ckid == nFrame) {
2400 idx.dwFlags = AVIIF_NOTIME;
2401 idx.ckid = MAKEAVICKID(cktypePALchange, pStream->nStream);
2402 idx.dwChunkLength = pStream->idxFmtChanges[pos].dwChunkLength;
2403 idx.dwChunkOffset = pStream->idxFmtChanges[pos].dwChunkOffset
2404 - This->dwMoviChunkPos;
2406 if (mmioWrite(This->hmmio, (HPSTR)&idx, sizeof(idx)) != sizeof(idx))
2407 return AVIERR_FILEWRITE;
2408 break;
2411 } /* if have formatchanges */
2413 idx.ckid = pStream->idxFrames[nFrame].ckid;
2414 idx.dwFlags = pStream->idxFrames[nFrame].dwFlags;
2415 idx.dwChunkLength = pStream->idxFrames[nFrame].dwChunkLength;
2416 idx.dwChunkOffset = pStream->idxFrames[nFrame].dwChunkOffset
2417 - This->dwMoviChunkPos;
2418 if (mmioWrite(This->hmmio, (HPSTR)&idx, sizeof(idx)) != sizeof(idx))
2419 return AVIERR_FILEWRITE;
2422 } else {
2423 /* not interleaved -- write index for each stream at once */
2424 for (nStream = 0; nStream < This->fInfo.dwStreams; nStream++) {
2425 pStream = This->ppStreams[nStream];
2427 for (n = 0; n <= pStream->lLastFrame; n++) {
2428 if ((pStream->sInfo.dwFlags & AVISTREAMINFO_FORMATCHANGES) &&
2429 (pStream->sInfo.dwFormatChangeCount != 0)) {
2430 DWORD pos;
2432 for (pos = 0; pos < pStream->sInfo.dwFormatChangeCount; pos++) {
2433 if (pStream->idxFmtChanges[pos].ckid == n) {
2434 idx.dwFlags = AVIIF_NOTIME;
2435 idx.ckid = MAKEAVICKID(cktypePALchange, pStream->nStream);
2436 idx.dwChunkLength = pStream->idxFmtChanges[pos].dwChunkLength;
2437 idx.dwChunkOffset =
2438 pStream->idxFmtChanges[pos].dwChunkOffset - This->dwMoviChunkPos;
2439 if (mmioWrite(This->hmmio, (HPSTR)&idx, sizeof(idx)) != sizeof(idx))
2440 return AVIERR_FILEWRITE;
2441 break;
2444 } /* if have formatchanges */
2446 idx.ckid = pStream->idxFrames[n].ckid;
2447 idx.dwFlags = pStream->idxFrames[n].dwFlags;
2448 idx.dwChunkLength = pStream->idxFrames[n].dwChunkLength;
2449 idx.dwChunkOffset = pStream->idxFrames[n].dwChunkOffset
2450 - This->dwMoviChunkPos;
2452 if (mmioWrite(This->hmmio, (HPSTR)&idx, sizeof(idx)) != sizeof(idx))
2453 return AVIERR_FILEWRITE;
2456 } /* if not interleaved */
2458 if (mmioAscend(This->hmmio, &ck, 0) != S_OK)
2459 return AVIERR_FILEWRITE;
2461 return AVIERR_OK;
2464 static ULONG AVIFILE_SearchStream(const IAVIFileImpl *This, DWORD fcc, LONG lSkip)
2466 UINT i;
2467 UINT nStream;
2469 /* pre-condition */
2470 assert(lSkip >= 0);
2472 if (fcc != 0) {
2473 /* search the number of the specified stream */
2474 nStream = (ULONG)-1;
2475 for (i = 0; i < This->fInfo.dwStreams; i++) {
2476 assert(This->ppStreams[i] != NULL);
2478 if (This->ppStreams[i]->sInfo.fccType == fcc) {
2479 if (lSkip == 0) {
2480 nStream = i;
2481 break;
2482 } else
2483 lSkip--;
2486 } else
2487 nStream = lSkip;
2489 return nStream;
2492 static void AVIFILE_UpdateInfo(IAVIFileImpl *This)
2494 UINT i;
2496 /* pre-conditions */
2497 assert(This != NULL);
2499 This->fInfo.dwMaxBytesPerSec = 0;
2500 This->fInfo.dwCaps = AVIFILECAPS_CANREAD|AVIFILECAPS_CANWRITE;
2501 This->fInfo.dwSuggestedBufferSize = 0;
2502 This->fInfo.dwWidth = 0;
2503 This->fInfo.dwHeight = 0;
2504 This->fInfo.dwScale = 0;
2505 This->fInfo.dwRate = 0;
2506 This->fInfo.dwLength = 0;
2507 This->dwInitialFrames = 0;
2509 for (i = 0; i < This->fInfo.dwStreams; i++) {
2510 AVISTREAMINFOW *psi;
2511 DWORD n;
2513 /* pre-conditions */
2514 assert(This->ppStreams[i] != NULL);
2516 psi = &This->ppStreams[i]->sInfo;
2517 assert(psi->dwScale != 0);
2518 assert(psi->dwRate != 0);
2520 if (i == 0) {
2521 /* use first stream timings as base */
2522 This->fInfo.dwScale = psi->dwScale;
2523 This->fInfo.dwRate = psi->dwRate;
2524 This->fInfo.dwLength = psi->dwLength;
2525 } else {
2526 n = AVIStreamSampleToSample((PAVISTREAM)This->ppStreams[0],
2527 (PAVISTREAM)This->ppStreams[i],psi->dwLength);
2528 if (This->fInfo.dwLength < n)
2529 This->fInfo.dwLength = n;
2532 if (This->dwInitialFrames < psi->dwInitialFrames)
2533 This->dwInitialFrames = psi->dwInitialFrames;
2535 if (This->fInfo.dwSuggestedBufferSize < psi->dwSuggestedBufferSize)
2536 This->fInfo.dwSuggestedBufferSize = psi->dwSuggestedBufferSize;
2538 if (psi->dwSampleSize != 0) {
2539 /* fixed sample size -- exact computation */
2540 This->fInfo.dwMaxBytesPerSec += MulDiv(psi->dwSampleSize, psi->dwRate,
2541 psi->dwScale);
2542 } else {
2543 /* variable sample size -- only upper limit */
2544 This->fInfo.dwMaxBytesPerSec += MulDiv(psi->dwSuggestedBufferSize,
2545 psi->dwRate, psi->dwScale);
2547 /* update dimensions */
2548 n = psi->rcFrame.right - psi->rcFrame.left;
2549 if (This->fInfo.dwWidth < n)
2550 This->fInfo.dwWidth = n;
2551 n = psi->rcFrame.bottom - psi->rcFrame.top;
2552 if (This->fInfo.dwHeight < n)
2553 This->fInfo.dwHeight = n;
2558 static HRESULT AVIFILE_WriteBlock(IAVIStreamImpl *This, DWORD block,
2559 FOURCC ckid, DWORD flags, LPCVOID buffer,
2560 LONG size)
2562 MMCKINFO ck;
2564 ck.ckid = ckid;
2565 ck.cksize = size;
2566 ck.fccType = 0;
2568 /* if no frame/block is already written, we must compute start of movi chunk */
2569 if (This->paf->dwMoviChunkPos == 0)
2570 AVIFILE_ComputeMoviStart(This->paf);
2572 if (mmioSeek(This->paf->hmmio, This->paf->dwNextFramePos, SEEK_SET) == -1)
2573 return AVIERR_FILEWRITE;
2575 if (mmioCreateChunk(This->paf->hmmio, &ck, 0) != S_OK)
2576 return AVIERR_FILEWRITE;
2577 if (buffer != NULL && size > 0) {
2578 if (mmioWrite(This->paf->hmmio, buffer, size) != size)
2579 return AVIERR_FILEWRITE;
2581 if (mmioAscend(This->paf->hmmio, &ck, 0) != S_OK)
2582 return AVIERR_FILEWRITE;
2584 This->paf->fDirty = TRUE;
2585 This->paf->dwNextFramePos = mmioSeek(This->paf->hmmio, 0, SEEK_CUR);
2587 return AVIFILE_AddFrame(This, ckid, size,
2588 ck.dwDataOffset - 2 * sizeof(DWORD), flags);