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
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.
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.
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
);
55 #define IDX_PER_BLOCK 2730
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
,
79 IAVIStream_fnFindSample
,
80 IAVIStream_fnReadFormat
,
81 IAVIStream_fnSetFormat
,
85 IAVIStream_fnReadData
,
86 IAVIStream_fnWriteData
,
90 typedef struct _IAVIFileImpl IAVIFileImpl
;
92 typedef struct _IAVIStreamImpl
{
94 const IAVIStreamVtbl
*lpVtbl
;
97 /* IAVIStream stuff */
99 DWORD nStream
; /* the n-th stream in file */
100 AVISTREAMINFOW sInfo
;
105 LPVOID lpHandlerData
;
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 */
121 struct _IAVIFileImpl
{
122 IUnknown IUnknown_inner
;
123 IAVIFile IAVIFile_iface
;
124 IPersistFile IPersistFile_iface
;
129 IAVIStreamImpl
*ppStreams
[MAX_AVISTREAMS
];
131 EXTRACHUNKS fileextra
;
133 DWORD dwMoviChunkPos
; /* some stuff for saving ... */
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 ... */
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
,
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
,
186 static void AVIFILE_UpdateInfo(IAVIFileImpl
*This
);
187 static HRESULT
AVIFILE_WriteBlock(IAVIStreamImpl
*This
, DWORD block
,
188 FOURCC ckid
, DWORD flags
, LPCVOID buffer
,
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
);
198 WARN("invalid parameter\n");
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
;
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
);
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
);
229 static ULONG WINAPI
IUnknown_fnRelease(IUnknown
*iface
)
231 IAVIFileImpl
*This
= impl_from_IUnknown(iface
);
232 ULONG ref
= InterlockedDecrement(&This
->ref
);
235 TRACE("(%p) ref=%d\n", This
, ref
);
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);
272 HeapFree(GetProcessHeap(), 0, This
);
277 static const IUnknownVtbl unk_vtbl
=
279 IUnknown_fnQueryInterface
,
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
);
312 return AVIERR_BADPARAM
;
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
;
325 static HRESULT WINAPI
IAVIFile_fnGetStream(IAVIFile
*iface
, IAVIStream
**avis
, DWORD fccType
,
328 IAVIFileImpl
*This
= impl_from_IAVIFile(iface
);
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
);
347 /* Sorry, but the specified stream doesn't exist */
348 return AVIERR_NODATA
;
351 static HRESULT WINAPI
IAVIFile_fnCreateStream(IAVIFile
*iface
, IAVIStream
**avis
,
354 IAVIFileImpl
*This
= impl_from_IAVIFile(iface
);
357 TRACE("(%p,%p,%p)\n", iface
, avis
, asi
);
359 /* check parameters */
360 if (avis
== NULL
|| asi
== NULL
)
361 return AVIERR_BADPARAM
;
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
++;
394 /* update our AVIFILEINFO structure */
395 AVIFILE_UpdateInfo(This
);
398 *avis
= (PAVISTREAM
)This
->ppStreams
[n
];
399 IAVIStream_AddRef(*avis
);
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 */
412 return AVIERR_BADPARAM
;
414 return AVIERR_BADSIZE
;
416 /* Do we have write permission? */
417 if ((This
->uMode
& MMIO_RWMODE
) == 0)
418 return AVIERR_READONLY
;
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
;
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
);
475 static HRESULT WINAPI
IAVIFile_fnDeleteStream(IAVIFile
*iface
, DWORD fccType
, LONG lParam
)
477 IAVIFileImpl
*This
= impl_from_IAVIFile(iface
);
480 TRACE("(%p,0x%08X,%d)\n", iface
, fccType
, lParam
);
482 /* check parameter */
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
--;
506 /* This->fInfo will be updated further when asked for */
509 return AVIERR_NODATA
;
512 static const struct IAVIFileVtbl avif_vt
= {
513 IAVIFile_fnQueryInterface
,
517 IAVIFile_fnGetStream
,
518 IAVIFile_fnCreateStream
,
519 IAVIFile_fnWriteData
,
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
;
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
);
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 */
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
;
617 return AVIFILE_LoadFile(This
);
620 static HRESULT WINAPI
IPersistFile_fnSave(IPersistFile
*iface
, LPCOLESTR pszFileName
,
623 TRACE("(%p,%s,%d)\n", iface
, debugstr_w(pszFileName
), fRemember
);
625 /* We write directly to disk, so nothing to do. */
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. */
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
);
663 static const struct IPersistFileVtbl pf_vt
= {
664 IPersistFile_fnQueryInterface
,
665 IPersistFile_fnAddRef
,
666 IPersistFile_fnRelease
,
667 IPersistFile_fnGetClassID
,
668 IPersistFile_fnIsDirty
,
671 IPersistFile_fnSaveCompleted
,
672 IPersistFile_fnGetCurFile
675 HRESULT
AVIFILE_CreateAVIFile(IUnknown
*pUnkOuter
, REFIID riid
, void **ppv
)
681 obj
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(IAVIFileImpl
));
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
;
690 obj
->outer_unk
= pUnkOuter
;
692 obj
->outer_unk
= &obj
->IUnknown_inner
;
694 hr
= IUnknown_QueryInterface(&obj
->IUnknown_inner
, riid
, ppv
);
695 IUnknown_Release(&obj
->IUnknown_inner
);
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
)) {
711 IAVIStream_AddRef(iface
);
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
);
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
);
747 static HRESULT WINAPI
IAVIStream_fnCreate(IAVIStream
*iface
, LPARAM lParam1
,
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
,
759 IAVIStreamImpl
*This
= (IAVIStreamImpl
*)iface
;
761 TRACE("(%p,%p,%d)\n", iface
, psi
, size
);
764 return AVIERR_BADPARAM
;
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
;
775 static LONG WINAPI
IAVIStream_fnFindSample(IAVIStream
*iface
, LONG pos
,
778 IAVIStreamImpl
*This
= (IAVIStreamImpl
*)iface
;
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
);
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
)
801 if (flags
& FIND_NEXT
)
806 } else if (flags
& FIND_ANY
) {
807 while (0 <= pos
&& pos
<= This
->lLastFrame
) {
808 if (This
->idxFrames
[pos
].dwChunkLength
> 0)
811 if (flags
& FIND_NEXT
)
817 } else if ((flags
& FIND_FORMAT
) && This
->idxFmtChanges
!= NULL
&&
818 This
->sInfo
.fccType
== streamtypeVIDEO
) {
819 if (flags
& FIND_NEXT
) {
822 for (n
= 0; n
< This
->sInfo
.dwFormatChangeCount
; n
++)
823 if (This
->idxFmtChanges
[n
].ckid
>= pos
) {
824 pos
= This
->idxFmtChanges
[n
].ckid
;
830 for (n
= (LONG
)This
->sInfo
.dwFormatChangeCount
; n
>= 0; n
--) {
831 if (This
->idxFmtChanges
[n
].ckid
<= pos
) {
832 pos
= This
->idxFmtChanges
[n
].ckid
;
837 if (pos
> (LONG
)This
->sInfo
.dwStart
)
838 return 0; /* format changes always for first frame */
846 if (pos
< (LONG
)This
->sInfo
.dwStart
)
849 switch (flags
& FIND_RET
) {
852 pos
= This
->idxFrames
[pos
].dwChunkLength
;
855 /* physical position */
856 pos
= This
->idxFrames
[pos
].dwChunkOffset
+ 2 * sizeof(DWORD
)
857 + offset
* This
->sInfo
.dwSampleSize
;
861 if (This
->sInfo
.dwSampleSize
)
862 pos
= This
->sInfo
.dwSampleSize
;
867 FIXME(": FIND_INDEX flag is not supported!\n");
868 /* This is an index in the index-table on disc. */
870 }; /* else logical position */
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
;
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
) {
904 lLastFmt
= IAVIStream_fnFindSample(iface
, pos
, FIND_FORMAT
|FIND_PREV
);
906 FIXME(": need to read formatchange for %d -- unimplemented!\n",lLastFmt
);
910 *formatsize
= This
->cbFormat
;
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
) {
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
) {
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
;
964 LPBITMAPINFOHEADER lpbiOld
= This
->lpFormat
;
965 RGBQUAD
*rgbNew
= (RGBQUAD
*)((LPBYTE
)lpbiNew
+ lpbiNew
->biSize
);
966 AVIPALCHANGE
*lppc
= NULL
;
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)
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
);
994 return AVIERR_MEMORY
;
996 lppc
->bFirstEntry
= 0;
997 lppc
->bNumEntries
= (lpbiOld
->biClrUsed
< 256 ? lpbiOld
->biClrUsed
: 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
,
1028 IAVIStreamImpl
*This
= (IAVIStreamImpl
*)iface
;
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
)
1039 if (samplesread
!= NULL
)
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
;
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 ... */
1070 if (This
->sInfo
.dwSampleSize
!= 0) {
1071 /* fixed samplesize -- we can read over frame/block boundaries */
1078 *bytesread
= samples
*This
->sInfo
.dwSampleSize
;
1080 *samplesread
= samples
;
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) {
1092 if (block
!= This
->dwCurrentFrame
) {
1093 hr
= AVIFILE_ReadBlock(This
, block
, NULL
, 0);
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
);
1106 buffer
= ((LPBYTE
)buffer
)+size
;
1110 /* fill out return parameters if given */
1111 if (bytesread
!= NULL
)
1113 if (samplesread
!= NULL
)
1114 *samplesread
+= size
/ This
->sInfo
.dwSampleSize
;
1120 return AVIERR_BUFFERTOOSMALL
;
1122 /* variable samplesize -- we can only read one full frame/block */
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
);
1132 } else if (buffer
!= NULL
)
1133 return AVIERR_BUFFERTOOSMALL
;
1135 /* fill out return parameters if given */
1136 if (bytesread
!= NULL
)
1138 if (samplesread
!= NULL
)
1139 *samplesread
= samples
;
1145 static HRESULT WINAPI
IAVIStream_fnWrite(IAVIStream
*iface
, LONG start
,
1146 LONG samples
, LPVOID buffer
,
1147 LONG buffersize
, DWORD flags
,
1149 LPLONG byteswritten
)
1151 IAVIStreamImpl
*This
= (IAVIStreamImpl
*)iface
;
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
)
1162 if (byteswritten
!= NULL
)
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
);
1178 if ((flags
& AVIIF_KEYFRAME
) && buffersize
!= 0)
1179 ckid
= MAKEAVICKID(cktypeDIBbits
, This
->nStream
);
1181 ckid
= MAKEAVICKID(cktypeDIBcompressed
, This
->nStream
);
1185 /* append to end of stream? */
1187 if (This
->lLastFrame
== -1)
1188 start
= This
->sInfo
.dwStart
;
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");
1210 /* variable sample size -- video like */
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);
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
;
1239 static HRESULT WINAPI
IAVIStream_fnDelete(IAVIStream
*iface
, LONG start
,
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
)
1254 /* Delete after end of stream? */
1255 if (start
> This
->sInfo
.dwLength
)
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.
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
;
1292 memcpy(lp
, This
->lpHandlerData
, min(This
->cbHandlerData
, *lpread
));
1293 if (*lpread
< This
->cbHandlerData
)
1294 return AVIERR_BUFFERTOOSMALL
;
1297 return AVIERR_NODATA
;
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 */
1311 return AVIERR_BADPARAM
;
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
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
);
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
);
1358 /***********************************************************************/
1360 static HRESULT
AVIFILE_AddFrame(IAVIStreamImpl
*This
, DWORD ckid
, DWORD size
, DWORD offset
, DWORD flags
)
1364 /* pre-conditions */
1365 assert(This
!= NULL
);
1367 switch (TWOCCFromFOURCC(ckid
)) {
1369 if (This
->paf
->fInfo
.dwFlags
& AVIFILEINFO_TRUSTCKTYPE
)
1370 flags
|= AVIIF_KEYFRAME
;
1372 case cktypeDIBcompressed
:
1373 if (This
->paf
->fInfo
.dwFlags
& AVIFILEINFO_TRUSTCKTYPE
)
1374 flags
&= ~AVIIF_KEYFRAME
;
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;
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
;
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
;
1407 case cktypeWAVEbytes
:
1408 if (This
->paf
->fInfo
.dwFlags
& AVIFILEINFO_TRUSTCKTYPE
)
1409 flags
|= AVIIF_KEYFRAME
;
1412 WARN(": unknown TWOCC 0x%04X found\n", TWOCCFromFOURCC(ckid
));
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
));
1429 This
->idxFrames
= HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, This
->idxFrames
,
1430 This
->nIdxFrames
* sizeof(AVIINDEXENTRY
));
1431 if (This
->idxFrames
== NULL
)
1432 return AVIERR_MEMORY
;
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;
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
++;
1473 static DWORD
AVIFILE_ComputeMoviStart(IAVIFileImpl
*This
)
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
);
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
;
1520 pstream
->nStream
= nr
;
1521 pstream
->dwCurrentFrame
= (DWORD
)-1;
1522 pstream
->lLastFrame
= -1;
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;
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
;
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
;
1584 if (This
->lpFormat
!= NULL
) {
1585 HeapFree(GetProcessHeap(), 0, This
->lpFormat
);
1586 This
->lpFormat
= NULL
;
1591 static HRESULT
AVIFILE_LoadFile(IAVIFileImpl
*This
)
1593 MainAVIHeader MainAVIHdr
;
1598 IAVIStreamImpl
*pStream
;
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
);
1621 /* get "avih" chunk */
1622 ck
.ckid
= ckidAVIMAINHDR
;
1623 hr
= FindChunkAndKeepExtras(&This
->fileextra
, This
->hmmio
, &ck
, &ckLIST1
, MMIO_FINDCHUNK
);
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
);
1681 while (mmioDescend(This
->hmmio
, &ck
, &ckLIST2
, 0) == S_OK
) {
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
;
1694 case ckidSTREAMFORMAT
:
1695 if (pStream
->lpFormat
!= NULL
)
1696 return AVIERR_BADFORMAT
;
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
);
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
;
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]));
1773 wsprintfW(szType
, streamTypeFmt
, (char*)&streamHdr
.fccType
);
1775 /* get count of this streamtype up to this stream */
1777 for (n
= nStream
; 0 <= n
; n
--) {
1778 if (This
->ppStreams
[n
]->sInfo
.fccHandler
== streamHdr
.fccType
)
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
);
1789 case ckidSTREAMNAME
:
1790 { /* streamname will be saved as ASCII string */
1791 LPSTR str
= HeapAlloc(GetProcessHeap(), 0, ck
.cksize
);
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
);
1807 case ckidAVIPADDING
:
1808 case mmioFOURCC('p','a','d','d'):
1811 WARN(": found extra chunk 0x%08X\n", ck
.ckid
);
1812 hr
= ReadChunkIntoExtra(&pStream
->extra
, This
->hmmio
, &ck
);
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
;
1828 /* nested chunks in "LIST","hdrl" which are not of type "LIST","strl" */
1829 hr
= ReadChunkIntoExtra(&This
->fileextra
, This
->hmmio
, &ckLIST2
);
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
,
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);
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);
1917 static HRESULT
AVIFILE_LoadIndex(const IAVIFileImpl
*This
, DWORD size
, DWORD offset
)
1921 HRESULT hr
= AVIERR_OK
;
1922 BOOL bAbsolute
= TRUE
;
1924 lp
= HeapAlloc(GetProcessHeap(), 0, IDX_PER_BLOCK
* sizeof(AVIINDEXENTRY
));
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
;
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
;
1961 LONG read
= min(IDX_PER_BLOCK
* sizeof(AVIINDEXENTRY
), size
);
1963 if (mmioRead(This
->hmmio
, (HPSTR
)lp
, read
) != read
) {
1964 hr
= AVIERR_FILEREAD
;
1969 if (pos
== (DWORD
)-1)
1970 pos
= offset
- lp
->dwChunkOffset
+ sizeof(DWORD
);
1972 AVIFILE_ParseIndex(This
, lp
, read
/ sizeof(AVIINDEXENTRY
),
1976 HeapFree(GetProcessHeap(), 0, lp
);
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
);
1991 static HRESULT
AVIFILE_ParseIndex(const IAVIFileImpl
*This
, AVIINDEXENTRY
*lp
,
1992 LONG count
, DWORD pos
, BOOL
*bAbsolute
)
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
)
2010 lp
->dwChunkOffset
+= sizeof(DWORD
);
2012 lp
->dwChunkOffset
+= pos
;
2014 if (FAILED(AVIFILE_AddFrame(This
->ppStreams
[nStream
], lp
->ckid
, lp
->dwChunkLength
, lp
->dwChunkOffset
, lp
->dwFlags
)))
2015 return AVIERR_MEMORY
;
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
;
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
;
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
;
2082 static void AVIFILE_SamplesToBlock(const IAVIStreamImpl
*This
, LPLONG pos
, LPLONG offset
)
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
;
2108 static HRESULT
AVIFILE_SaveFile(IAVIFileImpl
*This
)
2110 MainAVIHeader MainAVIHdr
;
2111 IAVIStreamImpl
* pStream
;
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
);
2154 ckRIFF
.fccType
= formtypeAVI
;
2155 if (mmioCreateChunk(This
->hmmio
, &ckRIFF
, MMIO_CREATERIFF
) != S_OK
)
2156 return AVIERR_FILEWRITE
;
2158 /* AVI headerlist */
2160 ckLIST1
.fccType
= listtypeAVIHEADER
;
2161 if (mmioCreateChunk(This
->hmmio
, &ckLIST1
, MMIO_CREATELIST
) != S_OK
)
2162 return AVIERR_FILEWRITE
;
2165 ck
.ckid
= ckidAVIMAINHDR
;
2166 ck
.cksize
= sizeof(MainAVIHdr
);
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 */
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) {
2251 ck
.ckid
= ckidSTREAMNAME
;
2252 ck
.cksize
= lstrlenW(pStream
->sInfo
.szName
) + 1;
2253 if (ck
.cksize
& 1) /* align */
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
);
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
);
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
);
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);
2334 static HRESULT
AVIFILE_SaveIndex(const IAVIFileImpl
*This
)
2336 IAVIStreamImpl
*pStream
;
2342 ck
.ckid
= ckidAVINEWINDEX
;
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;
2353 if (This
->ppStreams
[0]->sInfo
.dwSampleSize
== 0)
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
;
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
)
2386 if (pStream
->sInfo
.dwInitialFrames
< lInitialFrames
)
2387 nFrame
-= (lInitialFrames
- pStream
->sInfo
.dwInitialFrames
);
2389 /* reached end of this stream? */
2390 if (pStream
->lLastFrame
<= nFrame
)
2393 if ((pStream
->sInfo
.dwFlags
& AVISTREAMINFO_FORMATCHANGES
) &&
2394 pStream
->sInfo
.dwFormatChangeCount
!= 0 &&
2395 pStream
->idxFmtChanges
!= NULL
) {
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
;
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
;
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)) {
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
;
2438 pStream
->idxFmtChanges
[pos
].dwChunkOffset
- This
->dwMoviChunkPos
;
2439 if (mmioWrite(This
->hmmio
, (HPSTR
)&idx
, sizeof(idx
)) != sizeof(idx
))
2440 return AVIERR_FILEWRITE
;
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
;
2464 static ULONG
AVIFILE_SearchStream(const IAVIFileImpl
*This
, DWORD fcc
, LONG lSkip
)
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
) {
2492 static void AVIFILE_UpdateInfo(IAVIFileImpl
*This
)
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
;
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);
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
;
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
,
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
,
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
);