223 lines
6.0 KiB
C
223 lines
6.0 KiB
C
|
/************************************************************/
|
|||
|
/* Windows Write, Copyright 1985-1992 Microsoft Corporation */
|
|||
|
/************************************************************/
|
|||
|
|
|||
|
#define NOCLIPBOARD
|
|||
|
#define NOGDICAPMASKS
|
|||
|
#define NOCTLMGR
|
|||
|
#define NOVIRTUALKEYCODES
|
|||
|
#define NOWINMESSAGES
|
|||
|
#define NOWINSTYLES
|
|||
|
#define NOSYSMETRICS
|
|||
|
#define NOMENUS
|
|||
|
#include <windows.h>
|
|||
|
#include "mw.h"
|
|||
|
#include "doslib.h"
|
|||
|
#include "cmddefs.h"
|
|||
|
#include "docdefs.h"
|
|||
|
#include "filedefs.h"
|
|||
|
#include "str.h"
|
|||
|
#include "debug.h"
|
|||
|
|
|||
|
extern int vfnWriting;
|
|||
|
|
|||
|
|
|||
|
#define IibpHash(fn,pn) ((int) ((fn + 1) * (pn + 1)) & 077777) % iibpHashMax
|
|||
|
|
|||
|
#define FcMin(a,b) CpMin(a,b)
|
|||
|
|
|||
|
extern CHAR *rgibpHash;
|
|||
|
extern int vfSysFull;
|
|||
|
extern struct BPS *mpibpbps;
|
|||
|
extern struct FCB (**hpfnfcb)[];
|
|||
|
extern typeTS tsMruBps;
|
|||
|
extern CHAR (*rgbp)[cbSector];
|
|||
|
#ifdef CKSM
|
|||
|
#ifdef DEBUG
|
|||
|
extern unsigned (**hpibpcksm) [];
|
|||
|
#endif
|
|||
|
#endif
|
|||
|
|
|||
|
/* WriteDirtyPages kicks out of memory as much of the previous files as it can
|
|||
|
in order to fill the page buffers with a new file. This is called on
|
|||
|
every transfer load of a file. */
|
|||
|
WriteDirtyPages()
|
|||
|
{/*
|
|||
|
Description: Cleans the buffer pool of all dirty pages by writing them
|
|||
|
out to disk. If a disk full condition is reached, only
|
|||
|
pages which actually made it to disk are marked as non
|
|||
|
dirty.
|
|||
|
Returns: nothing.
|
|||
|
*/
|
|||
|
int ibp;
|
|||
|
struct BPS *pbps = &mpibpbps [0];
|
|||
|
|
|||
|
for (ibp = 0; ibp < ibpMax; ++ibp, ++pbps)
|
|||
|
{
|
|||
|
#ifdef CKSM
|
|||
|
#ifdef DEBUG
|
|||
|
if (pbps->fn != fnNil && !pbps->fDirty)
|
|||
|
Assert( (**hpibpcksm) [ibp] == CksmFromIbp( ibp ) );
|
|||
|
#endif
|
|||
|
#endif
|
|||
|
if (pbps->fn != fnNil && pbps->fDirty)
|
|||
|
{
|
|||
|
FFlushFn(pbps->fn);
|
|||
|
/* keep on flushing if failure ? */
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
ReadFilePages(fn)
|
|||
|
int fn;
|
|||
|
{
|
|||
|
/*
|
|||
|
Description: ReadFilePages tries to read in as much of a file as
|
|||
|
it can. The idea is to fill the page buffers in anticipation of
|
|||
|
much access. This is called on every Transfer Load of a file.
|
|||
|
If fn == fnNil or there are no characters in the file, ReadFilePages
|
|||
|
simply returns.
|
|||
|
Returns: nothing
|
|||
|
*/
|
|||
|
int ibp;
|
|||
|
int cfcRead;
|
|||
|
int cpnRead;
|
|||
|
int dfcMac;
|
|||
|
typeFC fcMac;
|
|||
|
int ibpReadMax;
|
|||
|
int cfcLastPage;
|
|||
|
int iibp;
|
|||
|
struct FCB *pfcb;
|
|||
|
typeTS ts;
|
|||
|
|
|||
|
if (fn == fnNil)
|
|||
|
return;
|
|||
|
|
|||
|
/* Write ALL dirty pages to disk */
|
|||
|
WriteDirtyPages(); /* Just in case */
|
|||
|
|
|||
|
pfcb = &(**hpfnfcb)[fn];
|
|||
|
|
|||
|
/* we read as much of the file as will fit in the page buffers */
|
|||
|
/* Note that we assume that fcMax is coercable to an integer. This
|
|||
|
is valid as long as ibpMax*cbSector < 32k */
|
|||
|
dfcMac = (int) FcMin(pfcb->fcMac, (typeFC) (ibpMax * cbSector));
|
|||
|
if (dfcMac == 0)
|
|||
|
return;
|
|||
|
if (vfSysFull) /* call to FFlushFn in WriteDirtyPages failed.
|
|||
|
the buffer algorithm assures us that the first
|
|||
|
cbpMustKeep ibp's do not contain scratch file
|
|||
|
information. Thus, there is no danger in overwriting
|
|||
|
these ibps. */
|
|||
|
dfcMac = imin( dfcMac, (cbpMustKeep * cbSector) );
|
|||
|
|
|||
|
Assert( ((int)dfcMac) >= 0 );
|
|||
|
|
|||
|
/* Read pages from the file */
|
|||
|
|
|||
|
cfcRead = CchReadAtPage( fn, (typePN) 0, rgbp [0], (int) dfcMac, FALSE );
|
|||
|
|
|||
|
/* cfcRead contains a count of bytes read from the file */
|
|||
|
ibpReadMax = ((cfcRead-1) / cbSector) + 1;
|
|||
|
cfcLastPage = cfcRead - (ibpReadMax-1)*cbSector;
|
|||
|
ts = ibpMax;
|
|||
|
|
|||
|
/* order time stamps so the beginning slots have the greatest ts.
|
|||
|
Lru allocation will start at the end of the buffer table and work
|
|||
|
backward. Thus, the first page of the current file is considered
|
|||
|
the most recently used item.
|
|||
|
*/
|
|||
|
|
|||
|
/* describe the newly filled pages */
|
|||
|
for(ibp = 0; ibp < ibpReadMax; ++ibp)
|
|||
|
{
|
|||
|
struct BPS *pbps = &mpibpbps[ibp];
|
|||
|
pbps->fn = fn;
|
|||
|
pbps->pn = ibp;
|
|||
|
pbps->ts = --ts;
|
|||
|
pbps->fDirty = false;
|
|||
|
pbps->cch = cbSector;
|
|||
|
pbps->ibpHashNext = ibpNil;
|
|||
|
}
|
|||
|
|
|||
|
#ifdef CKSM
|
|||
|
#ifdef DEBUG
|
|||
|
{
|
|||
|
int ibpT;
|
|||
|
|
|||
|
for ( ibpT = 0; ibpT < ibpReadMax; ibpT++ )
|
|||
|
(**hpibpcksm) [ibpT] = CksmFromIbp( ibpT );
|
|||
|
}
|
|||
|
#endif
|
|||
|
#endif
|
|||
|
|
|||
|
/* fix some boundary conditions */
|
|||
|
mpibpbps[ibpReadMax-1].cch = cfcLastPage; /* ?????? */
|
|||
|
#ifdef CKSM
|
|||
|
#ifdef DEBUG
|
|||
|
(**hpibpcksm) [ibpReadMax - 1] = CksmFromIbp( ibpReadMax - 1 );
|
|||
|
#endif
|
|||
|
#endif
|
|||
|
|
|||
|
/* update descriptions of untouched page buffers */
|
|||
|
for (ibp=ibpReadMax; ibp < ibpMax; ibp++)
|
|||
|
{
|
|||
|
struct BPS *pbps = &mpibpbps[ibp];
|
|||
|
pbps->ts = --ts;
|
|||
|
pbps->fDirty = false;
|
|||
|
pbps->ibpHashNext = ibpNil;
|
|||
|
|
|||
|
#ifdef CKSM
|
|||
|
#ifdef DEBUG
|
|||
|
if (pbps->fn != fnNil)
|
|||
|
(**hpibpcksm) [ibp] = CksmFromIbp( ibp );
|
|||
|
#endif
|
|||
|
#endif
|
|||
|
}
|
|||
|
|
|||
|
tsMruBps = ibpMax - 1;
|
|||
|
|
|||
|
/* recalculate the hash table */
|
|||
|
RehashRgibpHash();
|
|||
|
|
|||
|
} /* end of R e a d F i l e P a g e s */
|
|||
|
|
|||
|
|
|||
|
RehashRgibpHash()
|
|||
|
{
|
|||
|
int iibp;
|
|||
|
register struct BPS *pbps;
|
|||
|
struct BPS *pbpsMax = &mpibpbps[ibpMax];
|
|||
|
int iibpHash;
|
|||
|
int ibpT;
|
|||
|
int ibpPrev;
|
|||
|
int ibp;
|
|||
|
|
|||
|
for (iibp = 0; iibp < iibpHashMax; iibp++)
|
|||
|
rgibpHash[iibp] = ibpNil;
|
|||
|
|
|||
|
for (ibp = 0, pbps = &mpibpbps[0]; pbps < pbpsMax; pbps++, ibp++)
|
|||
|
{
|
|||
|
if (pbps->fn == fnNil)
|
|||
|
continue;
|
|||
|
iibpHash = IibpHash(pbps->fn, pbps->pn);
|
|||
|
ibpT = rgibpHash[iibpHash];
|
|||
|
ibpPrev = ibpNil;
|
|||
|
while (ibpT != ibpNil)
|
|||
|
{
|
|||
|
ibpPrev = ibpT;
|
|||
|
ibpT = mpibpbps[ibpT].ibpHashNext;
|
|||
|
}
|
|||
|
if (ibpPrev == ibpNil)
|
|||
|
rgibpHash[iibpHash] = ibp;
|
|||
|
else
|
|||
|
mpibpbps[ibpPrev].ibpHashNext = ibp;
|
|||
|
}
|
|||
|
#ifdef DEBUG
|
|||
|
CheckIbp();
|
|||
|
#endif
|
|||
|
} /* end of RehashRgibpHash */
|
|||
|
|
|||
|
|
|||
|
|