windows-nt/Source/XPSP1/NT/multimedia/media/avi/mciavi32/aviread.c

543 lines
13 KiB
C
Raw Permalink Normal View History

2020-09-26 03:20:57 -05:00
/*************************************************************************
* Copyright (C) Microsoft Corporation 1992. All rights reserved.
*
*************************************************************************/
/*
* aviread.c: read blocks from the avi file (using worker thread).
* Only built in WIN32 case.
*/
//#define AVIREAD
#ifdef AVIREAD
#include <windows.h>
#include <mmsystem.h>
#include <ntavi.h> // This must be included, for both versions
#include <string.h> // needed for memmove in nt
#include <mmddk.h>
#include <memory.h>
#include "common.h"
#include "ntaviprt.h"
#include "aviread.h"
#include "aviffmt.h"
#include "graphic.h"
/*
* overview of operation:
*
* creation of a avird object (via avird_startread) creates an avird_header
* data structure and a worker thread. The data structure is protected by
* a critical section, and contains two semaphores. the semEmpty semaphore is
* initialised to the number of buffers allocated (normally 2), and the
* semFull semaphore is initialised to 0 (there are initially no full buffers).
*
* The worker thread then loops waiting on semEmpty to get empty buffers,
* and once it finds them, filling them and signalling via semFull that they
* are ready. It fills them by callbacks to a AVIRD_FUNC function that
* we were given a pointer to on object creation.
*
* Getting a buffer via avird_getnextbuffer waits on semFull until there
* are full buffers, and returns the first on the list, after moving
* it to the 'in use' list. The caller
* will use/play the data in the buffer, and then call avird_emptybuffer:
* - this finds the buffer on the 'in use' list, moves it to the
* 'empty' list and then signals the worker thread via
* semEmpty.
*
* The worker thread checks the object state each time it is woken up. If this
* state is 'closing' (bOK == FALSE), the thread frees all memory and exits.
* avird_endread changes the object state and signals semEmpty to wake up the
* worker thread.
*/
/*
* each buffer is represented by one of these headers
*/
typedef struct avird_buffer {
/*
* size of the buffer in bytes
*/
long lSize;
/* size of read data in bytes */
long lDataSize;
/*
* pointer to the next buffer in this state
*/
struct avird_buffer * pNextBuffer;
/* FALSE if buffer read failed */
BOOL bOK;
/* request sequence */
int nSeq;
/*
* pointer to the actual block of buffer data
*/
PBYTE pData;
} AVIRD_BUFFER, * PAVIRD_BUFFER;
/* handles to HAVIRD are pointers to this data structure, but the
* contents of the struct are known only within this module
*/
typedef struct avird_header {
/*
* always hold the critical section before checking/changing the
* object state or any buffer state
*/
CRITICAL_SECTION critsec;
/*
* the count of this semaphore is the count of empty buffers
* waiting to be picked up by the worker thread
*/
HANDLE semEmpty;
/*
* the count of this semaphore is the count of full buffers waiting
* to be picked up by the caller.
*/
HANDLE semFull;
/* object state - FALSE indicates close-down request. */
BOOL bOK;
/* pointer to list of buffer headers ready to be filled */
PAVIRD_BUFFER pEmpty;
/* pointer to a list of buffer headers in use by the client */
PAVIRD_BUFFER pInUse;
/*
* pointer to an ordered list of buffer headers ready to be
* picked up by the client.
*/
PAVIRD_BUFFER pFull;
/*
* function to call to fill a buffer
*/
AVIRD_FUNC pFunc;
/* instance arg to pass to pFunc() */
DWORD dwInstanceData;
/* size of next buffer to be read */
long lNextSize;
/* request sequence */
int nNext;
/* total in sequence */
int nBlocks;
} AVIRD_HEADER, * PAVIRD_HEADER;
/* number of buffers to queue up */
#define MAX_Q_BUFS 4
/*
* worker thread function
*/
DWORD avird_worker(LPVOID lpvThreadData);
/*
* function to delete whole AVIRD_HEADER data structure.
*/
void avird_freeall(PAVIRD_HEADER phdr);
/*
* start an avird operation and return a handle to use in subsequent
* calls. This will cause an asynchronous read (achieved using a separate
* thread) to start reading the next few buffers
*/
HAVIRD
avird_startread(AVIRD_FUNC func, DWORD dwInstanceData, long lFirstSize,
int nFirst, int nBlocks)
{
PAVIRD_HEADER phdr;
PAVIRD_BUFFER pbuf;
int i;
HANDLE hThread;
DWORD dwThreadId;
int nBufferSize;
/*
* allocate and init the header
*/
phdr = (PAVIRD_HEADER) LocalLock(LocalAlloc(LHND, sizeof(AVIRD_HEADER)));
if (phdr == NULL) {
return(NULL);
}
InitializeCriticalSection(&phdr->critsec);
phdr->semEmpty = CreateSemaphore(NULL, MAX_Q_BUFS, MAX_Q_BUFS, NULL);
phdr->semFull = CreateSemaphore(NULL, 0, MAX_Q_BUFS, NULL);
phdr->bOK = TRUE;
phdr->pInUse = NULL;
phdr->pFull = NULL;
phdr->pEmpty = NULL;
phdr->pFunc = func;
phdr->dwInstanceData = dwInstanceData;
phdr->lNextSize = lFirstSize;
phdr->nNext = nFirst;
phdr->nBlocks = nBlocks;
/*
* round sizes up to 2k to reduce cost of small increases
*/
nBufferSize = (lFirstSize + 2047) & ~2047;
/*
* allocate and init the buffers
*/
for (i = 0; i < MAX_Q_BUFS; i++) {
pbuf = (PAVIRD_BUFFER) LocalLock(LocalAlloc(LHND, sizeof(AVIRD_BUFFER)));
pbuf->lSize = nBufferSize;
pbuf->pData = (PBYTE) LocalLock(LocalAlloc(LHND, pbuf->lSize));
pbuf->pNextBuffer = phdr->pEmpty;
phdr->pEmpty = pbuf;
}
/*
* create the worker thread
*/
hThread = CreateThread(NULL, 0, avird_worker, (LPVOID)phdr, 0, &dwThreadId);
if (hThread) {
/* thread was created ok */
CloseHandle(hThread);
return( phdr);
} else {
avird_freeall(phdr);
return(NULL);
}
}
/*
* return the next buffer from an HAVIRD object.
*/
PBYTE
avird_getnextbuffer(HAVIRD havird, long * plSize)
{
PAVIRD_HEADER phdr = havird;
PAVIRD_BUFFER pbuf;
/* wait for a full buffer -report if actual wait needed*/
if (WaitForSingleObject(phdr->semFull, 0) == WAIT_TIMEOUT) {
DPF(("..waiting.."));
WaitForSingleObject(phdr->semFull, INFINITE);
}
/* always hold critsec before messing with queues */
EnterCriticalSection(&phdr->critsec);
/* de-queue first full buffer and place on InUse queue */
pbuf = phdr->pFull;
phdr->pFull = pbuf->pNextBuffer;
pbuf->pNextBuffer = phdr->pInUse;
phdr->pInUse = pbuf;
/* finished with critical section */
LeaveCriticalSection(&phdr->critsec);
if (!pbuf->bOK) {
/* buffer read failed */
DPF(("reporting read failure on %d\n", pbuf->nSeq));
if (plSize) {
*plSize = 0;
}
return(NULL);
}
/* return size of buffer if requested */
if (plSize) {
*plSize = pbuf->lDataSize;
}
return(pbuf->pData);
}
/*
* return to the queue a buffer that has been finished with (is now empty)
*
* causes the worker thread to be woken up and to start filling the buffer
* again.
*/
void
avird_emptybuffer(HAVIRD havird, PBYTE pBuffer)
{
PAVIRD_HEADER phdr = havird;
PAVIRD_BUFFER pbuf, pprev;
/* always get the critsec before messing with queues */
EnterCriticalSection(&phdr->critsec);
pprev = NULL;
for (pbuf = phdr->pInUse; pbuf != NULL; pbuf = pbuf->pNextBuffer) {
if (pbuf->pData == pBuffer) {
/* this is the buffer */
break;
}
pprev = pbuf;
}
if (pbuf != NULL) {
/* de-queue from InUse and place on empty q */
if (pprev) {
pprev->pNextBuffer = pbuf->pNextBuffer;
} else {
phdr->pInUse = pbuf->pNextBuffer;
}
pbuf->pNextBuffer = phdr->pEmpty;
phdr->pEmpty = pbuf;
/* mark as not validly read */
pbuf->bOK = FALSE;
/* signal that there is another buffer to fill */
ReleaseSemaphore(phdr->semEmpty, 1, NULL);
} else {
DPF(("buffer 0x%x not found on InUse list\n", pBuffer));
}
LeaveCriticalSection(&phdr->critsec);
}
/*
* delete an avird object. the worker thread will be stopped and all
* data allocated will be freed. The HAVIRD handle is no longer valid after
* this call.
*/
void
avird_endread(HAVIRD havird)
{
PAVIRD_HEADER phdr = havird;
DPF(("killing an avird object\n"));
/* get the critsec before messing with states */
EnterCriticalSection(&phdr->critsec);
/* tell the worker thread to do all the work */
phdr->bOK = FALSE;
/* wake up the worker thread */
ReleaseSemaphore(phdr->semEmpty, 1, NULL);
/*
* we must hold the critsec past the semaphore signal: if we
* release the critsec first, the worker thread might see the
* state change before we have signalled the semaphore. He would
* then potentially have destroyed the semaphore AND freed the
* AVIRD_HEADER structure by the time we tried to signal the
* semaphore. This way, we are sure that until we release the
* critsec, everything is still valid
*/
LeaveCriticalSection(&phdr->critsec);
/* all done - phdr now may not exist */
}
/*
* worker thread function.
*
* loop waiting for semEmpty to tell us there are empty buffers. When
* we see one, fill it with phdr->pFunc and move it to the
* full queue. Each time we are woken up, check the state. If it
* changes to false, delete the whole thing and exit.
*
* the argument we are passed is the PAVIRD_HEADER.
*/
DWORD
avird_worker(LPVOID lpvThreadData)
{
PAVIRD_HEADER phdr = (PAVIRD_HEADER) lpvThreadData;
PAVIRD_BUFFER pbuf, pprev;
long lNextSize;
HANDLE hmem;
DPF(("Worker %d started\n", GetCurrentThreadId()));
for (; ;) {
/* wait for an empty buffer (or state change) */
WaitForSingleObject(phdr->semEmpty, INFINITE);
/* get the critical section before touching the state, queues */
EnterCriticalSection(&phdr->critsec);
if (phdr->bOK == FALSE) {
/* all over bar the shouting */
DPF(("%d exiting\n", GetCurrentThreadId()));
avird_freeall(phdr);
ExitThread(0);
}
/* dequeue the first empty buffer */
pbuf = phdr->pEmpty;
Assert(pbuf != NULL);
phdr->pEmpty = pbuf->pNextBuffer;
lNextSize = phdr->lNextSize;
pbuf->nSeq = phdr->nNext++;
if (pbuf->nSeq < phdr->nBlocks) {
/* we can now release the critsec until we need to re-Q the filled buf*/
LeaveCriticalSection(&phdr->critsec);
/* resize the buffer if not big enough */
if (pbuf->lSize < lNextSize) {
hmem = LocalHandle(pbuf->pData);
LocalUnlock(hmem);
LocalFree(hmem);
pbuf->lSize = ((lNextSize + 2047) & ~2047);
pbuf->pData = LocalLock(LocalAlloc(LHND, pbuf->lSize));
}
/* record the data content of the buffer */
pbuf->lDataSize = lNextSize;
/* call the filler function */
if ((*phdr->pFunc)(pbuf->pData, phdr->dwInstanceData, lNextSize,
&lNextSize)) {
pbuf->bOK = TRUE;
} else {
DPF(("filler reported failure on %d\n", pbuf->nSeq));
}
/* get the critsec before messing with q's or states */
EnterCriticalSection(&phdr->critsec);
/* size for next read */
phdr->lNextSize = lNextSize;
}
/* place buffer at end of Full queue */
if (phdr->pFull == NULL) {
phdr->pFull = pbuf;
} else {
for (pprev = phdr->pFull; pprev->pNextBuffer != NULL; ) {
pprev = pprev->pNextBuffer;
}
pprev->pNextBuffer = pbuf;
}
pbuf->pNextBuffer = NULL;
LeaveCriticalSection(&phdr->critsec);
/* signal calling thread that there's another buffer for him */
ReleaseSemaphore(phdr->semFull, 1, NULL);
}
/* silence compiler */
return (0);
}
/*
* free one buffer and buffer header
*/
void
avird_freebuffer(PAVIRD_BUFFER pbuf)
{
HANDLE hmem;
hmem = LocalHandle( (PSTR)pbuf->pData);
LocalUnlock(hmem);
LocalFree(hmem);
hmem = LocalHandle( (PSTR)pbuf);
LocalUnlock(hmem);
LocalFree(hmem);
}
/*
* function to delete whole AVIRD_HEADER data structure.
*
* called on calling thread if start-up fails, or on worker thread if
* asked to shutdown.
*/
void
avird_freeall(PAVIRD_HEADER phdr)
{
PAVIRD_BUFFER pbuf, pnext;
HANDLE hmem;
if (phdr->semEmpty) {
CloseHandle(phdr->semEmpty);
}
if (phdr->semEmpty) {
CloseHandle(phdr->semFull);
}
DeleteCriticalSection(&phdr->critsec);
for (pbuf = phdr->pInUse; pbuf != NULL; pbuf = pnext) {
DPF(("In Use buffers at EndRead\n"));
pnext = pbuf->pNextBuffer;
avird_freebuffer(pbuf);
}
for (pbuf = phdr->pEmpty; pbuf != NULL; pbuf = pnext) {
pnext = pbuf->pNextBuffer;
avird_freebuffer(pbuf);
}
for (pbuf = phdr->pFull; pbuf != NULL; pbuf = pnext) {
pnext = pbuf->pNextBuffer;
avird_freebuffer(pbuf);
}
hmem = LocalHandle((PSTR) phdr);
LocalUnlock(hmem);
LocalFree(hmem);
}
#endif //AVIREAD