| /* |
| * Copyright 2003 Michael Günnewig |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include <assert.h> |
| #include <stdarg.h> |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "winuser.h" |
| #include "wingdi.h" |
| #include "winerror.h" |
| #include "mmsystem.h" |
| #include "vfw.h" |
| |
| #include "avifile_private.h" |
| #include "extrachunk.h" |
| |
| #include "wine/debug.h" |
| #include "initguid.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(avifile); |
| |
| /***********************************************************************/ |
| |
| /* internal interface to get access to table of stream in an editable stream */ |
| |
| DEFINE_AVIGUID(IID_IEditStreamInternal, 0x0002000A,0,0); |
| |
| typedef struct _EditStreamTable { |
| PAVISTREAM pStream; /* stream which contains the data */ |
| DWORD dwStart; /* where starts the part which is also our */ |
| DWORD dwLength; /* how many is also in this stream */ |
| } EditStreamTable; |
| |
| #define EditStreamEnd(This,streamNr) ((This)->pStreams[streamNr].dwStart + \ |
| (This)->pStreams[streamNr].dwLength) |
| |
| typedef struct _IAVIEditStreamImpl IAVIEditStreamImpl; |
| |
| struct _IAVIEditStreamImpl { |
| IAVIEditStream IAVIEditStream_iface; |
| IAVIStream IAVIStream_iface; |
| |
| LONG ref; |
| |
| AVISTREAMINFOW sInfo; |
| |
| EditStreamTable *pStreams; |
| DWORD nStreams; /* current fill level of pStreams table */ |
| DWORD nTableSize; /* size of pStreams table */ |
| |
| BOOL bDecompress; |
| PAVISTREAM pCurStream; |
| PGETFRAME pg; /* IGetFrame for pCurStream */ |
| LPBITMAPINFOHEADER lpFrame; /* frame of pCurStream */ |
| }; |
| |
| static inline IAVIEditStreamImpl *impl_from_IAVIEditStream(IAVIEditStream *iface) |
| { |
| return CONTAINING_RECORD(iface, IAVIEditStreamImpl, IAVIEditStream_iface); |
| } |
| |
| static inline IAVIEditStreamImpl *impl_from_IAVIStream(IAVIStream *iface) |
| { |
| return CONTAINING_RECORD(iface, IAVIEditStreamImpl, IAVIStream_iface); |
| } |
| |
| /***********************************************************************/ |
| |
| static HRESULT AVIFILE_FindStreamInTable(IAVIEditStreamImpl* const This, |
| DWORD pos,PAVISTREAM *ppStream, |
| DWORD* streamPos, |
| DWORD* streamNr,BOOL bFindSample) |
| { |
| DWORD n; |
| |
| TRACE("(%p,%u,%p,%p,%p,%d)\n",This,pos,ppStream,streamPos, |
| streamNr,bFindSample); |
| |
| if (pos < This->sInfo.dwStart) |
| return AVIERR_BADPARAM; |
| |
| pos -= This->sInfo.dwStart; |
| for (n = 0; n < This->nStreams; n++) { |
| if (pos < This->pStreams[n].dwLength) { |
| *ppStream = This->pStreams[n].pStream; |
| *streamPos = This->pStreams[n].dwStart + pos; |
| if (streamNr != NULL) |
| *streamNr = n; |
| |
| return AVIERR_OK; |
| } |
| pos -= This->pStreams[n].dwLength; |
| } |
| if (pos == 0 && bFindSample) { |
| *ppStream = This->pStreams[--n].pStream; |
| *streamPos = EditStreamEnd(This, n); |
| if (streamNr != NULL) |
| *streamNr = n; |
| |
| TRACE(" -- pos=0 && b=1 -> (%p,%u,%u)\n",*ppStream, *streamPos, n); |
| return AVIERR_OK; |
| } else { |
| *ppStream = NULL; |
| *streamPos = 0; |
| if (streamNr != NULL) |
| *streamNr = 0; |
| |
| TRACE(" -> ERROR (NULL,0,0)\n"); |
| return AVIERR_BADPARAM; |
| } |
| } |
| |
| static LPVOID AVIFILE_ReadFrame(IAVIEditStreamImpl* const This, |
| PAVISTREAM pstream, LONG pos) |
| { |
| PGETFRAME pg; |
| |
| TRACE("(%p,%p,%d)\n",This,pstream,pos); |
| |
| if (pstream == NULL) |
| return NULL; |
| |
| /* if stream changes make sure that only palette changes */ |
| if (This->pCurStream != pstream) { |
| pg = AVIStreamGetFrameOpen(pstream, NULL); |
| if (pg == NULL) |
| return NULL; |
| if (This->pg != NULL) { |
| if (IGetFrame_SetFormat(pg, This->lpFrame, NULL, 0, 0, -1, -1) != S_OK) { |
| AVIStreamGetFrameClose(pg); |
| ERR(": IGetFrame_SetFormat failed\n"); |
| return NULL; |
| } |
| AVIStreamGetFrameClose(This->pg); |
| } |
| This->pg = pg; |
| This->pCurStream = pstream; |
| } |
| |
| /* now get the decompressed frame */ |
| This->lpFrame = AVIStreamGetFrame(This->pg, pos); |
| if (This->lpFrame != NULL) |
| This->sInfo.dwSuggestedBufferSize = This->lpFrame->biSizeImage; |
| |
| return This->lpFrame; |
| } |
| |
| static HRESULT AVIFILE_RemoveStream(IAVIEditStreamImpl* const This, DWORD nr) |
| { |
| assert(This != NULL); |
| assert(nr < This->nStreams); |
| |
| /* remove part nr */ |
| IAVIStream_Release(This->pStreams[nr].pStream); |
| This->nStreams--; |
| if (This->nStreams - nr > 0) { |
| memmove(This->pStreams + nr, This->pStreams + nr + 1, |
| (This->nStreams - nr) * sizeof(EditStreamTable)); |
| } |
| This->pStreams[This->nStreams].pStream = NULL; |
| This->pStreams[This->nStreams].dwStart = 0; |
| This->pStreams[This->nStreams].dwLength = 0; |
| |
| /* try to merge the part before the deleted one and the one after it */ |
| if (0 < nr && 0 < This->nStreams && |
| This->pStreams[nr - 1].pStream == This->pStreams[nr].pStream) { |
| if (EditStreamEnd(This, nr - 1) == This->pStreams[nr].dwStart) { |
| This->pStreams[nr - 1].dwLength += This->pStreams[nr].dwLength; |
| return AVIFILE_RemoveStream(This, nr); |
| } |
| } |
| |
| return AVIERR_OK; |
| } |
| |
| static BOOL AVIFILE_FormatsEqual(PAVISTREAM avi1, PAVISTREAM avi2) |
| { |
| LPVOID fmt1 = NULL, fmt2 = NULL; |
| LONG size1, size2, start1, start2; |
| BOOL status = FALSE; |
| |
| assert(avi1 != NULL && avi2 != NULL); |
| |
| /* get stream starts and check format sizes */ |
| start1 = AVIStreamStart(avi1); |
| start2 = AVIStreamStart(avi2); |
| if (FAILED(AVIStreamFormatSize(avi1, start1, &size1))) |
| return FALSE; |
| if (FAILED(AVIStreamFormatSize(avi2, start2, &size2))) |
| return FALSE; |
| if (size1 != size2) |
| return FALSE; |
| |
| /* sizes match, now get formats and compare them */ |
| fmt1 = HeapAlloc(GetProcessHeap(), 0, size1); |
| if (fmt1 == NULL) |
| return FALSE; |
| if (SUCCEEDED(AVIStreamReadFormat(avi1, start1, fmt1, &size1))) { |
| fmt2 = HeapAlloc(GetProcessHeap(), 0, size1); |
| if (fmt2 != NULL) { |
| if (SUCCEEDED(AVIStreamReadFormat(avi2, start2, fmt2, &size1))) |
| status = (memcmp(fmt1, fmt2, size1) == 0); |
| } |
| } |
| |
| HeapFree(GetProcessHeap(), 0, fmt2); |
| HeapFree(GetProcessHeap(), 0, fmt1); |
| |
| return status; |
| } |
| |
| /***********************************************************************/ |
| |
| static HRESULT WINAPI IAVIEditStream_fnQueryInterface(IAVIEditStream*iface,REFIID refiid,LPVOID *obj) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIEditStream(iface); |
| |
| TRACE("(%p,%s,%p)\n", This, debugstr_guid(refiid), obj); |
| |
| if (IsEqualGUID(&IID_IUnknown, refiid) || |
| IsEqualGUID(&IID_IAVIEditStream, refiid) || |
| IsEqualGUID(&IID_IEditStreamInternal, refiid)) { |
| *obj = iface; |
| IAVIEditStream_AddRef(iface); |
| |
| return S_OK; |
| } else if (IsEqualGUID(&IID_IAVIStream, refiid)) { |
| *obj = &This->IAVIStream_iface; |
| IAVIEditStream_AddRef(iface); |
| |
| return S_OK; |
| } |
| |
| return E_NOINTERFACE; |
| } |
| |
| static ULONG WINAPI IAVIEditStream_fnAddRef(IAVIEditStream*iface) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIEditStream(iface); |
| ULONG ref = InterlockedIncrement(&This->ref); |
| |
| TRACE("(%p) -> %d\n", iface, ref); |
| |
| return ref; |
| } |
| |
| static ULONG WINAPI IAVIEditStream_fnRelease(IAVIEditStream*iface) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIEditStream(iface); |
| DWORD i; |
| ULONG ref = InterlockedDecrement(&This->ref); |
| |
| TRACE("(%p) -> %d\n", iface, ref); |
| |
| if (!ref) { |
| /* release memory */ |
| if (This->pg != NULL) |
| AVIStreamGetFrameClose(This->pg); |
| if (This->pStreams != NULL) { |
| for (i = 0; i < This->nStreams; i++) { |
| if (This->pStreams[i].pStream != NULL) |
| IAVIStream_Release(This->pStreams[i].pStream); |
| } |
| HeapFree(GetProcessHeap(), 0, This->pStreams); |
| } |
| |
| HeapFree(GetProcessHeap(), 0, This); |
| return 0; |
| } |
| return ref; |
| } |
| |
| static HRESULT WINAPI IAVIEditStream_fnCut(IAVIEditStream*iface,LONG*plStart, |
| LONG*plLength,PAVISTREAM*ppResult) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIEditStream(iface); |
| PAVISTREAM stream; |
| DWORD start, len, streamPos, streamNr; |
| HRESULT hr; |
| |
| TRACE("(%p,%p,%p,%p)\n",iface,plStart,plLength,ppResult); |
| |
| if (ppResult != NULL) |
| *ppResult = NULL; |
| if (plStart == NULL || plLength == NULL || *plStart < 0) |
| return AVIERR_BADPARAM; |
| |
| /* if asked for cut part copy it before deleting */ |
| if (ppResult != NULL) { |
| hr = IAVIEditStream_Copy(iface, plStart, plLength, ppResult); |
| if (FAILED(hr)) |
| return hr; |
| } |
| |
| start = *plStart; |
| len = *plLength; |
| |
| /* now delete the requested part */ |
| while (len > 0) { |
| hr = AVIFILE_FindStreamInTable(This, start, &stream, |
| &streamPos, &streamNr, FALSE); |
| if (FAILED(hr)) |
| return hr; |
| if (This->pStreams[streamNr].dwStart == streamPos) { |
| /* deleting from start of part */ |
| if (len < This->pStreams[streamNr].dwLength) { |
| start += len; |
| This->pStreams[streamNr].dwStart += len; |
| This->pStreams[streamNr].dwLength -= len; |
| This->sInfo.dwLength -= len; |
| len = 0; |
| |
| /* we must return decompressed data now */ |
| This->bDecompress = TRUE; |
| } else { |
| /* deleting hole part */ |
| len -= This->pStreams[streamNr].dwLength; |
| AVIFILE_RemoveStream(This,streamNr); |
| } |
| } else if (EditStreamEnd(This, streamNr) <= streamPos + len) { |
| /* deleting at end of a part */ |
| DWORD count = EditStreamEnd(This, streamNr) - streamPos; |
| This->sInfo.dwLength -= count; |
| len -= count; |
| This->pStreams[streamNr].dwLength = |
| streamPos - This->pStreams[streamNr].dwStart; |
| } else { |
| /* splitting */ |
| if (This->nStreams + 1 >= This->nTableSize) { |
| This->pStreams = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, This->pStreams, |
| (This->nTableSize + 32) * sizeof(EditStreamTable)); |
| if (This->pStreams == NULL) |
| return AVIERR_MEMORY; |
| This->nTableSize += 32; |
| } |
| memmove(This->pStreams + streamNr + 1, This->pStreams + streamNr, |
| (This->nStreams - streamNr) * sizeof(EditStreamTable)); |
| This->nStreams++; |
| |
| IAVIStream_AddRef(This->pStreams[streamNr + 1].pStream); |
| This->pStreams[streamNr + 1].dwStart = streamPos + len; |
| This->pStreams[streamNr + 1].dwLength = |
| EditStreamEnd(This, streamNr) - This->pStreams[streamNr + 1].dwStart; |
| |
| This->pStreams[streamNr].dwLength = |
| streamPos - This->pStreams[streamNr].dwStart; |
| This->sInfo.dwLength -= len; |
| len = 0; |
| } |
| } |
| |
| This->sInfo.dwEditCount++; |
| |
| return AVIERR_OK; |
| } |
| |
| static HRESULT WINAPI IAVIEditStream_fnCopy(IAVIEditStream*iface,LONG*plStart, |
| LONG*plLength,PAVISTREAM*ppResult) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIEditStream(iface); |
| IAVIEditStreamImpl* pEdit; |
| HRESULT hr; |
| LONG start = 0; |
| |
| TRACE("(%p,%p,%p,%p)\n",iface,plStart,plLength,ppResult); |
| |
| if (ppResult == NULL) |
| return AVIERR_BADPARAM; |
| *ppResult = NULL; |
| if (plStart == NULL || plLength == NULL || *plStart < 0 || *plLength < 0) |
| return AVIERR_BADPARAM; |
| |
| /* check bounds */ |
| if (*(LPDWORD)plLength > This->sInfo.dwLength) |
| *(LPDWORD)plLength = This->sInfo.dwLength; |
| if (*(LPDWORD)plStart < This->sInfo.dwStart) { |
| *(LPDWORD)plLength -= This->sInfo.dwStart - *(LPDWORD)plStart; |
| *(LPDWORD)plStart = This->sInfo.dwStart; |
| if (*plLength < 0) |
| return AVIERR_BADPARAM; |
| } |
| if (*(LPDWORD)plStart + *(LPDWORD)plLength > This->sInfo.dwStart + This->sInfo.dwLength) |
| *(LPDWORD)plLength = This->sInfo.dwStart + This->sInfo.dwLength - |
| *(LPDWORD)plStart; |
| |
| pEdit = (IAVIEditStreamImpl*)AVIFILE_CreateEditStream(NULL); |
| if (pEdit == NULL) |
| return AVIERR_MEMORY; |
| |
| hr = IAVIEditStream_Paste((PAVIEDITSTREAM)pEdit, &start, plLength, &This->IAVIStream_iface, |
| *plStart, *plStart + *plLength); |
| *plStart = start; |
| if (FAILED(hr)) |
| IAVIEditStream_Release((PAVIEDITSTREAM)pEdit); |
| else |
| *ppResult = &This->IAVIStream_iface; |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI IAVIEditStream_fnPaste(IAVIEditStream*iface,LONG*plStart, |
| LONG*plLength,PAVISTREAM pSource, |
| LONG lStart,LONG lLength) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIEditStream(iface); |
| AVISTREAMINFOW srcInfo; |
| IAVIEditStreamImpl *pEdit = NULL; |
| PAVISTREAM pStream; |
| DWORD startPos, endPos, streamNr, nStreams; |
| ULONG n; |
| |
| TRACE("(%p,%p,%p,%p,%d,%d)\n",iface,plStart,plLength, |
| pSource,lStart,lLength); |
| |
| if (pSource == NULL) |
| return AVIERR_BADHANDLE; |
| if (plStart == NULL || *plStart < 0) |
| return AVIERR_BADPARAM; |
| if (This->sInfo.dwStart + This->sInfo.dwLength < *plStart) |
| return AVIERR_BADPARAM; /* Can't paste with holes */ |
| if (FAILED(IAVIStream_Info(pSource, &srcInfo, sizeof(srcInfo)))) |
| return AVIERR_ERROR; |
| if (lStart < srcInfo.dwStart || lStart >= srcInfo.dwStart + srcInfo.dwLength) |
| return AVIERR_BADPARAM; |
| if (This->sInfo.fccType == 0) { |
| /* This stream is empty */ |
| IAVIStream_Info(pSource, &This->sInfo, sizeof(This->sInfo)); |
| This->sInfo.dwStart = *plStart; |
| This->sInfo.dwLength = 0; |
| } |
| if (This->sInfo.fccType != srcInfo.fccType) |
| return AVIERR_UNSUPPORTED; /* different stream types */ |
| if (lLength == -1) /* Copy the hole stream */ |
| lLength = srcInfo.dwLength; |
| if (lStart + lLength > srcInfo.dwStart + srcInfo.dwLength) |
| lLength = srcInfo.dwStart + srcInfo.dwLength - lStart; |
| if (lLength + *plStart >= 0x80000000) |
| return AVIERR_MEMORY; |
| |
| /* streamtype specific tests */ |
| if (srcInfo.fccType == streamtypeVIDEO) { |
| LONG size; |
| |
| size = srcInfo.rcFrame.right - srcInfo.rcFrame.left; |
| if (size != This->sInfo.rcFrame.right - This->sInfo.rcFrame.left) |
| return AVIERR_UNSUPPORTED; /* FIXME: Can't GetFrame convert it? */ |
| size = srcInfo.rcFrame.bottom - srcInfo.rcFrame.top; |
| if (size != This->sInfo.rcFrame.bottom - This->sInfo.rcFrame.top) |
| return AVIERR_UNSUPPORTED; /* FIXME: Can't GetFrame convert it? */ |
| } else if (srcInfo.fccType == streamtypeAUDIO) { |
| if (!AVIFILE_FormatsEqual(&This->IAVIStream_iface, pSource)) |
| return AVIERR_UNSUPPORTED; |
| } else { |
| /* FIXME: streamtypeMIDI and streamtypeTEXT */ |
| return AVIERR_UNSUPPORTED; |
| } |
| |
| /* try to get an IEditStreamInternal interface */ |
| if (SUCCEEDED(IAVIStream_QueryInterface(pSource, &IID_IEditStreamInternal, (LPVOID*)&pEdit))) |
| IAVIEditStream_Release(&pEdit->IAVIEditStream_iface); /* pSource holds a reference */ |
| |
| /* for video must check for change of format */ |
| if (This->sInfo.fccType == streamtypeVIDEO) { |
| if (! This->bDecompress) { |
| /* Need to decompress if any of the following conditions matches: |
| * - pSource is an editable stream which decompresses |
| * - the nearest keyframe of pSource isn't lStart |
| * - the nearest keyframe of this stream isn't *plStart |
| * - the format of pSource doesn't match this one |
| */ |
| if ((pEdit != NULL && pEdit->bDecompress) || |
| AVIStreamNearestKeyFrame(pSource, lStart) != lStart || |
| AVIStreamNearestKeyFrame(&This->IAVIStream_iface, *plStart) != *plStart || |
| (This->nStreams > 0 && !AVIFILE_FormatsEqual(&This->IAVIStream_iface, pSource))) { |
| /* Use first stream part to get format to convert everything to */ |
| AVIFILE_ReadFrame(This, This->pStreams[0].pStream, |
| This->pStreams[0].dwStart); |
| |
| /* Check if we could convert the source streams to the desired format... */ |
| if (pEdit != NULL) { |
| if (FAILED(AVIFILE_FindStreamInTable(pEdit, lStart, &pStream, |
| &startPos, &streamNr, TRUE))) |
| return AVIERR_INTERNAL; |
| for (n = lStart; n < lStart + lLength; streamNr++) { |
| if (AVIFILE_ReadFrame(This, pEdit->pStreams[streamNr].pStream, startPos) == NULL) |
| return AVIERR_BADFORMAT; |
| startPos = pEdit->pStreams[streamNr].dwStart; |
| n += pEdit->pStreams[streamNr].dwLength; |
| } |
| } else if (AVIFILE_ReadFrame(This, pSource, lStart) == NULL) |
| return AVIERR_BADFORMAT; |
| |
| This->bDecompress = TRUE; |
| This->sInfo.fccHandler = 0; |
| } |
| } else if (AVIFILE_ReadFrame(This, pSource, lStart) == NULL) |
| return AVIERR_BADFORMAT; /* Can't convert source to own format */ |
| } /* FIXME: something special for the other formats? */ |
| |
| /* Make sure we have enough memory for parts */ |
| if (pEdit != NULL) { |
| DWORD nLastStream; |
| |
| AVIFILE_FindStreamInTable(pEdit, lStart + lLength, &pStream, |
| &endPos, &nLastStream, TRUE); |
| AVIFILE_FindStreamInTable(pEdit, lStart, &pStream, |
| &startPos, &streamNr, FALSE); |
| if (nLastStream == streamNr) |
| nLastStream++; |
| |
| nStreams = nLastStream - streamNr; |
| } else |
| nStreams = 1; |
| if (This->nStreams + nStreams + 1 > This->nTableSize) { |
| n = This->nStreams + nStreams + 33; |
| |
| This->pStreams = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, This->pStreams, n * sizeof(EditStreamTable)); |
| if (This->pStreams == NULL) |
| return AVIERR_MEMORY; |
| This->nTableSize = n; |
| } |
| |
| if (plLength != NULL) |
| *plLength = lLength; |
| |
| /* now do the real work */ |
| if (This->sInfo.dwStart + This->sInfo.dwLength > *plStart) { |
| AVIFILE_FindStreamInTable(This, *plStart, &pStream, |
| &startPos, &streamNr, FALSE); |
| if (startPos != This->pStreams[streamNr].dwStart) { |
| /* split stream streamNr at startPos */ |
| memmove(This->pStreams + streamNr + nStreams + 1, |
| This->pStreams + streamNr, |
| (This->nStreams + nStreams - streamNr + 1) * sizeof(EditStreamTable)); |
| |
| This->pStreams[streamNr + 2].dwLength = |
| EditStreamEnd(This, streamNr + 2) - startPos; |
| This->pStreams[streamNr + 2].dwStart = startPos; |
| This->pStreams[streamNr].dwLength = |
| startPos - This->pStreams[streamNr].dwStart; |
| IAVIStream_AddRef(This->pStreams[streamNr].pStream); |
| streamNr++; |
| } else { |
| /* insert before stream at streamNr */ |
| memmove(This->pStreams + streamNr + nStreams, This->pStreams + streamNr, |
| (This->nStreams + nStreams - streamNr) * sizeof(EditStreamTable)); |
| } |
| } else /* append the streams */ |
| streamNr = This->nStreams; |
| |
| if (pEdit != NULL) { |
| /* insert the parts of the editable stream instead of itself */ |
| AVIFILE_FindStreamInTable(pEdit, lStart + lLength, &pStream, |
| &endPos, NULL, FALSE); |
| AVIFILE_FindStreamInTable(pEdit, lStart, &pStream, &startPos, &n, FALSE); |
| |
| memcpy(This->pStreams + streamNr, pEdit->pStreams + n, |
| nStreams * sizeof(EditStreamTable)); |
| if (This->pStreams[streamNr].dwStart < startPos) { |
| This->pStreams[streamNr].dwLength = |
| EditStreamEnd(This, streamNr) - startPos; |
| This->pStreams[streamNr].dwStart = startPos; |
| } |
| if (endPos < EditStreamEnd(This, streamNr + nStreams)) |
| This->pStreams[streamNr + nStreams].dwLength = |
| endPos - This->pStreams[streamNr + nStreams].dwStart; |
| } else { |
| /* a simple stream */ |
| This->pStreams[streamNr].pStream = pSource; |
| This->pStreams[streamNr].dwStart = lStart; |
| This->pStreams[streamNr].dwLength = lLength; |
| } |
| |
| for (n = 0; n < nStreams; n++) { |
| IAVIStream_AddRef(This->pStreams[streamNr + n].pStream); |
| if (0 < streamNr + n && |
| This->pStreams[streamNr + n - 1].pStream != This->pStreams[streamNr + n].pStream) { |
| This->sInfo.dwFlags |= AVISTREAMINFO_FORMATCHANGES; |
| This->sInfo.dwFormatChangeCount++; |
| } |
| } |
| This->sInfo.dwEditCount++; |
| This->sInfo.dwLength += lLength; |
| This->nStreams += nStreams; |
| |
| return AVIERR_OK; |
| } |
| |
| static HRESULT WINAPI IAVIEditStream_fnClone(IAVIEditStream*iface, |
| PAVISTREAM*ppResult) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIEditStream(iface); |
| IAVIEditStreamImpl* pEdit; |
| DWORD i; |
| |
| TRACE("(%p,%p)\n",iface,ppResult); |
| |
| if (ppResult == NULL) |
| return AVIERR_BADPARAM; |
| *ppResult = NULL; |
| |
| pEdit = (IAVIEditStreamImpl*)AVIFILE_CreateEditStream(NULL); |
| if (pEdit == NULL) |
| return AVIERR_MEMORY; |
| if (This->nStreams > pEdit->nTableSize) { |
| pEdit->pStreams = HeapReAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, pEdit->pStreams, |
| This->nStreams * sizeof(EditStreamTable)); |
| if (pEdit->pStreams == NULL) |
| return AVIERR_MEMORY; |
| pEdit->nTableSize = This->nStreams; |
| } |
| pEdit->nStreams = This->nStreams; |
| memcpy(pEdit->pStreams, This->pStreams, |
| This->nStreams * sizeof(EditStreamTable)); |
| memcpy(&pEdit->sInfo,&This->sInfo,sizeof(This->sInfo)); |
| for (i = 0; i < This->nStreams; i++) { |
| if (pEdit->pStreams[i].pStream != NULL) |
| IAVIStream_AddRef(pEdit->pStreams[i].pStream); |
| } |
| |
| *ppResult = &This->IAVIStream_iface; |
| |
| return AVIERR_OK; |
| } |
| |
| static HRESULT WINAPI IAVIEditStream_fnSetInfo(IAVIEditStream*iface, |
| LPAVISTREAMINFOW asi,LONG size) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIEditStream(iface); |
| |
| TRACE("(%p,%p,%d)\n",iface,asi,size); |
| |
| /* check parameters */ |
| if (size >= 0 && size < sizeof(AVISTREAMINFOW)) |
| return AVIERR_BADSIZE; |
| |
| This->sInfo.wLanguage = asi->wLanguage; |
| This->sInfo.wPriority = asi->wPriority; |
| This->sInfo.dwStart = asi->dwStart; |
| This->sInfo.dwRate = asi->dwRate; |
| This->sInfo.dwScale = asi->dwScale; |
| This->sInfo.dwQuality = asi->dwQuality; |
| CopyRect(&This->sInfo.rcFrame, &asi->rcFrame); |
| memcpy(This->sInfo.szName, asi->szName, sizeof(asi->szName)); |
| This->sInfo.dwEditCount++; |
| |
| return AVIERR_OK; |
| } |
| |
| static const struct IAVIEditStreamVtbl ieditstream = { |
| IAVIEditStream_fnQueryInterface, |
| IAVIEditStream_fnAddRef, |
| IAVIEditStream_fnRelease, |
| IAVIEditStream_fnCut, |
| IAVIEditStream_fnCopy, |
| IAVIEditStream_fnPaste, |
| IAVIEditStream_fnClone, |
| IAVIEditStream_fnSetInfo |
| }; |
| |
| static HRESULT WINAPI IEditAVIStream_fnQueryInterface(IAVIStream*iface, |
| REFIID refiid,LPVOID*obj) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| return IAVIEditStream_QueryInterface(&This->IAVIEditStream_iface,refiid,obj); |
| } |
| |
| static ULONG WINAPI IEditAVIStream_fnAddRef(IAVIStream*iface) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| return IAVIEditStream_AddRef(&This->IAVIEditStream_iface); |
| } |
| |
| static ULONG WINAPI IEditAVIStream_fnRelease(IAVIStream*iface) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| return IAVIEditStream_Release(&This->IAVIEditStream_iface); |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnCreate(IAVIStream*iface, |
| LPARAM lParam1,LPARAM lParam2) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| |
| if (lParam2 != 0) |
| return AVIERR_ERROR; |
| |
| if (This->pStreams == NULL) { |
| This->pStreams = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, 256 * sizeof(EditStreamTable)); |
| if (This->pStreams == NULL) |
| return AVIERR_MEMORY; |
| This->nTableSize = 256; |
| } |
| |
| if (lParam1 != 0) { |
| IAVIStream_Info((PAVISTREAM)lParam1, &This->sInfo, sizeof(This->sInfo)); |
| IAVIStream_AddRef((PAVISTREAM)lParam1); |
| This->pStreams[0].pStream = (PAVISTREAM)lParam1; |
| This->pStreams[0].dwStart = This->sInfo.dwStart; |
| This->pStreams[0].dwLength = This->sInfo.dwLength; |
| This->nStreams = 1; |
| } |
| return AVIERR_OK; |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnInfo(IAVIStream*iface, |
| AVISTREAMINFOW *psi,LONG size) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| |
| TRACE("(%p,%p,%d)\n",iface,psi,size); |
| |
| if (psi == NULL) |
| return AVIERR_BADPARAM; |
| if (size < 0) |
| return AVIERR_BADSIZE; |
| |
| if (This->bDecompress) |
| This->sInfo.fccHandler = 0; |
| |
| memcpy(psi, &This->sInfo, min((DWORD)size, sizeof(This->sInfo))); |
| |
| if ((DWORD)size < sizeof(This->sInfo)) |
| return AVIERR_BUFFERTOOSMALL; |
| return AVIERR_OK; |
| } |
| |
| static LONG WINAPI IEditAVIStream_fnFindSample(IAVIStream*iface,LONG pos, |
| LONG flags) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| PAVISTREAM stream; |
| DWORD streamPos, streamNr; |
| |
| TRACE("(%p,%d,0x%08X)\n",iface,pos,flags); |
| |
| if (flags & FIND_FROM_START) |
| pos = (LONG)This->sInfo.dwStart; |
| |
| /* outside of stream? */ |
| if (pos < (LONG)This->sInfo.dwStart || |
| (LONG)This->sInfo.dwStart + (LONG)This->sInfo.dwLength <= pos) |
| return -1; |
| |
| /* map our position to a stream and position in it */ |
| if (AVIFILE_FindStreamInTable(This, pos, &stream, &streamPos, |
| &streamNr, TRUE) != S_OK) |
| return -1; /* doesn't exist */ |
| |
| if (This->bDecompress) { |
| /* only one stream -- format changes only at start */ |
| if (flags & FIND_FORMAT) |
| return (flags & FIND_NEXT ? -1 : 0); |
| |
| /* FIXME: map positions back to us */ |
| return IAVIStream_FindSample(stream, streamPos, flags); |
| } else { |
| /* assume change of format every frame */ |
| return pos; |
| } |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnReadFormat(IAVIStream*iface,LONG pos, |
| LPVOID format,LONG*fmtsize) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| LPBITMAPINFOHEADER lp; |
| PAVISTREAM stream; |
| DWORD n; |
| HRESULT hr; |
| |
| TRACE("(%p,%d,%p,%p)\n",iface,pos,format,fmtsize); |
| |
| if (fmtsize == NULL || pos < This->sInfo.dwStart || |
| This->sInfo.dwStart + This->sInfo.dwLength <= pos) |
| return AVIERR_BADPARAM; |
| |
| /* find stream corresponding to position */ |
| hr = AVIFILE_FindStreamInTable(This, pos, &stream, &n, NULL, FALSE); |
| if (FAILED(hr)) |
| return hr; |
| |
| if (! This->bDecompress) |
| return IAVIStream_ReadFormat(stream, n, format, fmtsize); |
| |
| lp = AVIFILE_ReadFrame(This, stream, n); |
| if (lp == NULL) |
| return AVIERR_ERROR; |
| if (lp->biBitCount <= 8) { |
| n = (lp->biClrUsed > 0 ? lp->biClrUsed : 1 << lp->biBitCount); |
| n *= sizeof(RGBQUAD); |
| } else |
| n = 0; |
| n += lp->biSize; |
| |
| memcpy(format, lp, min((LONG)n, *fmtsize)); |
| hr = ((LONG)n > *fmtsize ? AVIERR_BUFFERTOOSMALL : AVIERR_OK); |
| *fmtsize = n; |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnSetFormat(IAVIStream*iface,LONG pos, |
| LPVOID format,LONG formatsize) |
| { |
| TRACE("(%p,%d,%p,%d)\n",iface,pos,format,formatsize); |
| |
| return AVIERR_UNSUPPORTED; |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnRead(IAVIStream*iface,LONG start, |
| LONG samples,LPVOID buffer, |
| LONG buffersize,LONG*bytesread, |
| LONG*samplesread) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| PAVISTREAM stream; |
| DWORD streamPos, streamNr; |
| LONG readBytes, readSamples, count; |
| HRESULT hr; |
| |
| TRACE("(%p,%d,%d,%p,%d,%p,%p) -- 0x%08X\n",iface,start,samples, |
| buffer,buffersize,bytesread,samplesread,This->sInfo.fccType); |
| |
| /* check parameters */ |
| if (bytesread != NULL) |
| *bytesread = 0; |
| if (samplesread != NULL) |
| *samplesread = 0; |
| if (buffersize < 0) |
| return AVIERR_BADSIZE; |
| if ((DWORD)start < This->sInfo.dwStart || |
| This->sInfo.dwStart + This->sInfo.dwLength < (DWORD)start) |
| return AVIERR_BADPARAM; |
| |
| if (! This->bDecompress) { |
| /* audio like data -- sample-based */ |
| do { |
| if (samples == 0) |
| return AVIERR_OK; /* nothing at all or already done */ |
| |
| if (FAILED(AVIFILE_FindStreamInTable(This, start, &stream, |
| &streamPos, &streamNr, FALSE))) |
| return AVIERR_ERROR; |
| |
| /* limit to end of the stream */ |
| count = samples; |
| if (streamPos + count > EditStreamEnd(This, streamNr)) |
| count = EditStreamEnd(This, streamNr) - streamPos; |
| |
| hr = IAVIStream_Read(stream, streamPos, count, buffer, buffersize, |
| &readBytes, &readSamples); |
| if (FAILED(hr)) |
| return hr; |
| if (readBytes == 0 && readSamples == 0 && count != 0) |
| return AVIERR_FILEREAD; /* for bad stream implementations */ |
| |
| if (samplesread != NULL) |
| *samplesread += readSamples; |
| if (bytesread != NULL) |
| *bytesread += readBytes; |
| if (buffer != NULL) { |
| buffer = ((LPBYTE)buffer)+readBytes; |
| buffersize -= readBytes; |
| } |
| start += count; |
| samples -= count; |
| } while (This->sInfo.dwStart + This->sInfo.dwLength > start); |
| } else { |
| /* video like data -- frame-based */ |
| LPBITMAPINFOHEADER lp; |
| |
| if (samples == 0) |
| return AVIERR_OK; |
| |
| if (FAILED(AVIFILE_FindStreamInTable(This, start, &stream, |
| &streamPos, &streamNr, FALSE))) |
| return AVIERR_ERROR; |
| |
| lp = AVIFILE_ReadFrame(This, stream, streamPos); |
| if (lp == NULL) |
| return AVIERR_ERROR; |
| |
| if (buffer != NULL) { |
| /* need size of format to skip */ |
| if (lp->biBitCount <= 8) { |
| count = lp->biClrUsed > 0 ? lp->biClrUsed : 1 << lp->biBitCount; |
| count *= sizeof(RGBQUAD); |
| } else |
| count = 0; |
| count += lp->biSize; |
| |
| if (buffersize < lp->biSizeImage) |
| return AVIERR_BUFFERTOOSMALL; |
| memcpy(buffer, (LPBYTE)lp + count, lp->biSizeImage); |
| } |
| |
| if (bytesread != NULL) |
| *bytesread = lp->biSizeImage; |
| if (samplesread != NULL) |
| *samplesread = 1; |
| } |
| |
| return AVIERR_OK; |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnWrite(IAVIStream*iface,LONG start, |
| LONG samples,LPVOID buffer, |
| LONG buffersize,DWORD flags, |
| LONG*sampwritten,LONG*byteswritten) |
| { |
| TRACE("(%p,%d,%d,%p,%d,0x%08X,%p,%p)\n",iface,start,samples,buffer, |
| buffersize,flags,sampwritten,byteswritten); |
| |
| /* be sure return parameters have correct values */ |
| if (sampwritten != NULL) |
| *sampwritten = 0; |
| if (byteswritten != NULL) |
| *byteswritten = 0; |
| |
| return AVIERR_UNSUPPORTED; |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnDelete(IAVIStream*iface,LONG start, |
| LONG samples) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| |
| TRACE("(%p,%d,%d)\n",iface,start,samples); |
| |
| return IAVIEditStream_Cut(&This->IAVIEditStream_iface,&start,&samples,NULL); |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnReadData(IAVIStream*iface,DWORD fcc, |
| LPVOID lp,LONG *lpread) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| DWORD n; |
| |
| TRACE("(%p,0x%08X,%p,%p)\n",iface,fcc,lp,lpread); |
| |
| /* check parameters */ |
| if (lp == NULL || lpread == NULL) |
| return AVIERR_BADPARAM; |
| |
| /* simply ask every stream and return the first block found */ |
| for (n = 0; n < This->nStreams; n++) { |
| HRESULT hr = IAVIStream_ReadData(This->pStreams[n].pStream,fcc,lp,lpread); |
| |
| if (SUCCEEDED(hr)) |
| return hr; |
| } |
| |
| *lpread = 0; |
| return AVIERR_NODATA; |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnWriteData(IAVIStream*iface,DWORD fcc, |
| LPVOID lp,LONG size) |
| { |
| TRACE("(%p,0x%08X,%p,%d)\n",iface,fcc,lp,size); |
| |
| return AVIERR_UNSUPPORTED; |
| } |
| |
| static HRESULT WINAPI IEditAVIStream_fnSetInfo(IAVIStream*iface, |
| AVISTREAMINFOW*info,LONG len) |
| { |
| IAVIEditStreamImpl *This = impl_from_IAVIStream( iface ); |
| |
| TRACE("(%p,%p,%d)\n",iface,info,len); |
| |
| return IAVIEditStream_SetInfo(&This->IAVIEditStream_iface,info,len); |
| } |
| |
| static const struct IAVIStreamVtbl ieditstast = { |
| IEditAVIStream_fnQueryInterface, |
| IEditAVIStream_fnAddRef, |
| IEditAVIStream_fnRelease, |
| IEditAVIStream_fnCreate, |
| IEditAVIStream_fnInfo, |
| IEditAVIStream_fnFindSample, |
| IEditAVIStream_fnReadFormat, |
| IEditAVIStream_fnSetFormat, |
| IEditAVIStream_fnRead, |
| IEditAVIStream_fnWrite, |
| IEditAVIStream_fnDelete, |
| IEditAVIStream_fnReadData, |
| IEditAVIStream_fnWriteData, |
| IEditAVIStream_fnSetInfo |
| }; |
| |
| PAVIEDITSTREAM AVIFILE_CreateEditStream(PAVISTREAM pstream) |
| { |
| IAVIEditStreamImpl *pedit = NULL; |
| |
| pedit = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IAVIEditStreamImpl)); |
| if (pedit == NULL) |
| return NULL; |
| |
| pedit->IAVIEditStream_iface.lpVtbl = &ieditstream; |
| pedit->IAVIStream_iface.lpVtbl = &ieditstast; |
| pedit->ref = 1; |
| |
| IAVIStream_Create(&pedit->IAVIStream_iface, (LPARAM)pstream, 0); |
| |
| return (PAVIEDITSTREAM)pedit; |
| } |