210 lines
5.8 KiB
C++
210 lines
5.8 KiB
C++
// stream.cpp
|
|
// Copyright (C) 1997, 1998 Microsoft Corporation. All Rights Reserved
|
|
//
|
|
// @doc EXTERNAL
|
|
|
|
#include <objbase.h>
|
|
#include <mmsystem.h>
|
|
#include <dsoundp.h>
|
|
#include "debug.h"
|
|
|
|
#include "dmusicc.h"
|
|
#include "dmusici.h"
|
|
#include "riff.h"
|
|
#include "dswave.h"
|
|
|
|
STDAPI AllocRIFFStream( IStream* pStream, IRIFFStream** ppRiff )
|
|
{
|
|
if( ( *ppRiff = (IRIFFStream*) new CRIFFStream( pStream ) ) == NULL )
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
/* MyRead, MyWrite, MySeek
|
|
*
|
|
* These are functionally identical to mmioRead, mmioWrite, and mmioSeek,
|
|
* except for the absence of the HMMIO parameter.
|
|
*/
|
|
|
|
long CRIFFStream::MyRead(void *pv, long cb)
|
|
{
|
|
ULONG cbRead;
|
|
if (FAILED(m_pStream->Read(pv, cb, &cbRead)))
|
|
return -1;
|
|
return cbRead;
|
|
}
|
|
|
|
long CRIFFStream::MyWrite(const void *pv, long cb)
|
|
{
|
|
ULONG cbWritten;
|
|
if (FAILED(m_pStream->Write(pv, cb, &cbWritten)))
|
|
return -1;
|
|
return cbWritten;
|
|
}
|
|
|
|
long CRIFFStream::MySeek(long lOffset, int iOrigin)
|
|
{
|
|
LARGE_INTEGER dlibSeekTo;
|
|
ULARGE_INTEGER dlibNewPos;
|
|
|
|
dlibSeekTo.HighPart = 0;
|
|
dlibSeekTo.LowPart = lOffset;
|
|
if (FAILED(m_pStream->Seek(dlibSeekTo, iOrigin, &dlibNewPos)))
|
|
return -1;
|
|
|
|
return dlibNewPos.LowPart;
|
|
}
|
|
|
|
|
|
UINT CRIFFStream::Descend(LPMMCKINFO lpck, LPMMCKINFO lpckParent, UINT wFlags)
|
|
{
|
|
FOURCC ckidFind; // chunk ID to find (or NULL)
|
|
FOURCC fccTypeFind; // form/list type to find (or NULL)
|
|
|
|
/* figure out what chunk id and form/list type to search for */
|
|
if (wFlags & MMIO_FINDCHUNK)
|
|
ckidFind = lpck->ckid, fccTypeFind = NULL;
|
|
else
|
|
if (wFlags & MMIO_FINDRIFF)
|
|
ckidFind = FOURCC_RIFF, fccTypeFind = lpck->fccType;
|
|
else
|
|
if (wFlags & MMIO_FINDLIST)
|
|
ckidFind = FOURCC_LIST, fccTypeFind = lpck->fccType;
|
|
else
|
|
ckidFind = fccTypeFind = NULL;
|
|
|
|
lpck->dwFlags = 0L;
|
|
|
|
for(;;)
|
|
{
|
|
UINT w;
|
|
|
|
/* read the chunk header */
|
|
if (MyRead(lpck, 2 * sizeof(DWORD)) !=
|
|
2 * sizeof(DWORD))
|
|
return MMIOERR_CHUNKNOTFOUND;
|
|
FixBytes( FBT_LONG, &lpck->cksize );
|
|
|
|
/* store the offset of the data part of the chunk */
|
|
if ((lpck->dwDataOffset = MySeek(0L, SEEK_CUR)) == -1)
|
|
return MMIOERR_CANNOTSEEK;
|
|
|
|
/* see if the chunk is within the parent chunk (if given) */
|
|
if ((lpckParent != NULL) &&
|
|
(lpck->dwDataOffset - 8L >=
|
|
lpckParent->dwDataOffset + lpckParent->cksize))
|
|
return MMIOERR_CHUNKNOTFOUND;
|
|
|
|
/* if the chunk if a 'RIFF' or 'LIST' chunk, read the
|
|
* form type or list type
|
|
*/
|
|
if ((lpck->ckid == FOURCC_RIFF) || (lpck->ckid == FOURCC_LIST))
|
|
{
|
|
if (MyRead(&lpck->fccType,
|
|
sizeof(DWORD)) != sizeof(DWORD))
|
|
return MMIOERR_CHUNKNOTFOUND;
|
|
}
|
|
else
|
|
lpck->fccType = NULL;
|
|
|
|
/* if this is the chunk we're looking for, stop looking */
|
|
if ( ((ckidFind == NULL) || (ckidFind == lpck->ckid)) &&
|
|
((fccTypeFind == NULL) || (fccTypeFind == lpck->fccType)) )
|
|
break;
|
|
|
|
/* ascend out of the chunk and try again */
|
|
if ((w = Ascend(lpck, 0)) != 0)
|
|
return w;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
UINT CRIFFStream::Ascend(LPMMCKINFO lpck, UINT /*wFlags*/)
|
|
{
|
|
if (lpck->dwFlags & MMIO_DIRTY)
|
|
{
|
|
/* <lpck> refers to a chunk created by CreateChunk();
|
|
* check that the chunk size that was written when
|
|
* CreateChunk() was called is the real chunk size;
|
|
* if not, fix it
|
|
*/
|
|
LONG lOffset; // current offset in file
|
|
LONG lActualSize; // actual size of chunk data
|
|
|
|
if ((lOffset = MySeek(0L, SEEK_CUR)) == -1)
|
|
return MMIOERR_CANNOTSEEK;
|
|
if ((lActualSize = lOffset - lpck->dwDataOffset) < 0)
|
|
return MMIOERR_CANNOTWRITE;
|
|
|
|
if (LOWORD(lActualSize) & 1)
|
|
{
|
|
/* chunk size is odd -- write a null pad byte */
|
|
if (MyWrite("\0", 1) != 1)
|
|
return MMIOERR_CANNOTWRITE;
|
|
|
|
}
|
|
|
|
if (lpck->cksize == (DWORD)lActualSize)
|
|
return 0;
|
|
|
|
/* fix the chunk header */
|
|
lpck->cksize = lActualSize;
|
|
if (MySeek(lpck->dwDataOffset - sizeof(DWORD), SEEK_SET) == -1)
|
|
return MMIOERR_CANNOTSEEK;
|
|
FixBytes( FBT_LONG, &lpck->cksize );
|
|
if (MyWrite(&lpck->cksize, sizeof(DWORD)) != sizeof(DWORD)) {
|
|
FixBytes( FBT_LONG, &lpck->cksize );
|
|
return MMIOERR_CANNOTWRITE;
|
|
}
|
|
FixBytes( FBT_LONG, &lpck->cksize );
|
|
}
|
|
|
|
/* seek to the end of the chunk, past the null pad byte
|
|
* (which is only there if chunk size is odd)
|
|
*/
|
|
if (MySeek(lpck->dwDataOffset + lpck->cksize + (lpck->cksize & 1L),
|
|
SEEK_SET) == -1)
|
|
return MMIOERR_CANNOTSEEK;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
UINT CRIFFStream::CreateChunk(LPMMCKINFO lpck, UINT wFlags)
|
|
{
|
|
int iBytes; // bytes to write
|
|
LONG lOffset; // current offset in file
|
|
|
|
/* store the offset of the data part of the chunk */
|
|
if ((lOffset = MySeek(0L, SEEK_CUR)) == -1)
|
|
return MMIOERR_CANNOTSEEK;
|
|
lpck->dwDataOffset = lOffset + 2 * sizeof(DWORD);
|
|
|
|
/* figure out if a form/list type needs to be written */
|
|
if (wFlags & MMIO_CREATERIFF)
|
|
lpck->ckid = FOURCC_RIFF, iBytes = 3 * sizeof(DWORD);
|
|
else
|
|
if (wFlags & MMIO_CREATELIST)
|
|
lpck->ckid = FOURCC_LIST, iBytes = 3 * sizeof(DWORD);
|
|
else
|
|
iBytes = 2 * sizeof(DWORD);
|
|
|
|
/* write the chunk header */
|
|
FixBytes( FBT_MMCKINFO, lpck );
|
|
if (MyWrite(lpck, (LONG) iBytes) != (LONG) iBytes) {
|
|
FixBytes( FBT_MMCKINFO, lpck );
|
|
return MMIOERR_CANNOTWRITE;
|
|
}
|
|
FixBytes( FBT_MMCKINFO, lpck );
|
|
|
|
lpck->dwFlags = MMIO_DIRTY;
|
|
|
|
return 0;
|
|
}
|
|
|