NT4/private/windows/media/avi/avicap.io/capavi.c
2020-09-30 17:12:29 +02:00

2664 lines
86 KiB
C

/****************************************************************************
*
* capavi.c
*
* Main video capture module.
*
* Microsoft Video for Windows Sample Capture Class
*
* Copyright (c) 1992 - 1995 Microsoft Corporation. All Rights Reserved.
*
* You have a royalty-free right to use, modify, reproduce and
* distribute the Sample Files (and/or any modified version) in
* any way you find useful, provided that you agree that
* Microsoft has no warranty obligations or liability for any
* Sample Application Files which are modified.
*
***************************************************************************/
#define INC_OLE2
#pragma warning(disable:4103)
#include <windows.h>
#include <windowsx.h>
#include <win32.h>
#include <mmsystem.h>
#include <msvideo.h>
#include <drawdib.h>
#include <mmreg.h>
#include <mmddk.h>
#include "ivideo32.h"
#include "mmdebug.h"
#ifdef USE_ACM
#include <msacm.h>
#endif
#include <avifmt.h>
#include "avicap.h"
#include "avicapi.h"
#include "time.h"
#define JMK_HACK_TIMERS TRUE
#ifdef JMK_HACK_TIMERS
#define _INC_MMTIMERS_CODE_ TRUE
#define CLIPBOARDLOGSIZE 1000
#ifndef MAKEFOURCC
#define MAKEFOURCC(a,b,c,d) ((DWORD)(a) | ((DWORD)(b) << 8) | ((DWORD)(c) << 16) | ((DWORD)(d) << 24))
#endif
#define RIFFTYPE(dw) (((dw & 0xFF) << 24) | ((dw & 0xFF00) << 8) | ((dw & 0xFF0000) >> 8) | ((dw & 0xFF000000) >> 24))
#include "mmtimers.h"
typedef struct _timerstuff {
DWORD dwFrameTickTime; // What we think the current frame time should be
DWORD dwFrameStampTime; // Stamped in the VIDEOHDR
DWORD dwTimeWritten; // Time WriteFile called
DWORD dwTimeToWrite; // Time WriteFile returned
WORD nFramesAppended; // Accumulated appended dropped frames
WORD nDummyFrames; // frames calc'ed as dropped
DWORD dwVideoChunkCount; // current 'frame'
WORD nAudioIndex; // next audio buffer
WORD nVideoIndex; // next video buffer
WORD bPending;
WORD nSleepCount;
DWORD dwSleepBegin;
DWORD dwSleepEnd;
};
STATICDT PCTIMER pctWriteBase;
STATICDT struct _timerstuff * pCurTimerStuff;
STATICDT struct _timerstuff * pTimerStuff;
STATICDT HGLOBAL hMemTimers;
STATICDT struct _timerriff {
FOURCC fccRIFF; // 'RIFF'
DWORD cbTotal; // total (inclusive) size of riff data
FOURCC fccJMKD; // 'JMKD' data type identifier
DWORD fccVCHD; // 'VCHD' capture data header
DWORD cbVCHD; // sizeof vchd data
struct _vchd {
DWORD nPrio;
DWORD dwFramesCaptured;
DWORD dwFramesDropped;
DWORD dwDropFramesAppended;
DWORD dwDropFramesNotAppended;
DWORD dwTimerFrequency;
DWORD dwSpare[2];
CAPTUREPARMS cap;
BITMAPINFOHEADER bmih;
DWORD nMaxVideoBuffers;
struct _thkvideohdr {
VIDEOHDR vh;
LPBYTE p32Buff;
DWORD p16Alloc;
DWORD dwMemHandle;
DWORD dwReserved;
} atvh[64];
} vchd;
DWORD fccChunk; // chunk data type tag
DWORD cbChunk; // non-inclusive size of chunk data
} * pTimerRiff;
STATICDT UINT nTimerIndex;
STATICDT UINT nSleepCount;
#endif
#ifdef _DEBUG
#define DSTATUS(lpcs, sz) statusUpdateStatus(lpcs, IDS_CAP_INFO, (LPTSTR) TEXT(sz))
#else
#define DSTATUS(lpcs, sz)
#endif
// Allocate memory on a sector boundary
LPVOID FAR PASCAL AllocSectorAlignedMem (DWORD dwRequest, DWORD dwAlign)
{
LPVOID pbuf;
dwRequest = ROUNDUPTOSECTORSIZE (dwRequest, dwAlign) + dwAlign; // round up to next page boundary
pbuf = VirtualAlloc (NULL, dwRequest,
MEM_COMMIT | MEM_RESERVE,
PAGE_READWRITE);
AuxDebugEx(4, DEBUGLINE "Allocated %d bytes of sector aligned memory at %8x\r\n", dwRequest, pbuf);
return pbuf;
}
void FAR PASCAL FreeSectorAlignedMem (LPVOID pbuf)
{
// the pointer we free had better be aligned on at least a 256 byte
// boundary
//
assert (!((DWORD)pbuf & 255));
VirtualFree ((LPVOID)((DWORD)pbuf & ~255), 0, MEM_RELEASE);
}
#define ONEMEG (1024L * 1024L)
DWORD GetFreePhysicalMemory(void)
{
MEMORYSTATUS ms;
ms.dwLength = sizeof(ms);
GlobalMemoryStatus(&ms);
if (ms.dwTotalPhys > 8L * ONEMEG)
return ms.dwTotalPhys - ONEMEG * 4;
return(ms.dwTotalPhys /2);
}
// ****************************************************************
// ******************** Audio Buffer Control **********************
// ****************************************************************
// Audio buffers are always allocated under the presumption that
// audio capture may be enabled at any time.
// AVIAudioInit must be matched with AVIAudioFini (both only called once)
// AVIAudioPrepare must be matched with AVIAudioUnPrepare
// (which may be called multiple times to enable and disable audio)
// AVI AudioInit - Allocate and initialize buffers for audio capture.
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIAudioInit (LPCAPSTREAM lpcs)
{
int i;
LPVOID pHdr;
LPVOID p;
if (lpcs->sCapParms.wNumAudioRequested == 0)
lpcs->sCapParms.wNumAudioRequested = DEF_WAVE_BUFFERS;
// .5 second of audio per buffer (or 10K, whichever is larger)
if (lpcs->sCapParms.dwAudioBufferSize == 0)
lpcs->dwWaveSize = CalcWaveBufferSize (lpcs);
else {
lpcs->dwWaveSize = 0;
if (lpcs->lpWaveFormat)
lpcs->dwWaveSize = lpcs->sCapParms.dwAudioBufferSize;
}
// Alloc the wave memory
for(i = 0; i < (int)lpcs->sCapParms.wNumAudioRequested; i++) {
pHdr = GlobalAllocPtr(GPTR, sizeof(WAVEHDR));
if (pHdr == NULL)
break;
lpcs->alpWaveHdr[i] = pHdr;
p = AllocSectorAlignedMem( sizeof(RIFF) + lpcs->dwWaveSize, lpcs->dwBytesPerSector);
if (p == NULL) {
GlobalFreePtr (pHdr);
lpcs->alpWaveHdr[i] = NULL;
break;
}
lpcs->alpWaveHdr[i]->lpData = (LPBYTE)p + sizeof(RIFF);
lpcs->alpWaveHdr[i]->dwBufferLength = lpcs->dwWaveSize;
lpcs->alpWaveHdr[i]->dwBytesRecorded = 0;
lpcs->alpWaveHdr[i]->dwUser = 0;
lpcs->alpWaveHdr[i]->dwFlags = 0;
lpcs->alpWaveHdr[i]->dwLoops = 0;
((LPRIFF)p)->dwType = MAKEAVICKID(cktypeWAVEbytes, 1);
((LPRIFF)p)->dwSize = lpcs->dwWaveSize;
}
lpcs->iNumAudio = i;
return ((lpcs->iNumAudio == 0) ? IDS_CAP_WAVE_ALLOC_ERROR : 0);
}
//
// AVI AudioFini - UnPrepares headers
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIAudioFini (LPCAPSTREAM lpcs)
{
int ii;
/* free headers and data */
for (ii=0; ii < MAX_WAVE_BUFFERS; ++ii) {
if (lpcs->alpWaveHdr[ii]) {
if (lpcs->alpWaveHdr[ii]->lpData)
FreeSectorAlignedMem((LPBYTE)lpcs->alpWaveHdr[ii]->lpData - sizeof (RIFF));
GlobalFreePtr(lpcs->alpWaveHdr[ii]);
lpcs->alpWaveHdr[ii] = NULL;
}
}
return 0;
}
//
// AVI AudioPrepare - Opens the wave device and adds the buffers
// Prepares headers and adds buffers to the device
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIAudioPrepare (LPCAPSTREAM lpcs)
{
UINT uiError;
int ii;
/* See if we can open that format for input */
// register event callback to avoid polling
uiError = waveInOpen(&lpcs->hWaveIn,
WAVE_MAPPER, lpcs->lpWaveFormat,
(DWORD) lpcs->hCaptureEvent, 0, CALLBACK_EVENT );
if (uiError != MMSYSERR_NOERROR)
return IDS_CAP_WAVE_OPEN_ERROR;
lpcs->fAudioYield = FALSE; // ACM is separate thread, don't yield
lpcs->fAudioBreak = FALSE;
DPF("AudioYield = %d", lpcs->fAudioYield);
for (ii = 0; ii < (int)lpcs->sCapParms.wNumAudioRequested; ++ii) {
if (waveInPrepareHeader (lpcs->hWaveIn, lpcs->alpWaveHdr[ii],
sizeof(WAVEHDR)))
return IDS_CAP_WAVE_ALLOC_ERROR;
if (waveInAddBuffer (lpcs->hWaveIn, lpcs->alpWaveHdr[ii],
sizeof(WAVEHDR)))
return IDS_CAP_WAVE_ALLOC_ERROR;
AuxDebugEx(3, DEBUGLINE "Added wave buffer %d (%8x)\r\n", ii, lpcs->alpWaveHdr[ii]);
}
lpcs->iNextWave = 0; // current wave
lpcs->dwWaveBytes = 0L; // number of wave bytes
lpcs->dwWaveChunkCount = 0; // number of wave frames
return 0;
}
//
// AVI AudioUnPrepare - UnPrepares headers and closes the wave device.
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIAudioUnPrepare (LPCAPSTREAM lpcs)
{
int ii;
if (lpcs->hWaveIn)
{
waveInReset(lpcs->hWaveIn);
// unprepare any headers that have been prepared
//
for (ii=0; ii < lpcs->iNumAudio; ++ii)
if (lpcs->alpWaveHdr[ii] &&
(lpcs->alpWaveHdr[ii]->dwFlags & WHDR_PREPARED))
waveInUnprepareHeader (lpcs->hWaveIn,
lpcs->alpWaveHdr[ii],
sizeof(WAVEHDR));
waveInClose(lpcs->hWaveIn);
lpcs->hWaveIn = NULL;
}
return 0;
}
// ****************************************************************
// ******************** Video Buffer Control **********************
// ****************************************************************
#if defined CHICAGO
// Win95 capavi code
// AVIVideoInit - Allocates, and initialize buffers for video capture.
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIVideoInit (LPCAPSTREAM lpcs)
{
UINT iMaxVideo;
DWORD mmr;
LPTHKVIDEOHDR ptvh;
UINT ii;
DWORD cbVideo;
lpcs->iNumVideo = 0;
lpcs->iNextVideo = 0;
lpcs->dwVideoChunkCount = 0;
lpcs->dwFramesDropped = 0;
lpcs->fBuffersOnHardware = FALSE;
// When performing MCI step capture, buffer array is not used
if (lpcs->sCapParms.fStepMCIDevice)
return 0;
cbVideo = ROUNDUPTOSECTORSIZE (lpcs->lpBitsInfo->bmiHeader.biSizeImage
+ sizeof(RIFF),
lpcs->dwBytesPerSector)
+ lpcs->dwBytesPerSector;
// If the user hasn't specified the number of video buffers to use,
// assume the minimum
if (lpcs->sCapParms.wNumVideoRequested == 0) {
iMaxVideo = lpcs->sCapParms.wNumVideoRequested = MIN_VIDEO_BUFFERS;
lpcs->fCaptureFlags |= CAP_fDefaultVideoBuffers;
} else {
// use the number of video buffers that the user requested
// or the maximum that will fit in memory.
//
iMaxVideo = min (MAX_VIDEO_BUFFERS, lpcs->sCapParms.wNumVideoRequested);
}
if (iMaxVideo > 1)
{
DWORD dwFreeMem;
DWORD dwUserRequests;
DWORD dwAudioMem;
// How much actual free physical memory exists?
dwFreeMem = GetFreePhysicalMemory();
dwAudioMem = lpcs->dwWaveSize * lpcs->sCapParms.wNumAudioRequested;
#define FOREVER_FREE 32768L // Always keep this free for swap space
// How much memory will be used if we allocate per the request?
//
dwUserRequests = dwAudioMem
+ cbVideo * iMaxVideo
+ FOREVER_FREE;
// If request is greater than available memory, force fewer buffers
//
if (dwUserRequests > dwFreeMem)
{
if (dwFreeMem > dwAudioMem)
dwFreeMem -= dwAudioMem;
iMaxVideo = (int)(((dwFreeMem * 8) / 10) / cbVideo);
iMaxVideo = min (MAX_VIDEO_BUFFERS, iMaxVideo);
dprintf("iMaxVideo = %d\n", iMaxVideo);
}
}
mmr = vidxAllocHeaders(lpcs->hVideoIn, iMaxVideo, &ptvh);
if (mmr != MMSYSERR_NOERROR)
return IDS_CAP_VIDEO_ALLOC_ERROR;
AuxDebugEx (3, DEBUGLINE "vidxAllocHdrs returned ptvh=%X\r\n", ptvh);
AuxDebugDump (8, ptvh, sizeof(*ptvh) * iMaxVideo);
for (ii = 0; ii < iMaxVideo; ++ii)
{
LPVIDEOHDR pvh = NULL;
LPRIFF priff;
// in chicago we let the thunk layer allocate memory
// so that we can be assured that the memory can be easily
// thunked.
//
// the pointer will be rounded up to a sector size boundary
//
mmr = vidxAllocBuffer (lpcs->hVideoIn, ii, &ptvh, cbVideo);
if ((mmr != MMSYSERR_NOERROR) || (ptvh == NULL))
break;
lpcs->alpVideoHdr[ii] = pvh = &ptvh->vh;
// vidxAllocBuffer actually returns a couple of extra fields
// after the video header. the first of these holds the
// linear address of the buffer.
//
priff = (LPVOID) ROUNDUPTOSECTORSIZE (ptvh->p32Buff, lpcs->dwBytesPerSector);
#ifdef DEBUG
{
LPBYTE pb = (LPVOID)ptvh->p32Buff;
AuxDebugEx (4, DEBUGLINE "buffer[%d] at %x linear. Doing touch test\r\n",
ii, ptvh->p32Buff);
pb[0] = 0;
pb[cbVideo-1] = 0;
}
#endif
// write the riff header for this chunk.
//
priff->dwType = MAKEAVICKID(cktypeDIBbits, 0);
if (lpcs->lpBitsInfo->bmiHeader.biCompression == BI_RLE8)
priff->dwType = MAKEAVICKID(cktypeDIBcompressed, 0);
priff->dwSize = lpcs->lpBitsInfo->bmiHeader.biSizeImage;
// init the video header
//
pvh->lpData = (LPVOID)(priff + 1);
pvh->dwBufferLength = priff->dwSize;
pvh->dwBytesUsed = 0;
pvh->dwTimeCaptured = 0;
pvh->dwUser = 0;
pvh->dwFlags = 0;
AuxDebugEx (4, DEBUGLINE "lpVideoHdr[%d]==%X\r\n", ii, lpcs->alpVideoHdr[ii]);
AuxDebugDump (8, lpcs->alpVideoHdr[ii], sizeof(*ptvh));
}
lpcs->iNumVideo = ii;
lpcs->cbVideoAllocation = cbVideo;
dprintf("cbVideo = %ld \n", cbVideo);
dprintf("iNumVideo Allocated = %d \n", lpcs->iNumVideo);
return lpcs->iNumVideo ? 0 : IDS_CAP_VIDEO_ALLOC_ERROR;
}
//
// AVIVideoPrepare - Prepares headers and adds buffers to the device
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIVideoPrepare (LPCAPSTREAM lpcs)
{
int ii;
// When performing MCI step capture, buffer array is not used
if (lpcs->sCapParms.fStepMCIDevice)
return 0;
#ifdef JMK_HACK_CHECKHDR
{
LPTHKVIDEOHDR lptvh = (LPVOID)lpcs->alpVideoHdr[0];
if (HIWORD(lptvh->vh.lpData) != HIWORD(lptvh->p32Buff))
{
AuxDebugEx (0, DEBUGLINE "before stream init: hdr trouble\r\n");
AuxDebugEx (0, DEBUGLINE "iNext=%d, ptvh=%X\r\n", lpcs->iNextVideo, lptvh);
AuxDebugDump (0, lptvh, sizeof(*lptvh));
AuxDebugEx (0, DEBUGLINE "alpVideoHdrs=%X\r\n", lpcs->alpVideoHdr);
AuxDebugDump (0, lpcs->alpVideoHdr, sizeof(lpcs->alpVideoHdr[0]) * 8);
INLINE_BREAK;
return IDS_CAP_VIDEO_OPEN_ERROR;
}
}
#endif
// Open the video stream, setting the capture rate
//
if (videoStreamInit(lpcs->hVideoIn,
lpcs->sCapParms.dwRequestMicroSecPerFrame,
lpcs->hRing0CapEvt,
0,
CALLBACK_EVENT))
{
dprintf("cant open video device!\n");
return IDS_CAP_VIDEO_OPEN_ERROR;
}
#ifdef JMK_HACK_CHECKHDR
{
LPTHKVIDEOHDR lptvh = (LPVOID)lpcs->alpVideoHdr[0];
if (HIWORD(lptvh->vh.lpData) != HIWORD(lptvh->p32Buff))
{
AuxDebugEx (0, DEBUGLINE "after stream init: hdr trouble\r\n");
AuxDebugEx (0, DEBUGLINE "iNext=%d, ptvh=%X\r\n", lpcs->iNextVideo, lptvh);
AuxDebugDump (0, lptvh, sizeof(*lptvh));
AuxDebugEx (0, DEBUGLINE "alpVideoHdrs=%X\r\n", lpcs->alpVideoHdr);
AuxDebugDump (0, lpcs->alpVideoHdr, sizeof(lpcs->alpVideoHdr[0]) * 8);
INLINE_BREAK;
return IDS_CAP_VIDEO_OPEN_ERROR;
}
}
#endif
// Prepare (lock) the buffers, and give them to the device
//
for (ii = 0; ii < lpcs->iNumVideo; ++ii)
{
if (vidxAddBuffer (lpcs->hVideoIn,
lpcs->alpVideoHdr[ii],
sizeof(VIDEOHDR)))
{
lpcs->iNumVideo = ii;
dprintf("**** could only prepare %d Video buffers!\n", lpcs->iNumVideo);
break;
}
}
#ifdef JMK_HACK_CHECKHDR
{
LPTHKVIDEOHDR lptvh = (LPVOID)lpcs->alpVideoHdr[0];
if (IsBadWritePtr (lptvh, sizeof(*lptvh)) ||
HIWORD(lptvh->vh.lpData) != HIWORD(lptvh->p16Alloc))
{
AuxDebugEx (0, DEBUGLINE "after add buffers: hdr trouble\r\n");
AuxDebugEx (0, DEBUGLINE "iNext=%d, ptvh=%X\r\n", lpcs->iNextVideo, lptvh);
AuxDebugDump (0, lptvh, sizeof(*lptvh));
AuxDebugEx (0, DEBUGLINE "alpVideoHdrs=%X\r\n", lpcs->alpVideoHdr);
AuxDebugDump (0, lpcs->alpVideoHdr, sizeof(lpcs->alpVideoHdr[0]) * 8);
INLINE_BREAK;
return IDS_CAP_VIDEO_OPEN_ERROR;
}
}
#endif
return 0;
}
#else // code below is !CHICAGO
// this structure is used to keep track of memory allocation
// for video buffers used in capture. it is allocated when
// allocating a videohdr would be called for
//
typedef struct _cap_videohdr {
VIDEOHDR vh;
LPBYTE pAlloc; // address of allocated buffer
DWORD dwMemIdent; // identity of allocation (used in Chicago)
DWORD dwReserved; // used in chicago
BOOL bHwBuffer; // TRUE if buffer is allocated using videoStreamAllocBuffer
} CAPVIDEOHDR, FAR *LPCAPVIDEOHDR;
// AVIVideoInit - Allocates, and initialize buffers for video capture.
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIVideoInit (LPCAPSTREAM lpcs)
{
int iMaxVideo;
int ii;
LPCAPVIDEOHDR pcvh;
LPVOID pbuf;
DWORD cbVideo;
BOOL fAllowHardwareBuffers;
//#define SINGLEHEADERBLOCK
lpcs->iNumVideo = 0;
lpcs->iNextVideo = 0;
lpcs->dwVideoChunkCount = 0;
lpcs->dwFramesDropped = 0;
lpcs->fBuffersOnHardware = FALSE;
fAllowHardwareBuffers = GetProfileIntA ("Avicap32", "AllowHardwareBuffers", TRUE);
// When performing MCI step capture, buffer array is not used
if (lpcs->sCapParms.fStepMCIDevice)
return 0;
cbVideo = ROUNDUPTOSECTORSIZE (lpcs->lpBitsInfo->bmiHeader.biSizeImage
+ sizeof(RIFF),
lpcs->dwBytesPerSector)
+ lpcs->dwBytesPerSector;
// If the user hasn't specified the number of video buffers to use,
// assume the minimum
if (lpcs->sCapParms.wNumVideoRequested == 0) {
UINT cDefaultVideoBuffers = GetProfileIntA ("Avicap32", "nVideoBuffers", MIN_VIDEO_BUFFERS);
cDefaultVideoBuffers = min(MAX_VIDEO_BUFFERS, max(MIN_VIDEO_BUFFERS, cDefaultVideoBuffers));
iMaxVideo = lpcs->sCapParms.wNumVideoRequested = cDefaultVideoBuffers;
lpcs->fCaptureFlags |= CAP_fDefaultVideoBuffers;
} else {
// use the number of video buffers that the user requested
// or the maximum that will fit in memory.
//
iMaxVideo = min (MAX_VIDEO_BUFFERS, lpcs->sCapParms.wNumVideoRequested);
}
// Post VFW 1.1a, see if the driver can allocate memory
//
#ifdef ALLOW_HW_BUFFERS
if (fAllowHardwareBuffers && (videoStreamAllocBuffer (lpcs->hVideoIn, (LPVOID *) &pbuf, cbVideo)
== DV_ERR_OK))
{
DWORD dwRet;
dprintf("Allocated test h/w buffer at address %8x, size %d bytes", pbuf, cbVideo);
lpcs->fBuffersOnHardware = TRUE;
dwRet = videoStreamFreeBuffer (lpcs->hVideoIn, pbuf);
dprintf("Freed test h/w buffer at address %8x, retcode 0x%x", pbuf, dwRet);
}
else
#endif
{
DWORD dwFreeMem;
DWORD dwUserRequests;
DWORD dwAudioMem;
lpcs->fBuffersOnHardware = FALSE;
// How much actual free physical memory exists?
dwFreeMem = GetFreePhysicalMemory();
dwAudioMem = lpcs->dwWaveSize * lpcs->sCapParms.wNumAudioRequested;
#define FOREVER_FREE 32768L // Always keep this free for swap space
// How much memory will be used if we allocate per the request?
//
dwUserRequests = dwAudioMem
+ cbVideo * iMaxVideo
+ FOREVER_FREE;
// If request is greater than available memory, force fewer buffers
//
if (dwUserRequests > dwFreeMem)
{
if (dwFreeMem > dwAudioMem)
dwFreeMem -= dwAudioMem;
iMaxVideo = (int)(((dwFreeMem * 8) / 10) / cbVideo);
iMaxVideo = min (MAX_VIDEO_BUFFERS, iMaxVideo);
dprintf("iMaxVideo = %d\n", iMaxVideo);
}
}
#ifdef SINGLEHEADERBLOCK
pcvh = GlobalAllocPtr (GMEM_MOVEABLE, iMaxVideo * sizeof(CAPVIDEOHDR));
// note: pcvh is freed by referencing through alpVideoHdr[0]
if ( ! pcvh)
return IDS_CAP_VIDEO_ALLOC_ERROR;
AuxDebugEx (3, DEBUGLINE "allocated video headers pcvh=%X\r\n", pcvh);
#endif
// Set up the buffers presuming fixed size DIBs and Junk chunks
// These will be modified later if the device provides compressed data
for (ii = 0; ii < iMaxVideo; ++ii)
{
LPVIDEOHDR pvh = NULL;
LPRIFF priff;
#ifndef SINGLEHEADERBLOCK
pcvh = (LPCAPVIDEOHDR)GlobalAllocPtr(GMEM_MOVEABLE, sizeof(CAPVIDEOHDR));
if (pcvh== NULL)
break;
lpcs->alpVideoHdr[ii] = (LPVIDEOHDR)pcvh;
ZeroMemory(pcvh, sizeof (CAPVIDEOHDR));
#endif
#ifdef ALLOW_HW_BUFFERS
//
// for the first buffer, always try to allocate on hardware,
// NO. If we are not to use hardware buffers, then do not use them.
// if that fails, grab virtual memory for the buffer.
// for all but the first buffer, we use whatever worked for
// the first buffer, and if that fails. we stop allocating buffers
//
if (lpcs->fBuffersOnHardware)
{
MMRESULT mmr;
pbuf = NULL;
mmr = videoStreamAllocBuffer (lpcs->hVideoIn, (LPVOID) &pbuf, cbVideo);
if ((mmr != MMSYSERR_NOERROR) || (pbuf == NULL))
{
if (0 == ii)
break; // nothing allocated
dprintf("Failed to allocate hardware buffer %d, rc=0x%x", ii, mmr);
// if the user did not ask for a specific number of buffers,
// or the hardware is set up to work with ONLY hardware
// allocated buffers, take what we've got and work with that.
if ((lpcs->fCaptureFlags & CAP_fDefaultVideoBuffers)
|| (GetProfileIntA ("Avicap32", "HardwareBuffersOnly", FALSE)))
{
break;
}
lpcs->fBuffersOnHardware = FALSE;
// use normal memory for the remaining video buffers.
pbuf = AllocSectorAlignedMem (cbVideo, lpcs->dwBytesPerSector);
}
else {
lpcs->fBuffersOnHardware = TRUE;
dprintf("Allocated hardware buffer %d at address %8x", ii, pbuf);
}
}
else
pbuf = AllocSectorAlignedMem (cbVideo, lpcs->dwBytesPerSector);
#else ! dont allow hw buffers
pbuf = AllocSectorAlignedMem (cbVideo, lpcs->dwBytesPerSector);
#endif // ALLOW_HW_BUFFERS
if (pbuf == NULL) {
#ifndef SINGLEHEADERBLOCK
GlobalFreePtr(pcvh);
lpcs->alpVideoHdr[ii] = NULL;
#endif
break;
}
// save the original allocation pointer to the buffer
// in the extra fields of the capture header. also remember
// whether we got the buffer from the driver or not
//
#ifndef SINGLEHEADERBLOCK
pcvh->pAlloc = pbuf;
pcvh->bHwBuffer = lpcs->fBuffersOnHardware;
lpcs->alpVideoHdr[ii] = pvh = &pcvh->vh;
#else
pcvh[ii].pAlloc = pbuf;
pcvh[ii].bHwBuffer = lpcs->fBuffersOnHardware;
lpcs->alpVideoHdr[ii] = pvh = &pcvh[ii].vh;
#endif
priff = (LPVOID) ROUNDUPTOSECTORSIZE (pbuf, lpcs->dwBytesPerSector);
// write the riff header for this frame
//
priff->dwType = MAKEAVICKID(cktypeDIBbits, 0);
if (lpcs->lpBitsInfo->bmiHeader.biCompression == BI_RLE8)
priff->dwType = MAKEAVICKID(cktypeDIBcompressed, 0);
priff->dwSize = lpcs->lpBitsInfo->bmiHeader.biSizeImage;
// fill in the video hdr for this frame
//
pvh->lpData = (LPVOID)(priff + 1);
pvh->dwBufferLength = priff->dwSize;
pvh->dwBytesUsed = 0;
pvh->dwTimeCaptured = 0;
pvh->dwUser = 0;
pvh->dwFlags = 0;
AuxDebugEx (4, DEBUGLINE "lpVideoHdr[%d]==%X\r\n", ii, lpcs->alpVideoHdr[ii]);
AuxDebugDump (8, lpcs->alpVideoHdr[ii], sizeof(*pcvh));
}
lpcs->iNumVideo = ii;
lpcs->cbVideoAllocation = cbVideo;
// if we did not create even a single buffer, free the headers
//
#ifdef SINGLEHEADERBLOCK
if ( ! lpcs->iNumVideo)
GlobalFreePtr (pcvh);
#else
// we allocate video headers as we proceed. There is nothing to free
#endif
#ifdef ALLOW_HW_BUFFERS
if (lpcs->fBuffersOnHardware)
dprintf("HARDWARE iNumVideo Allocated = %d \n", lpcs->iNumVideo);
else
#endif
dprintf("HIGH iNumVideo Allocated = %d \n", lpcs->iNumVideo);
return lpcs->iNumVideo ? 0 : IDS_CAP_VIDEO_ALLOC_ERROR;
}
void CALLBACK
VideoCallback(
HVIDEO hvideo,
UINT msg,
DWORD dwInstance,
DWORD lParam1,
DWORD lParam2
)
{
LPCAPSTREAM lpcs = (LPCAPSTREAM) dwInstance;
if (lpcs && lpcs->hCaptureEvent) {
SetEvent(lpcs->hCaptureEvent);
} else {
AuxDebugEx(1, DEBUGLINE "VideoCallback with NO instance data\r\n");
}
}
//
// AVIVideoPrepare - Prepares headers and adds buffers to the device
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIVideoPrepare (LPCAPSTREAM lpcs)
{
MMRESULT mmr;
int ii;
// When performing MCI step capture, buffer array is not used
//
if (lpcs->sCapParms.fStepMCIDevice)
return 0;
// Open the video stream, setting the capture rate
//
mmr = videoStreamInit (lpcs->hVideoIn,
lpcs->sCapParms.dwRequestMicroSecPerFrame,
(DWORD) VideoCallback,
(DWORD) lpcs,
CALLBACK_FUNCTION);
if (mmr) {
dprintf("cannot open video device! Error is %d\n", mmr);
return IDS_CAP_VIDEO_OPEN_ERROR;
}
// Prepare (lock) the buffers, and give them to the device
//
for (ii = 0; ii < lpcs->iNumVideo; ++ii)
{
mmr = videoStreamPrepareHeader (lpcs->hVideoIn,
lpcs->alpVideoHdr[ii],
sizeof(VIDEOHDR));
if (mmr)
{
lpcs->iNumVideo = ii;
dprintf("**** could only prepare %d Video buffers!\n", lpcs->iNumVideo);
break;
}
mmr = videoStreamAddBuffer (lpcs->hVideoIn,
lpcs->alpVideoHdr[ii],
sizeof(VIDEOHDR));
if (mmr)
return IDS_CAP_VIDEO_ALLOC_ERROR;
AuxDebugEx(3, DEBUGLINE "Added video buffer %d (%8x)\r\n", ii, lpcs->alpVideoHdr[ii]);
}
return 0;
}
#endif // not chicago
//
// AVI VideoUnPrepare - UnPrepares headers, frees memory, and
// resets the video in device.
// This routine is also used by MCI capture.
// Returns: 0 on success, otherwise an error code.
UINT AVIVideoUnPrepare (LPCAPSTREAM lpcs)
{
// When performing MCI step capture, buffer array is not used
//
if (lpcs->sCapParms.fStepMCIDevice)
return 0;
// Reset the buffers so they can be freed
//
if (lpcs->hVideoIn) {
videoStreamReset(lpcs->hVideoIn);
// unprepare headers
// Unlock and free headers and data
#if defined CHICAGO
vidxFreeHeaders (lpcs->hVideoIn);
ZeroMemory (lpcs->alpVideoHdr, sizeof(lpcs->alpVideoHdr));
#else
{
int ii;
#ifdef SINGLEHEADERBLOCK
LPCAPVIDEOHDR pcvhAll = (LPVOID)lpcs->alpVideoHdr[0];
#endif
for (ii = 0; ii < lpcs->iNumVideo; ++ii)
{
LPCAPVIDEOHDR pcvh = (LPVOID)lpcs->alpVideoHdr[ii];
if (pcvh)
{
if (pcvh->vh.dwFlags & VHDR_PREPARED)
videoStreamUnprepareHeader (lpcs->hVideoIn,
&pcvh->vh,
sizeof(VIDEOHDR));
if (pcvh->pAlloc) {
#ifdef ALLOW_HW_BUFFERS
if (pcvh->bHwBuffer)
{
dprintf("Freeing hardware buffer %d at address %8x", ii, pcvh->pAlloc);
videoStreamFreeBuffer (lpcs->hVideoIn, (LPVOID)pcvh->pAlloc);
}
else
#endif
{
dprintf("Freeing video buffer %d at address %8x", ii, pcvh->pAlloc);
if (lpcs->fBuffersOnHardware) DebugBreak();
FreeSectorAlignedMem (pcvh->pAlloc);
}
} else {
dprintf("NO buffer allocated for index %d", ii);
}
#ifndef SINGLEHEADERBLOCK
GlobalFreePtr(pcvh);
#endif
lpcs->alpVideoHdr[ii] = NULL;
} else {
dprintf("NO video header for index %d", ii);
}
}
#ifdef SINGLEHEADERBLOCK
// free the array of video headers
//
if (pcvhAll) {
GlobalFreePtr (pcvhAll);
}
#endif
}
#endif
// Shut down the video stream
videoStreamFini(lpcs->hVideoIn);
}
return 0;
}
/*
* AVI Fini - undo the mess that AVIInit did.
*
*/
void AVIFini(LPCAPSTREAM lpcs)
{
AuxDebugEx (2, "AVIFini(%08x)\r\n", lpcs);
if (lpcs->lpDropFrame) {
FreeSectorAlignedMem (lpcs->lpDropFrame), lpcs->lpDropFrame = NULL;
}
AVIVideoUnPrepare (lpcs); // Free the video device and buffers
AVIAudioUnPrepare (lpcs); // Free the audio device
AVIAudioFini (lpcs); // Free the audio buffers
if (lpcs->hCaptureEvent) {
CloseHandle (lpcs->hCaptureEvent), lpcs->hCaptureEvent = NULL;
}
if (lpcs->heSyncWrite) {
CloseHandle (lpcs->heSyncWrite), lpcs->heSyncWrite = NULL;
}
if (hmodKernel) {
pfnCreateIoCompletionPort = NULL;
pfnGetQueuedCompletionStatus = NULL;
FreeLibrary(hmodKernel);
hmodKernel = 0;
}
AuxDebugEx (2, "AVIFini(...) exits\r\n");
}
//
// AVI Init
// This routine does all the non-File initalization for AVICapture.
// Returns: 0 on success, Error string value on failure.
//
UINT AVIInit (LPCAPSTREAM lpcs)
{
UINT wError = 0; // Success
LPBITMAPINFO lpBitsInfoOut; // Possibly compressed output format
// Allocate a DropFrame buffer
if (lpcs->lpDropFrame == NULL) {
assert (lpcs->dwBytesPerSector);
lpcs->lpDropFrame = AllocSectorAlignedMem (lpcs->dwBytesPerSector, lpcs->dwBytesPerSector);
}
/* No special video format given -- use the default */
#ifdef NEW_COMPMAN
if (lpcs->CompVars.hic == NULL)
lpBitsInfoOut = lpcs->lpBitsInfo;
else
lpBitsInfoOut = lpcs->CompVars.lpbiOut;
#else
lpBitsInfoOut = lpcs->lpBitsInfo;
#endif
// -------------------------------------------------------
// figure out buffer sizes
// -------------------------------------------------------
// Init all pointers to NULL
ZeroMemory (lpcs->alpVideoHdr, sizeof(lpcs->alpVideoHdr));
ZeroMemory (lpcs->alpWaveHdr, sizeof(lpcs->alpWaveHdr));
// -------------------------------------------------------
// Init Sound
// -------------------------------------------------------
if (lpcs->sCapParms.fCaptureAudio) {
if ((DWORD)(wError = AVIAudioInit (lpcs))) {
dprintf("can't init audio buffers!\n");
goto AVIInitFailed;
}
}
// -------------------------------------------------------
// Init Video
// -------------------------------------------------------
if ((DWORD)(wError = AVIVideoInit (lpcs))) {
dprintf("AVIVideoInitFailed (no buffers alloc'd)!\n");
goto AVIInitFailed;
}
// --------------------------------------------------------------
// Prepare audio buffers (lock em down) and give them to the device
// --------------------------------------------------------------
if (lpcs->sCapParms.fCaptureAudio) {
if ((DWORD)(wError = AVIAudioPrepare (lpcs))) {
dprintf("can't prepare audio buffers!\n");
goto AVIInitFailed;
}
}
// --------------------------------------------------------------
// Prepare video buffers (lock em down) and give them to the device
// --------------------------------------------------------------
if ((DWORD)(wError = AVIVideoPrepare (lpcs))) {
dprintf("can't prepare video buffers!\n");
goto AVIInitFailed;
}
// -------------------------------------------------------
// all done, return success
// -------------------------------------------------------
return (0); // SUCCESS !
// -------------------------------------------------------
// we got a error, return string ID of error message
// -------------------------------------------------------
AVIInitFailed:
AVIFini(lpcs); // Shutdown everything
return wError;
}
// Maintains info chunks which are written to the AVI header
//
BOOL FAR PASCAL SetInfoChunk(LPCAPSTREAM lpcs, LPCAPINFOCHUNK lpcic)
{
DWORD ckid = lpcic->fccInfoID;
LPVOID lpData = lpcic->lpData;
LONG cbData = lpcic->cbData;
LPBYTE lp;
LPBYTE lpw;
LPBYTE lpEnd;
LPBYTE lpNext;
LONG cbSizeThis;
BOOL fOK = FALSE;
// Delete all info chunks?
if (ckid == 0) {
if (lpcs->lpInfoChunks) {
GlobalFreePtr (lpcs->lpInfoChunks);
lpcs->lpInfoChunks = NULL;
lpcs->cbInfoChunks = 0;
}
return TRUE;
}
// Try removing an entry if it already exists...
// Also used if lpData is NULL to just remove an entry
// note: lpw and lpEnd are LPRIFF values... except the code is written
// to use them as pointers to an array of DWORD values. (yuk)
//
lpw = (LPBYTE)lpcs->lpInfoChunks; // always points at fcc
lpEnd = (LPBYTE)lpcs->lpInfoChunks + lpcs->cbInfoChunks;
while (lpw < lpEnd) {
cbSizeThis = ((DWORD UNALIGNED FAR *)lpw)[1];
cbSizeThis += cbSizeThis & 1; // force WORD (16 bit) alignment
// Point lpNext at the next RIFF block
lpNext = lpw + cbSizeThis + sizeof (DWORD) * 2;
// If this info chunk is the same as that passed in... we can delete the
// existing information
if ((*(DWORD UNALIGNED FAR *) lpw) == ckid) {
lpcs->cbInfoChunks -= cbSizeThis + sizeof (DWORD) * 2;
// could have coded: lpcs->cbInfoChunks -= lpNext - lpw;
// the next line should always be true...
if (lpNext <= lpEnd) {
if (lpEnd - lpNext)
CopyMemory (lpw, lpNext, lpEnd - lpNext);
if (lpcs->cbInfoChunks) {
lpcs->lpInfoChunks = (LPBYTE) GlobalReAllocPtr( // shrink it
lpcs->lpInfoChunks,
lpcs->cbInfoChunks,
GMEM_MOVEABLE);
}
else {
if (lpcs->lpInfoChunks)
GlobalFreePtr (lpcs->lpInfoChunks);
lpcs->lpInfoChunks = NULL;
}
fOK = TRUE;
}
break;
}
else
lpw = lpNext;
}
if (lpData == NULL || cbData == 0) // Only deleting, get out
return fOK;
// Add a new entry
cbData += cbData & 1; // force WORD (16 bit) alignment
cbData += sizeof(RIFF); // add sizeof RIFF
if (lpcs->lpInfoChunks)
lp = GlobalReAllocPtr(lpcs->lpInfoChunks, lpcs->cbInfoChunks + cbData, GMEM_MOVEABLE);
else
lp = GlobalAllocPtr(GMEM_MOVEABLE, cbData);
if (!lp)
return FALSE;
// Save the pointer in our status block
lpcs->lpInfoChunks = lp;
// build RIFF chunk in block
//
((LPRIFF)(lp + lpcs->cbInfoChunks))->dwType = ckid;
((LPRIFF)(lp + lpcs->cbInfoChunks))->dwSize = lpcic->cbData;
CopyMemory (lp + lpcs->cbInfoChunks + sizeof(RIFF),
lpData,
cbData - sizeof(RIFF));
// Update the length of the info chunk
lpcs->cbInfoChunks += cbData;
return TRUE;
}
/*+ ProcessNextVideoBuffer
*
*-===============================================================*/
STATICFN BOOL _inline ProcessNextVideoBuffer (
LPCAPSTREAM lpcs,
BOOL fStopping,
LPUINT lpuError,
LPVIDEOHDR * plpvhDraw,
LPBOOL lpbPending)
{
LPVIDEOHDR lpvh;
*lpuError = 0;
*plpvhDraw = NULL;
*lpbPending = FALSE;
lpvh = lpcs->alpVideoHdr[lpcs->iNextVideo];
if (!(lpvh->dwFlags & VHDR_DONE)) {
return fStopping;
}
#if defined CHICAGO
{
LPTHKVIDEOHDR lptvh = (LPVOID)lpvh;
#ifdef JMK_HACK_CHECKHDR
if (IsBadWritePtr (lptvh, sizeof(*lptvh)) ||
HIWORD(lptvh->vh.lpData) != HIWORD(lptvh->p16Alloc))
{
OutputDebugStringA(DEBUGLINE "trouble with video hdr\r\n");
AuxDebugEx (0, DEBUGLINE "iNext=%d, ptvh=%X\r\n", lpcs->iNextVideo, lptvh);
AuxDebugDump (0, lptvh, sizeof(*lptvh));
AuxDebugEx (0, DEBUGLINE "alpVideoHdrs=%X\r\n", lpcs->alpVideoHdr);
AuxDebugDump (0, lpcs->alpVideoHdr, sizeof(lpcs->alpVideoHdr[0]) * 8);
INLINE_BREAK;
return TRUE;
}
#endif
// Swap the linear pointer back (was swapped in vidxAddBuffer)
//
lptvh->vh.lpData = (LPVOID)(ROUNDUPTOSECTORSIZE(lptvh->p32Buff, lpcs->dwBytesPerSector) + sizeof(RIFF));
}
#endif
if (lpvh->dwBytesUsed)
{
DWORD dwTime;
DWORD dwBytesUsed = lpvh->dwBytesUsed;
BOOL fKeyFrame = lpvh->dwFlags & VHDR_KEYFRAME;
LPVOID lpData = lpvh->lpData;
// get expected time for this frame in milliseconds
//
dwTime = MulDiv (lpcs->dwVideoChunkCount + 1,
lpcs->sCapParms.dwRequestMicroSecPerFrame,
1000);
#ifdef NEW_COMPMAN
//
// We are automatically compressing during capture, so
// compress the frame before we pass it on to be written
//
if (lpcs->CompVars.hic)
{
LPRIFF priff;
dwBytesUsed = 0;
lpData = ICSeqCompressFrame(&lpcs->CompVars, 0,
lpvh->lpData,
&fKeyFrame,
&dwBytesUsed);
priff = ((LPRIFF)lpData) -1;
priff->dwType = MAKEAVICKID(cktypeDIBbits, 0);
priff->dwSize = dwBytesUsed;
}
#endif // NEW_COMPMAN
// do video stream callback for this frame
//
if (lpcs->CallbackOnVideoStream)
lpcs->CallbackOnVideoStream (lpcs->hwnd, lpvh);
lpvh->dwFlags &= ~VHDR_DONE;
// if we are not capturing to disk, just increment
// the 'chunk count' (i.e. frame count?) and go on
// otherwise we want to queue the frame up to write
// here
//
if ( ! (lpcs->fCaptureFlags & CAP_fCapturingToDisk))
{
// Warning: Kludge to create frame chunk count when net capture
// follows.
++lpcs->dwVideoChunkCount;
}
else
{
int nAppendDummyFrames = 0;
// if the expected time for this frame is less than the
// timestamp for the frame. we may have dropped some frames
// before this frame.
//
if (lpcs->dwVideoChunkCount && (dwTime < lpvh->dwTimeCaptured))
{
int nDropCount;
BOOL bPending;
// calculate how many frames have been dropped.
// NOTE: this number may be zero if dwTimeCaptured is just
// a little bit late.
//
nDropCount = MulDiv(lpvh->dwTimeCaptured - dwTime,
1000,
lpcs->sCapParms.dwRequestMicroSecPerFrame);
#ifdef JMK_HACK_TIMERS
if (pTimerRiff)
pTimerRiff->vchd.dwDropFramesNotAppended += nDropCount;
#endif
// If any frames have been dropped, write them out before
// we get back to writing the current frame.
//
if (nDropCount > 0)
{
AuxDebugEx(2,"*****Adding %d to the dropcount\r\n", nDropCount);
lpcs->dwFramesDropped += nDropCount;
if (! AVIWriteDummyFrames (lpcs, nDropCount, lpuError, &bPending))
fStopping = TRUE;
}
}
#ifdef JMK_HACK_TIMERS
if (pTimerRiff) {
if (nTimerIndex == CLIPBOARDLOGSIZE)
nTimerIndex = 0;
// nTimerIndex will be OK if ((nTimerIndex < CLIPBOARDLOGSIZE) && pTimerStuff)
if (pTimerStuff)
{
pCurTimerStuff = &pTimerStuff[nTimerIndex];
++nTimerIndex;
pCurTimerStuff->nFramesAppended = 0;
pCurTimerStuff->nDummyFrames = (WORD)lpcs->dwFramesDropped;
pCurTimerStuff->dwFrameTickTime = dwTime;
pCurTimerStuff->dwFrameStampTime = lpvh->dwTimeCaptured;
pCurTimerStuff->dwVideoChunkCount = lpcs->dwVideoChunkCount;
pCurTimerStuff->dwTimeWritten = pcDeltaTicks(&pctWriteBase);
pCurTimerStuff->dwTimeToWrite = 0;
pCurTimerStuff->nVideoIndex = (WORD)lpcs->iNextVideo;
pCurTimerStuff->nAudioIndex = (WORD)lpcs->iNextWave;
}
} // fClipboardLogging
#endif // JMK_HACK_TIMERS
// look ahead for dummy frames and try to
// append them to the current frame
//
nAppendDummyFrames = 0;
#define LOOKAHEAD_FOR_DUMMYS 1
#ifdef LOOKAHEAD_FOR_DUMMYS
{
int iNext;
LPVIDEOHDR lpvhNext;
iNext = lpcs->iNextVideo+1;
if (iNext >= lpcs->iNumVideo)
iNext = 0;
// is the next frame done already? if so
// we can append any dropped frames to the end of
// this frame before we write it out
//
lpvhNext = lpcs->alpVideoHdr[iNext];
if (lpvhNext->dwFlags & VHDR_DONE)
{
// Recalculate the current time, which may have
// changed if dummy frames were inserted above
dwTime = MulDiv (lpcs->dwVideoChunkCount + 1,
lpcs->sCapParms.dwRequestMicroSecPerFrame,
1000);
nAppendDummyFrames =
MulDiv (lpvhNext->dwTimeCaptured - dwTime,
1000,
lpcs->sCapParms.dwRequestMicroSecPerFrame);
if ((--nAppendDummyFrames) < 0)
nAppendDummyFrames = 0;
else {
AuxDebugEx(3, DEBUGLINE "Appending %d dummy frames", nAppendDummyFrames);
}
AuxDebugEx(1,"*****Adding %d to the dropcount in lookahead mode\r\n", nAppendDummyFrames);
lpcs->dwFramesDropped += nAppendDummyFrames;
#ifdef JMK_HACK_TIMERS
if (pTimerRiff) {
pTimerRiff->vchd.dwDropFramesAppended += nAppendDummyFrames;
pCurTimerStuff->nFramesAppended = (WORD)nAppendDummyFrames;
}
#endif
}
}
#endif
if ( ! AVIWriteVideoFrame (lpcs,
lpData,
dwBytesUsed,
fKeyFrame,
lpcs->iNextVideo,
nAppendDummyFrames,
lpuError, lpbPending))
fStopping = TRUE;
#ifdef JMK_HACK_TIMERS
if (pCurTimerStuff)
{
pCurTimerStuff->dwTimeToWrite = pcDeltaTicks(&pctWriteBase);
pCurTimerStuff->bPending = *lpbPending;
}
#endif
}
}
// return lpvh to the caller so that the frame can be
// drawn (time permitting)
//
*plpvhDraw = lpvh;
// increment the next Video buffer pointer
//
if (++lpcs->iNextVideo >= lpcs->iNumVideo)
lpcs->iNextVideo = 0;
return fStopping;
}
/*+ ProcessAudioBuffers
*
*-===============================================================*/
STATICFN BOOL _inline ProcessAudioBuffers (
LPCAPSTREAM lpcs,
BOOL fStopping,
LPUINT lpuError)
{
int iLastWave;
UINT ii;
LPWAVEHDR lpwh;
*lpuError = 0;
assert (lpcs->sCapParms.fCaptureAudio);
// if all buffers are done, we have broke audio.
//
iLastWave = lpcs->iNextWave == 0 ? lpcs->iNumAudio -1 : lpcs->iNextWave-1;
if (!fStopping && lpcs->alpWaveHdr[iLastWave]->dwFlags & WHDR_DONE)
lpcs->fAudioBreak = TRUE;
// process all done buffers, but no more than iNumAudio at one
// pass (to avoid getting stuck here forever)
//
for (ii = 0; ii < (UINT)lpcs->iNumAudio; ++ii)
{
BOOL bPending;
// if the next buffer is not done, break out of the loop
// and return to the caller
//
lpwh = lpcs->alpWaveHdr[lpcs->iNextWave];
if (!(lpwh->dwFlags & WHDR_DONE))
break;
lpwh->dwFlags &= ~WHDR_DONE;
// is there any data in the buffer ?
// if so, do wave stream callback, then write the
// buffer
//
bPending = FALSE;
if (lpwh->dwBytesRecorded)
{
if (lpcs->CallbackOnWaveStream)
lpcs->CallbackOnWaveStream (lpcs->hwnd, lpwh);
if ( ! (lpcs->fCaptureFlags & CAP_fCapturingToDisk))
{
lpcs->dwWaveChunkCount++;
lpcs->dwWaveBytes += lpwh->dwBytesRecorded;
}
else
{
// write the audio buffer, bPending will be true
// if the write will complete asynchronously
//
if ( ! AVIWriteAudio (lpcs, lpwh, lpcs->iNextWave,
lpuError, &bPending))
fStopping = TRUE;
}
}
// if we are not writing async, we can put the buffer
// back on the wave driver's queue now
//
if ( ! bPending)
{
lpwh->dwBytesRecorded = 0;
AuxDebugEx(3, DEBUGLINE "Calling waveInAddBuffer for address %8x", lpwh);
if (waveInAddBuffer(lpcs->hWaveIn, lpwh, sizeof(WAVEHDR)))
{
fStopping = TRUE;
*lpuError = IDS_CAP_WAVE_ADD_ERROR;
}
}
// increment the next wave buffer pointer
//
if (++lpcs->iNextWave >= lpcs->iNumAudio)
lpcs->iNextWave = 0;
}
return fStopping;
}
/*+ ProcessAsyncIOBuffers
*
*-===============================================================*/
STATICFN BOOL _inline ProcessAsyncIOBuffers (
LPCAPSTREAM lpcs,
BOOL fStopping,
LPUINT lpuError)
{
UINT ii;
struct _avi_async * lpah;
// if there are no async buffer headers, there is nothing to do!
//
*lpuError = 0;
assert (lpcs->pAsync);
//
// process all done buffers, stopping when there are no more outstanding
// iNextAsync can never go past iLastAsync.
//
while(lpcs->iNextAsync != lpcs->iLastAsync)
{
DWORD dwUsed;
// if this async header has never been used,
// we are done
//
lpah = &lpcs->pAsync[lpcs->iNextAsync];
assert (lpah->uType);
AuxDebugEx (2, DEBUGLINE "processing async io buffer %d off=%x\r\n",
lpcs->iNextAsync, lpah->ovl.Offset);
// if the next buffer is not done, or failed break
// out of the loop
//
// if the io on this block has already completed (because the IO
// completed out of order) queue it to the device without waiting
// otherwise get the next completion status. If a block has
// completed, and it is the block at the head of the async queue,
// then it can be passed straight back to the device queue. If the
// completed block is not the one we are expecting, we mark the IO
// as complete, then return. Thought..call GetQueuedCompletionStatus
// in a loop, until there are no more blocks pending. This way we
// might get to complete the block we want on this call to
// ProcessAsyncIOBuffers.
//
if (lpah->uType & ASYNCIOPENDING) {
DWORD dwWritten;
DWORD key;
LPOVERLAPPED povl;
BOOL fResult =
pfnGetQueuedCompletionStatus(lpcs->hCompletionPort,
&dwWritten,
&key,
&povl,
0);
if (fResult) {
// we dequeued a block. Did we dequeue the one we wanted?
((struct _avi_async *)povl)->uType &= ~ASYNCIOPENDING;
if ((PVOID)povl == (PVOID)lpah) {
// this is the one we wanted
// fall through and add back to the device queue
AuxDebugEx(2,"Dequeued the block we wanted at %8x\r\n", lpah);
} else {
// the io block completed out of order.
// Clear the io pending flag and return.
AuxDebugEx(1,"Dequeued out of order at %8x\r\n", povl->hEvent);
break;
}
} else {
if (povl) {
// a failed io operation
*lpuError = IDS_CAP_FILE_WRITE_ERROR;
AuxDebugEx(1, DEBUGLINE "A failed IO operation (GQCS)\r\n");
fStopping = TRUE;
} else {
// nothing completed
AuxDebugEx(3, DEBUGLINE "Nothing completed on call to GQCS\r\n");
break;
}
}
} else {
// IO is already complete for this block
}
// the buffer is done, so now we need to queue the wave/video
// buffer back to the wave/video driver
//
assert (!(lpah->uType & ASYNCIOPENDING));
switch (lpah->uType)
{
case ASYNC_BUF_VIDEO:
{
LPVIDEOHDR lpvh = lpcs->alpVideoHdr[lpah->uIndex];
#if defined CHICAGO
if (vidxAddBuffer(lpcs->hVideoIn, lpvh, sizeof (VIDEOHDR)))
#else
AuxDebugEx(3, DEBUGLINE "Queueing video buffer lpvh=%x (index %d)\r\n", lpvh, lpah->uIndex);
if (videoStreamAddBuffer(lpcs->hVideoIn, lpvh, sizeof (VIDEOHDR)))
#endif
{
fStopping = TRUE;
*lpuError = IDS_CAP_VIDEO_ADD_ERROR;
}
}
break;
case ASYNC_BUF_AUDIO:
{
LPWAVEHDR lpwh = lpcs->alpWaveHdr[lpah->uIndex];
lpwh->dwBytesRecorded = 0;
AuxDebugEx(3, DEBUGLINE "Queueing audio buffer lpwh=%x (index %d)\r\n", lpwh, lpah->uIndex);
if (waveInAddBuffer (lpcs->hWaveIn, lpwh, sizeof(WAVEHDR)))
{
fStopping = TRUE;
*lpuError = IDS_CAP_WAVE_ADD_ERROR;
}
}
break;
//case ASYNC_BUF_DROP:
//{
//}
//break;
}
// mark the overlapped header structure as vacant
lpah->uType = 0;
lpah->uIndex = 0;
// increment to the next async io buffer
//
if (++lpcs->iNextAsync >= lpcs->iNumAsync)
lpcs->iNextAsync = 0; // wrapped...
}
return fStopping;
}
/*+ ShowCompletionStatus
*
*-===============================================================*/
STATICFN void ShowCompletionStatus (
LPCAPSTREAM lpcs,
BOOL fCapturedOK)
{
// Notify if there was an error while recording
//
if ( ! fCapturedOK)
errorUpdateError (lpcs, IDS_CAP_RECORDING_ERROR);
// put up completion message on status line
//
if (lpcs->fCaptureFlags & CAP_fCapturingToDisk)
{
DWORD dw;
// The muldiv32 doesn't give 0 if numerator is zero
dw = 0;
if (lpcs->dwVideoChunkCount)
dw = muldiv32(lpcs->dwVideoChunkCount,1000000,lpcs->dwTimeElapsedMS);
if (lpcs->sCapParms.fCaptureAudio)
{
// "Captured %d.%03d sec. %ld frames (%ld dropped) (%d.%03d fps). %ld audio bytes (%d.%03d sps)"
statusUpdateStatus(lpcs, IDS_CAP_STAT_VIDEOAUDIO,
(UINT)(lpcs->dwTimeElapsedMS/1000),
(UINT)(lpcs->dwTimeElapsedMS%1000),
lpcs->dwVideoChunkCount,
lpcs->dwFramesDropped,
(UINT)(dw / 1000),
(UINT)(dw % 1000),
lpcs->dwWaveBytes,
(UINT) lpcs->lpWaveFormat->nSamplesPerSec / 1000,
(UINT) lpcs->lpWaveFormat->nSamplesPerSec % 1000);
}
else
{
// "Captured %d.%03d sec. %ld frames (%ld dropped) (%d.%03d fps)."
statusUpdateStatus(lpcs, IDS_CAP_STAT_VIDEOONLY,
(UINT)(lpcs->dwTimeElapsedMS/1000),
(UINT)(lpcs->dwTimeElapsedMS%1000),
lpcs->dwVideoChunkCount,
lpcs->dwFramesDropped,
(UINT)(dw / 1000),
(UINT)(dw % 1000));
}
} // endif capturing to disk (no warnings or errors if to net)
// if capture was successful, warn the user about various abnormal
// conditions.
//
if (fCapturedOK)
{
if (lpcs->dwVideoChunkCount == 0)
{
// No frames captured, warn user that interrupts are probably not enabled.
errorUpdateError (lpcs, IDS_CAP_NO_FRAME_CAP_ERROR);
}
else if (lpcs->sCapParms.fCaptureAudio && lpcs->dwWaveBytes == 0)
{
// No audio captured, warn user audio card is hosed
errorUpdateError (lpcs, IDS_CAP_NO_AUDIO_CAP_ERROR);
}
else if (lpcs->sCapParms.fCaptureAudio && lpcs->fAudioBreak)
{
// some of the audio was dropped
if(lpcs->CompVars.hic) {
errorUpdateError (lpcs, IDS_CAP_AUDIO_DROP_COMPERROR);
} else {
errorUpdateError (lpcs, IDS_CAP_AUDIO_DROP_ERROR);
}
}
else if (lpcs->fCaptureFlags & CAP_fCapturingToDisk)
{
DWORD dwPctDropped;
assert (lpcs->dwVideoChunkCount);
dwPctDropped = 100 * lpcs->dwFramesDropped / lpcs->dwVideoChunkCount;
//
// dropped > 10% (default) of the frames
//
if (dwPctDropped > lpcs->sCapParms.wPercentDropForError)
errorUpdateError (lpcs, IDS_CAP_STAT_FRAMESDROPPED,
lpcs->dwFramesDropped,
lpcs->dwVideoChunkCount,
(UINT)(muldiv32(lpcs->dwFramesDropped,10000,lpcs->dwVideoChunkCount)/100),
(UINT)(muldiv32(lpcs->dwFramesDropped,10000,lpcs->dwVideoChunkCount)%100)/10
);
}
}
}
/*
* AVI Capture
* This is the main streaming capture loop for both audio and
* video. It will first init all buffers and drivers and then go into a
* loop checking for buffers to be filled. When a buffer is filled then
* the data for it is written out.
* Afterwards it cleans up after itself (frees buffers etc...)
* Returns: 0 on success, else error code
*/
void FAR PASCAL _LOADDS AVICapture1(LPCAPSTREAM lpcs)
{
BOOL fCapturedOK = TRUE;
BOOL fStopping; // True when finishing capture
BOOL fStopped; // True if driver notified to stop
TCHAR ach[128];
TCHAR achMsg[128];
UINT wError; // Error String ID
LPVIDEOHDR lpVidHdr;
LPWAVEHDR lpWaveHdr;
DWORD dwTimeStarted; // When did we start in milliseconds
DWORD dwTimeStopped;
DWORD dwTimeToStop; // Lesser of MCI capture time or frame limit
BOOL fTryToPaint = FALSE;
BOOL fTryToPaintAgain = FALSE;
HDC hdc;
HPALETTE hpalT;
HCURSOR hOldCursor;
RECT rcDrawRect;
CAPINFOCHUNK cic;
DWORD dwOldPrio;
BOOL bVideoWritePending;
LPVIDEOHDR lpvhDraw;
lpcs->fCaptureFlags |= CAP_fCapturingNow;
// we should Assert that CAP_fCapturingNow is already turned on
lpcs->dwReturn = DV_ERR_OK;
hOldCursor = SetCursor(lpcs->hWaitCursor);
statusUpdateStatus(lpcs, IDS_CAP_BEGIN); // Always the first message
// If not 1 Meg. free, give it up!!!
if (GetFreePhysicalMemory () < (1024L * 1024L)) {
errorUpdateError (lpcs, IDS_CAP_OUTOFMEM);
goto EarlyExit;
}
statusUpdateStatus(lpcs, IDS_CAP_STAT_CAP_INIT);
// Try painting the DIB only if Live window
fTryToPaintAgain = fTryToPaint = lpcs->fLiveWindow;
if (fTryToPaint) {
hdc = GetDC(lpcs->hwnd);
SetWindowOrgEx(hdc, lpcs->ptScroll.x, lpcs->ptScroll.y, NULL);
hpalT = DrawDibGetPalette (lpcs->hdd);
if (hpalT)
hpalT = SelectPalette( hdc, hpalT, FALSE);
RealizePalette(hdc);
if (lpcs->fScale)
GetClientRect (lpcs->hwnd, &rcDrawRect);
else
SetRect (&rcDrawRect, 0, 0, lpcs->dxBits, lpcs->dyBits);
}
// -------------------------------------------------------
// When should capture stop?
// -------------------------------------------------------
// If using MCI, capture for the shorter of the MCI period,
// or the capture limit
if (lpcs->sCapParms.fLimitEnabled)
dwTimeToStop = (DWORD) ((DWORD) 1000 * lpcs->sCapParms.wTimeLimit);
else
dwTimeToStop = (DWORD) -1L; // very large
if (lpcs->sCapParms.fMCIControl) {
DWORD dwTime;
// if MCI stop time not given, use lpcs->sCapParms.wTimeLimit
if (lpcs->sCapParms.dwMCIStopTime == lpcs->sCapParms.dwMCIStartTime)
lpcs->sCapParms.dwMCIStopTime = lpcs->sCapParms.dwMCIStartTime +
(DWORD) ((DWORD)1000 * lpcs->sCapParms.wTimeLimit);
dwTime = lpcs->sCapParms.dwMCIStopTime - lpcs->sCapParms.dwMCIStartTime;
if (lpcs->sCapParms.fLimitEnabled)
dwTimeToStop = min (dwTime, dwTimeToStop);
else
dwTimeToStop = dwTime;
}
//
// never ever try to capture more than the index size!
//
if (lpcs->fCaptureFlags & CAP_fCapturingToDisk)
{
DWORD dwTime = MulDiv (lpcs->sCapParms.dwIndexSize,
lpcs->sCapParms.dwRequestMicroSecPerFrame,
1000l);
dwTimeToStop = min (dwTime, dwTimeToStop);
}
// if doing MCI capture, initialize MCI device. if init fails
// go straight to the exit code
//
if (lpcs->sCapParms.fMCIControl)
{
if ( ! MCIDeviceOpen (lpcs) ||
! MCIDeviceSetPosition (lpcs, lpcs->sCapParms.dwMCIStartTime))
{
fCapturedOK = FALSE;
errorUpdateError (lpcs, IDS_CAP_MCI_CONTROL_ERROR);
statusUpdateStatus(lpcs, 0); // Clear status
goto EarlyExit;
}
}
//
// If we're compressing while capturing, warm up the compressor
//
#ifdef NEW_COMPMAN
if (lpcs->CompVars.hic)
{
if ( ! ICSeqCompressFrameStart (&lpcs->CompVars, lpcs->lpBitsInfo))
{
// !!! We're in trouble here!
dprintf("ICSeqCompressFrameStart failed !!!\n");
errorUpdateError (lpcs, IDS_CAP_COMPRESSOR_ERROR);
goto EarlyExit;
}
// HACK WARNING !!!
// Kludge, offset the lpBitsOut ptr
// Compman allocates the compress buffer too large by
// 2048 + 16 so we will still have room
// By stepping on 8 bytes we give ourselves room for a RIFF header
//
((LPBYTE)lpcs->CompVars.lpBitsOut) += 8;
assert(lpcs->CompVars.lpbiOut != NULL);
}
#endif
// -------------------------------------------------------
// Open the output file
// -------------------------------------------------------
if (lpcs->fCaptureFlags & CAP_fCapturingToDisk) {
if (!CapFileInit(lpcs))
{
errorUpdateError (lpcs, IDS_CAP_FILE_OPEN_ERROR);
goto EarlyExit;
}
} else {
AuxDebugEx (3, DEBUGLINE "Setting dwBytesPerSector to %d\r\n",DEFAULT_BYTESPERSECTOR);
lpcs->dwBytesPerSector=DEFAULT_BYTESPERSECTOR;
}
#ifdef JMK_HACK_TIMERS
// Allocate memory for logging capture results to the clipboard if requested
if (GetProfileIntA ("Avicap32", "ClipboardLogging", FALSE))
{
AuxDebugEx (2, DEBUGLINE "ClipboardLogging Enabled\r\n");
InitPerformanceCounters();
pcBegin(), pctWriteBase = pc.base;
hMemTimers = GlobalAlloc(GHND | GMEM_ZEROINIT,
sizeof(struct _timerriff) +
sizeof(struct _timerstuff) * CLIPBOARDLOGSIZE);
if (hMemTimers && ((DWORD)(pTimerRiff = GlobalLock (hMemTimers))))
;
else if (hMemTimers)
{
GlobalFree(hMemTimers);
pTimerRiff = 0;
pTimerStuff = 0;
hMemTimers = 0;
}
nTimerIndex = 0;
nSleepCount = 0;
} // if ClipboardLogging
#endif // JMK_HACK_TIMERS
// Make sure the parent has been repainted
//
UpdateWindow(lpcs->hwnd);
//
// call AVIInit() to get all the capture memory we will need
//
wError = IDS_CAP_AVI_INIT_ERROR;
lpcs->hCaptureEvent = CreateEvent(NULL, FALSE, FALSE, NULL);
if (lpcs->hCaptureEvent)
{
#ifdef CHICAGO
lpcs->hRing0CapEvt = OpenVxDHandle (lpcs->hCaptureEvent);
if ( ! lpcs->hRing0CapEvt)
CloseHandle (lpcs->hCaptureEvent), lpcs->hCaptureEvent = NULL;
else
#endif
wError = AVIInit(lpcs);
}
// if avifile init failed, cleanup and return error.
//
if (wError)
{
// Error in initalization - return
//
errorUpdateError (lpcs, wError);
AVIFini(lpcs);
AVIFileFini(lpcs, TRUE, TRUE);
statusUpdateStatus(lpcs, 0); // Clear status
goto EarlyExit;
}
// Click OK to capture string (must follow AVIInit)
//
LoadString(lpcs->hInst, IDS_CAP_SEQ_MSGSTART, ach, NUMELMS(ach));
wsprintf(achMsg, ach, (LPBYTE)lpcs->achFile);
// clear status
//
statusUpdateStatus(lpcs, 0);
// -------------------------------------------------------
// Ready to go, make the user click OK?
// -------------------------------------------------------
if (lpcs->sCapParms.fMakeUserHitOKToCapture && (lpcs->fCaptureFlags & CAP_fCapturingToDisk))
{
UINT idBtn;
idBtn = MessageBox (lpcs->hwnd, achMsg, TEXT(""),
MB_OKCANCEL | MB_ICONEXCLAMATION);
if (idBtn == IDCANCEL)
{
AVIFini(lpcs);
AVIFileFini (lpcs, TRUE, TRUE);
statusUpdateStatus (lpcs, 0);
goto EarlyExit;
}
}
// update the status, so the user knows how to stop
//
statusUpdateStatus(lpcs, IDS_CAP_SEQ_MSGSTOP);
UpdateWindow(lpcs->hwnd);
// this should be an ASSERT. After all, we turned the flag on at the
// top of the routine
//lpcs->fCaptureFlags |= CAP_fCapturingNow;
// query async key states to 'reset' them to current values
//
GetAsyncKeyState(lpcs->sCapParms.vKeyAbort);
GetAsyncKeyState(VK_ESCAPE);
GetAsyncKeyState(VK_LBUTTON);
GetAsyncKeyState(VK_RBUTTON);
// Insert the digitization time
// strings written to the file should be ascii, since this is
// an ascii file format.
//
//
// no point in pulling in the whole C runtime just to get a silly
// timestamp, so we just cook the system time into ascii right here.
//
{
SYSTEMTIME time;
// Note: both szDay and szMonth are explicitly null-terminated by virtue
// of being C strings ... "xxx"
static char szDay[] = "Sun\0Mon\0Tue\0Wed\0Thu\0Fri\0Sat";
#define DAYLENGTH (sizeof(szDay)/7)
static char szMonth[] = "Jan\0Feb\0Mar\0Apr\0May\0Jun\0Jul\0Aug\0Sep\0Oct\0Nov\0Dec";
#define MONTHLENGTH (sizeof(szMonth)/12)
char sz[30];
GetLocalTime (&time);
// note: GetLocalTime returns months in range 1-12
// returns days in range 0-6
//example: Fri Apr 29 8:25:12 1994
wsprintfA(sz, "%s %s %2d %2d:%02d:%02d %4d",
szDay + time.wDayOfWeek * DAYLENGTH,
szMonth-MONTHLENGTH + time.wMonth * MONTHLENGTH,
time.wDay, time.wHour, time.wMinute, time.wSecond, time.wYear);
cic.fccInfoID = mmioFOURCC ('I','D','I','T');
cic.lpData = sz;
cic.cbData = 25; // WARNING: this length is static.
SetInfoChunk (lpcs, &cic);
}
// -------------------------------------------------------
// Start MCI, Audio, and video streams
// -------------------------------------------------------
// Callback will preroll, then return on frame accurate postion
// The 1 indicates recording is about to start
// Callback can return FALSE to exit without capturing
//
if (lpcs->CallbackOnControl &&
!lpcs->CallbackOnControl(lpcs->hwnd, CONTROLCALLBACK_PREROLL))
{
AVIFini(lpcs);
AVIFileFini(lpcs, TRUE, TRUE);
statusUpdateStatus(lpcs, 0);
goto EarlyExit;
}
dwOldPrio = GetThreadPriority(GetCurrentThread());
if (dwOldPrio != THREAD_PRIORITY_HIGHEST)
SetThreadPriority (GetCurrentThread(), THREAD_PRIORITY_HIGHEST);
#ifdef JMK_HACK_TIMERS
if (pTimerRiff)
{
UINT ii;
pTimerRiff->fccRIFF = RIFFTYPE('RIFF'); //MAKEFOURCC('R','I','F','F');
pTimerRiff->cbTotal = sizeof(struct _timerriff) - 8 +
sizeof(struct _timerstuff) * CLIPBOARDLOGSIZE;
pTimerRiff->fccJMKD = RIFFTYPE('JMKD'); //MAKEFOURCC('J','M','K','D');
pTimerRiff->fccVCHD = RIFFTYPE('VCHD'); //MAKEFOURCC('V','C','H','D');
pTimerRiff->cbVCHD = sizeof(struct _vchd);
pTimerRiff->vchd.nPrio = GetThreadPriority(GetCurrentThread());
pTimerRiff->vchd.bmih = lpcs->lpBitsInfo->bmiHeader;
pTimerRiff->vchd.cap = lpcs->sCapParms;
pTimerRiff->vchd.dwDropFramesAppended = 0;
pTimerRiff->vchd.dwDropFramesNotAppended = 0;
pTimerRiff->vchd.dwTimerFrequency = pcGetTickRate();
for (ii = 0; ii < NUMELMS(pTimerRiff->vchd.atvh); ++ii)
{
if (lpcs->alpVideoHdr[ii])
{
struct _thkvideohdr * ptvh = (LPVOID)lpcs->alpVideoHdr[ii];
#ifndef CHICAGO
assert (sizeof(CAPVIDEOHDR) == sizeof(*ptvh));
#endif
pTimerRiff->vchd.atvh[ii] = *ptvh;
pTimerRiff->vchd.nMaxVideoBuffers = ii;
}
}
pTimerRiff->fccChunk = RIFFTYPE('VCAP'); //MAKEFOURCC('V','C','A','P');
pTimerRiff->cbChunk = pTimerRiff->cbTotal - sizeof(*pTimerRiff);
pTimerStuff = (LPVOID)(pTimerRiff + 1);
pCurTimerStuff = &pTimerStuff[0];
} // fClipboardLogging
#endif // JMK_HACK_TIMERS
// make sure that the fat is loaded before we begin capturing
//
AVIPreloadFat (lpcs);
// start the MCI device
//
if (lpcs->sCapParms.fMCIControl)
MCIDevicePlay (lpcs);
dwTimeStarted = timeGetTime();
// start audio & video streams
//
if (lpcs->sCapParms.fCaptureAudio)
waveInStart(lpcs->hWaveIn);
videoStreamStart(lpcs->hVideoIn);
// -------------------------------------------------------
// MAIN CAPTURE LOOP
// -------------------------------------------------------
fCapturedOK=TRUE;
fStopping = FALSE; // TRUE when we need to stop
fStopped = FALSE; // TRUE if drivers notified we have stopped
lpcs->dwTimeElapsedMS = 0;
assert (lpcs->iNextVideo == 0);
if (lpcs->sCapParms.fCaptureAudio) {
assert (lpcs->iNextWave == 0);
lpWaveHdr = lpcs->alpWaveHdr[lpcs->iNextWave];
// lpWaveHdr is only interesting when we capture audio
}
lpVidHdr = lpcs->alpVideoHdr[lpcs->iNextVideo];
DPF("Start of main capture loop");
for (;;)
{
// The INTEL driver uses the GetError message to
// process buffers, so call it often...
// FIX JAYBO videoStreamGetError (lpcs->hVideoIn, &dwStreamError, &dwDriverDropCount);
// if there are no buffers to process, we either wait
// or leave the loop forever (depending on whether we expect
// more buffers to be done in the future)
//
if (!(lpVidHdr->dwFlags & VHDR_DONE) &&
!(lpcs->sCapParms.fCaptureAudio
&& (lpWaveHdr->dwFlags & WHDR_DONE)))
{
if (fStopped)
break;
#ifdef JMK_HACK_TIMERS
if (pCurTimerStuff)
{
pCurTimerStuff->nSleepCount = ++nSleepCount;
pCurTimerStuff->dwSleepBegin = pcGetTicks();
}
#endif
AuxDebugEx(2,DEBUGLINE "***** Waiting for something interesting to happen while capturing\r\n");
WaitForSingleObject (lpcs->hCaptureEvent, 300);
#ifdef JMK_HACK_TIMERS
if (pCurTimerStuff)
{
pCurTimerStuff->dwSleepEnd = pcGetTicks();
}
#endif
}
// What time is it?
lpcs->dwTimeElapsedMS = timeGetTime() - dwTimeStarted;
// -------------------------------------------------------
// Is video buffer ready to be written?
// -------------------------------------------------------
if ((DWORD)(fStopping = ProcessNextVideoBuffer (lpcs,
fStopping,
&wError,
&lpvhDraw, // captured frame to draw if time permits
&bVideoWritePending))) // TRUE if Write pending on lpvhDraw
{
AuxDebugEx (1, DEBUGLINE "ProcessVideo stopping\r\n");
if (wError)
{
AuxDebugEx (1, DEBUGLINE "ProcessVideo return error %d\r\n", wError);
errorUpdateError (lpcs, wError);
fCapturedOK = FALSE;
break;
}
}
lpVidHdr = lpcs->alpVideoHdr[lpcs->iNextVideo];
// if there is still more time, (or at least every 100 frames)
// show status if we're not ending the capture
//
if (!fStopping &&
(lpcs->fCaptureFlags & CAP_fCapturingToDisk) &&
(!(lpVidHdr->dwFlags & VHDR_DONE) ||
(lpcs->dwVideoChunkCount && (lpcs->dwVideoChunkCount % 100 == 0))))
{
// Captured %ld frames (Dropped %ld) %d.%03d sec. Hit Escape to Stop
//
statusUpdateStatus(lpcs, IDS_CAP_STAT_VIDEOCURRENT,
lpcs->dwVideoChunkCount,
lpcs->dwFramesDropped,
(UINT)(lpcs->dwTimeElapsedMS/1000),
(UINT)(lpcs->dwTimeElapsedMS%1000));
}
// If the yield callback returns FALSE, abort
//
if (lpcs->CallbackOnYield && !lpcs->CallbackOnYield (lpcs->hwnd))
fStopping = TRUE;
#if 0 // this is a 16 bit ism??
// Don't do peekMessage yield for ACM
if (lpcs->sCapParms.fYield) {
MSG msg;
if (PeekMessage(&msg,NULL,0,0,PM_REMOVE)) {
// Kludge to get rid of timers from lpcs->hwnd
if (msg.message == WM_TIMER && msg.hwnd == lpcs->hwnd)
;
else {
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
}
#endif
// Outside routine is handling when to stop
// The CONTROLCALLBACK_CAPTURING indicates we're asking when to stop
//
if (lpcs->CallbackOnControl &&
!lpcs->CallbackOnControl (lpcs->hwnd, CONTROLCALLBACK_CAPTURING))
fStopping = TRUE;
// -------------------------------------------------------
// Is audio buffer ready to be written?
// -------------------------------------------------------
if (lpcs->sCapParms.fCaptureAudio) {
if ((DWORD)(fStopping = ProcessAudioBuffers (lpcs, fStopping, &wError)))
{
AuxDebugEx (1, DEBUGLINE "ProcessAudio stopping\r\n");
if (wError)
{
AuxDebugEx (1, DEBUGLINE "ProcessAudio return error %d\r\n", wError);
errorUpdateError (lpcs, wError);
fCapturedOK = FALSE;
break;
}
}
lpWaveHdr = lpcs->alpWaveHdr[lpcs->iNextWave];
}
// if we are not writing the frame async, we can put the video buffer
// back on the video driver's queue now
//
if (lpvhDraw)
{
// if the next video header is not ready yet, and
// we have no outstanding io buffers (in async mode) draw
// the current one
//
if ( !(lpVidHdr->dwFlags & VHDR_DONE) &&
(!lpcs->pAsync || (lpcs->iNextAsync == lpcs->iLastAsync)) &&
lpvhDraw->dwBytesUsed)
{
AuxDebugEx (4, DEBUGLINE "time enough to draw!\r\n");
if (fTryToPaintAgain &&
lpcs->dwVideoChunkCount &&
lpvhDraw->dwFlags & VHDR_KEYFRAME)
{
fTryToPaintAgain = DrawDibDraw(lpcs->hdd, hdc,
0, 0,
rcDrawRect.right - rcDrawRect.left,
rcDrawRect.bottom - rcDrawRect.top,
/*lpcs->dxBits, lpcs->dyBits, */
(LPBITMAPINFOHEADER)lpcs->lpBitsInfo,
lpvhDraw->lpData, 0, 0, -1, -1,
DDF_SAME_HDC | DDF_SAME_DIB | DDF_SAME_SIZE);
}
}
// if there is not a write pending for the draw frame
// put it back into the video drivers queue now
//
if ( ! bVideoWritePending)
{
AuxDebugEx(3, DEBUGLINE "Queueing video buffer, lpvh=%8x", lpvhDraw);
// return the emptied buffer to the que
//
#if defined CHICAGO
if (vidxAddBuffer(lpcs->hVideoIn, lpvhDraw, sizeof (VIDEOHDR)))
#else
if (videoStreamAddBuffer(lpcs->hVideoIn, lpvhDraw, sizeof (VIDEOHDR)))
#endif
{
AuxDebugEx (2, DEBUGLINE "Failed to Queue Video buffer %08x\r\n", lpvhDraw);
errorUpdateError (lpcs, IDS_CAP_VIDEO_ADD_ERROR);
fCapturedOK = FALSE;
fStopping = TRUE;
break;
}
}
}
// ------------------------------------------------------------
// Any completed I/O buffers?
// ------------------------------------------------------------
if (lpcs->pAsync)
if ((DWORD)(fStopping = ProcessAsyncIOBuffers (lpcs, fStopping, &wError)))
{
if (wError)
{
errorUpdateError (lpcs, wError);
fCapturedOK = FALSE;
break;
}
}
// -------------------------------------------------------
// is there any reason to stop?
// -------------------------------------------------------
if (!fStopping)
{
if (lpcs->sCapParms.vKeyAbort &&
(GetAsyncKeyState(lpcs->sCapParms.vKeyAbort & 0x00ff) & 0x0001))
{
BOOL fT = TRUE;
if (lpcs->sCapParms.vKeyAbort & 0x8000) // Ctrl?
fT = fT && (GetAsyncKeyState(VK_CONTROL) & 0x8000);
if (lpcs->sCapParms.vKeyAbort & 0x4000) // Shift?
fT = fT && (GetAsyncKeyState(VK_SHIFT) & 0x8000);
fStopping = fT; // User aborts
}
if (lpcs->sCapParms.fAbortLeftMouse && (GetAsyncKeyState(VK_LBUTTON) & 0x0001))
fStopping = TRUE; // User aborts
if (lpcs->sCapParms.fAbortRightMouse && (GetAsyncKeyState(VK_RBUTTON) & 0x0001))
fStopping = TRUE; // User aborts
if ((lpcs->fCaptureFlags & CAP_fAbortCapture) || (lpcs->fCaptureFlags & CAP_fStopCapture))
fStopping = TRUE; // Somebody above wants us to quit
if (lpcs->dwTimeElapsedMS > dwTimeToStop)
fStopping = TRUE; // all done
#ifdef DEBUG
if (fStopping)
AuxDebugEx (1, DEBUGLINE "user stop\r\n");
#endif
}
// -------------------------------------------------------
// Tell all the devices to stop
// -------------------------------------------------------
if (fStopping)
{
if ( ! fStopped)
{
fStopped = TRUE;
DSTATUS(lpcs, "Stopping....");
if (lpcs->sCapParms.fCaptureAudio)
{
DSTATUS(lpcs, "Stopping Audio");
waveInStop(lpcs->hWaveIn);
}
DSTATUS(lpcs, "Stopping Video");
videoStreamStop(lpcs->hVideoIn); // Stop everybody
dwTimeStopped = timeGetTime ();
if (lpcs->sCapParms.fMCIControl)
{
DSTATUS(lpcs, "Stopping MCI");
MCIDevicePause (lpcs);
}
DSTATUS(lpcs, "Stopped");
// Force cursor back to hourglass
//
SetCursor(lpcs->hWaitCursor);
}
// "Finished capture, now writing frame %ld"
//
if (fCapturedOK)
statusUpdateStatus(lpcs, IDS_CAP_STAT_CAP_FINI, lpcs->dwVideoChunkCount);
else
{
statusUpdateStatus(lpcs, IDS_CAP_RECORDING_ERROR2);
break;
}
// Wait for all the async IO to complete ??
//
}
// -------------------------------------------------------
// END OF MAIN CAPTURE LOOP
// -------------------------------------------------------
}
DPF("End of main capture loop");
// eat any keys that have been pressed
//
while(GetKey(FALSE))
;
// flush stuff to disk, close everything etc.
//
AVIFini(lpcs);
AVIFileFini(lpcs, TRUE, !fCapturedOK);
// This is the corrected capture duration, based on audio samples
lpcs->dwTimeElapsedMS = lpcs->dwActualMicroSecPerFrame *
lpcs->dwVideoChunkCount / 1000;
// update the status line with information about the completed
// capture, or with erroor information
//
ShowCompletionStatus (lpcs, fCapturedOK);
EarlyExit:
//
// If we we're compressing while capturing, close it down
//
#ifdef NEW_COMPMAN
if (lpcs->CompVars.hic) {
// Kludge, reset the lpBitsOut pointer
if (lpcs->CompVars.lpBitsOut)
((LPBYTE) lpcs->CompVars.lpBitsOut) -= 8;
ICSeqCompressFrameEnd(&lpcs->CompVars);
}
#endif
if (fTryToPaint) {
if (hpalT)
SelectPalette(hdc, hpalT, FALSE);
ReleaseDC (lpcs->hwnd, hdc);
}
if (lpcs->sCapParms.fMCIControl)
MCIDeviceClose (lpcs);
// Let the user see where capture stopped
if ((!lpcs->fLiveWindow) && (!lpcs->fOverlayWindow))
videoFrame( lpcs->hVideoIn, &lpcs->VidHdr );
InvalidateRect( lpcs->hwnd, NULL, TRUE);
SetThreadPriority (GetCurrentThread(), dwOldPrio);
SetCursor(hOldCursor);
lpcs->fCapFileExists = (lpcs->dwReturn == DV_ERR_OK);
lpcs->fCaptureFlags &= ~CAP_fCapturingNow;
statusUpdateStatus(lpcs, IDS_CAP_END); // Always the last message
#ifdef JMK_HACK_TIMERS
if (pTimerRiff)
{
UINT ii;
UINT kk;
LPSTR psz;
HGLOBAL hMem;
kk = (lpcs->dwVideoChunkCount >= CLIPBOARDLOGSIZE) ?
CLIPBOARDLOGSIZE : nTimerIndex;
hMem = GlobalAlloc (GHND, (16 * 5 + 2) * kk + 80);
if (hMem && ((DWORD)(psz = GlobalLock (hMem))))
{
pTimerRiff->vchd.dwFramesCaptured = lpcs->dwVideoChunkCount;
pTimerRiff->vchd.dwFramesDropped = lpcs->dwFramesDropped;
pTimerRiff->cbTotal = sizeof(struct _timerriff) - 8 +
sizeof(struct _timerstuff) * nTimerIndex;
pTimerRiff->cbChunk = pTimerRiff->cbTotal - sizeof(*pTimerRiff);
lstrcpyA(psz, "Slot#, VideoIndex, ExpectedTime, DriverTime, AccumulatedDummyFrames, CurrentAppendedDummies");
for (ii = 0; ii < kk; ++ii)
{
psz += lstrlenA(psz);
wsprintfA(psz, "\r\n%d, %ld, %ld, %ld, %d, %d",
ii,
pTimerStuff[ii].dwVideoChunkCount,
pTimerStuff[ii].dwFrameTickTime,
pTimerStuff[ii].dwFrameStampTime,
pTimerStuff[ii].nDummyFrames,
pTimerStuff[ii].nFramesAppended
);
}
GlobalUnlock (hMem);
GlobalUnlock (hMemTimers);
if (OpenClipboard (lpcs->hwnd))
{
EmptyClipboard ();
SetClipboardData (CF_RIFF, hMemTimers);
SetClipboardData (CF_TEXT, hMem);
CloseClipboard ();
}
else
{
GlobalFree (hMem);
GlobalFree (hMemTimers);
}
}
else
{
// Failed to allocate or lock hMem. Cleanup.
//
if (hMem)
GlobalFree(hMem);
// Free off the timer block. (We have not set the
// clipboard data.)
//
if (hMemTimers)
{
GlobalUnlock(hMemTimers);
GlobalFree(hMemTimers);
}
}
hMemTimers = NULL;
pTimerRiff = NULL;
pTimerStuff = NULL;
pCurTimerStuff = NULL;
}
#endif
return;
}
// Returns TRUE if the capture task was created, or
// capture completed OK.
BOOL AVICapture (LPCAPSTREAM lpcs)
{
CAPINFOCHUNK cic;
void (WINAPI _LOADDS * pfnCapture) (LPCAPSTREAM lpcs);
if (lpcs->fCaptureFlags & CAP_fCapturingNow) {
AuxDebugEx(4, DEBUGLINE "rejecting capture as previous capture still running\r\n");
return IDS_CAP_VIDEO_OPEN_ERROR;
}
// if there is a previous capture thread, wait for it to finish and
// clean it up
// -it has set fCapturingNow to FALSE, so it will end 'soon' !
if (lpcs->hThreadCapture) {
AuxDebugEx(4, DEBUGLINE "Starting capture while previous capture thread still active\r\n");
WaitForSingleObject(lpcs->hThreadCapture, INFINITE);
CloseHandle(lpcs->hThreadCapture);
lpcs->hThreadCapture = NULL;
}
// Turn off the STOP and ABORT capture bits
lpcs->fCaptureFlags &= ~(CAP_fStopCapture | CAP_fAbortCapture);
lpcs->dwReturn = 0;
#if DONT_CLEAR_SMPTE_JAYBO
// Prior to Win95, we always cleared out old SMPTE chunks,
// but since Adobe may have created a chunk manually, don't
// zap existing chunks.
cic.fccInfoID = mmioFOURCC ('I','S','M','T');
cic.lpData = NULL;
cic.cbData = 0;
SetInfoChunk (lpcs, &cic);
#endif
// And get ready to write a SMPTE info chunk
if (lpcs->sCapParms.fMCIControl) {
// create SMPTE string
CHAR szSMPTE[40]; // must write ansi
TimeMSToSMPTE (lpcs->sCapParms.dwMCIStartTime, szSMPTE);
cic.lpData = szSMPTE;
cic.cbData = lstrlenA(szSMPTE) + 1;
cic.fccInfoID = mmioFOURCC ('I','S','M','T');
SetInfoChunk (lpcs, &cic);
}
// set pfnCapture to point to the capture function of choice.
// Use an MCI device to do step capture capture???
// assume No MCI device, just a normal streaming capture
//
pfnCapture = AVICapture1;
if (lpcs->sCapParms.fStepMCIDevice && lpcs->sCapParms.fMCIControl)
pfnCapture = MCIStepCapture;
// if the fYield flag is true, create a thread to do the
// capture loop. otherwise do the capture loop inline
//
if (lpcs->sCapParms.fYield)
{
DWORD tid;
lpcs->fCaptureFlags |= CAP_fCapturingNow;
// future operations on this thread are now locked out.
// we must turn this flag off if the thread creation fails
lpcs->hThreadCapture = CreateThread (NULL,
0,
(LPTHREAD_START_ROUTINE) pfnCapture,
lpcs,
0,
&tid);
// if thread creation failed, turn off the capturing flag
//
if ( ! lpcs->hThreadCapture) {
AuxDebugEx(1,"Failed to create capture thread");
lpcs->fCaptureFlags &= ~CAP_fCapturingNow;
}
return (lpcs->hThreadCapture != NULL);
}
else
{
pfnCapture (lpcs);
return (0 == lpcs->dwReturn);
}
}