322 lines
8.1 KiB
C
322 lines
8.1 KiB
C
|
/************************************************************/
|
|||
|
/* Windows Write, Copyright 1985-1992 Microsoft Corporation */
|
|||
|
/************************************************************/
|
|||
|
|
|||
|
/* fileres.c -- functions from file.c that are usually resident */
|
|||
|
|
|||
|
#define NOCLIPBOARD
|
|||
|
#define NOGDICAPMASKS
|
|||
|
#define NOCTLMGR
|
|||
|
#define NOVIRTUALKEYCODES
|
|||
|
#define NOWINMESSAGES
|
|||
|
#define NOWINSTYLES
|
|||
|
#define NOSYSMETRICS
|
|||
|
#define NOMENUS
|
|||
|
#define NOICON
|
|||
|
#define NOKEYSTATE
|
|||
|
#define NORASTEROPS
|
|||
|
#define NOSHOWWINDOW
|
|||
|
#define NOSYSCOMMANDS
|
|||
|
#define NOATOM
|
|||
|
#define NOCOLOR
|
|||
|
#define NOCREATESTRUCT
|
|||
|
#define NOCTLMGR
|
|||
|
#define NODRAWTEXT
|
|||
|
#define NOMETAFILE
|
|||
|
#define NOMSG
|
|||
|
#define NOHDC
|
|||
|
#define NOGDI
|
|||
|
#define NOMB
|
|||
|
#define NOFONT
|
|||
|
#define NOPEN
|
|||
|
#define NOBRUSH
|
|||
|
#define NOWNDCLASS
|
|||
|
#define NOSOUND
|
|||
|
#define NOCOMM
|
|||
|
#define NOPOINT
|
|||
|
#define NORECT
|
|||
|
#define NOREGION
|
|||
|
#define NOSCROLL
|
|||
|
#define NOTEXTMETRIC
|
|||
|
#define NOWH
|
|||
|
#define NOWINOFFSETS
|
|||
|
#include <windows.h>
|
|||
|
|
|||
|
#include "mw.h"
|
|||
|
#include "doslib.h"
|
|||
|
#define NOUAC
|
|||
|
#include "cmddefs.h"
|
|||
|
#include "docdefs.h"
|
|||
|
#include "filedefs.h"
|
|||
|
#include "str.h"
|
|||
|
#include "debug.h"
|
|||
|
|
|||
|
|
|||
|
extern int vfDiskFull;
|
|||
|
extern int vfSysFull;
|
|||
|
extern int vfnWriting;
|
|||
|
extern CHAR (*rgbp)[cbSector];
|
|||
|
extern typeTS tsMruRfn;
|
|||
|
extern struct BPS *mpibpbps;
|
|||
|
extern int ibpMax;
|
|||
|
extern struct FCB (**hpfnfcb)[];
|
|||
|
extern typeTS tsMruBps;
|
|||
|
extern struct ERFN dnrfn[rfnMax];
|
|||
|
extern int iibpHashMax;
|
|||
|
extern CHAR *rgibpHash;
|
|||
|
extern int rfnMac;
|
|||
|
extern int ferror;
|
|||
|
extern CHAR szWriteDocPrompt[];
|
|||
|
extern CHAR szScratchFilePrompt[];
|
|||
|
extern CHAR szSaveFilePrompt[];
|
|||
|
|
|||
|
|
|||
|
#define IibpHash(fn,pn) ((int) ((fn + 1) * (pn + 1)) & 077777) % iibpHashMax
|
|||
|
|
|||
|
|
|||
|
#ifdef DEBUG
|
|||
|
#define STATIC
|
|||
|
#else
|
|||
|
#define STATIC static
|
|||
|
#define ErrorWithMsg( idpmt, szModule ) Error( idpmt )
|
|||
|
#define DiskErrorWithMsg( idpmt, szModule ) DiskError( idpmt )
|
|||
|
#endif
|
|||
|
|
|||
|
#define osfnNil (-1)
|
|||
|
|
|||
|
|
|||
|
CHAR *PchFromFc(fn, fc, pcch)
|
|||
|
int fn;
|
|||
|
typeFC fc;
|
|||
|
int *pcch;
|
|||
|
{ /*
|
|||
|
Description: Reads from a file, starting at virtual character
|
|||
|
position fc. Reads until end of buffer page.
|
|||
|
Returns: Pointer to char buffer starting at fc.
|
|||
|
The number of characters read is returned in *pcch.
|
|||
|
*/
|
|||
|
int dfc;
|
|||
|
CHAR *pch;
|
|||
|
typePN pn;
|
|||
|
int ibp;
|
|||
|
struct BPS *pbps;
|
|||
|
|
|||
|
dfc = (int) (fc % cfcPage);
|
|||
|
pn = (typePN) (fc / cfcPage);
|
|||
|
|
|||
|
ibp = IbpEnsureValid(fn, pn);
|
|||
|
pbps = &mpibpbps[ibp];
|
|||
|
*pcch = pbps->cch - dfc;
|
|||
|
return &rgbp[ibp][dfc];
|
|||
|
}
|
|||
|
/* end of P c h F r o m F c */
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
/*** PchGetPn - Assure file page loaded, return pointer
|
|||
|
*
|
|||
|
*/
|
|||
|
|
|||
|
CHAR *PchGetPn(fn, pn, pcch, fWrite)
|
|||
|
int fn;
|
|||
|
typePN pn;
|
|||
|
int *pcch;
|
|||
|
BOOL fWrite; // missing before?? (2.11.91) D. Kent
|
|||
|
{ /*
|
|||
|
Description: Get char pointer to page buffer, option to mark
|
|||
|
page as dirty.
|
|||
|
Returns: Pointer to buffer.
|
|||
|
cch in *pcch
|
|||
|
*/
|
|||
|
|
|||
|
int ibp = IbpEnsureValid(fn, pn);
|
|||
|
struct BPS *pbps = &mpibpbps[ibp];
|
|||
|
|
|||
|
*pcch = pbps->cch;
|
|||
|
pbps->fDirty |= fWrite;
|
|||
|
return rgbp[ibp];
|
|||
|
} /* end of P c h G e t P n */
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
int IbpEnsureValid(fn, pn)
|
|||
|
int fn;
|
|||
|
typePN pn;
|
|||
|
{ /*
|
|||
|
Description: Get page pn of file fn into memory.
|
|||
|
If already in memory, return.
|
|||
|
Returns: Bp index (buffer slot #) where the page resides
|
|||
|
in memory.
|
|||
|
*/
|
|||
|
|
|||
|
int ibp;
|
|||
|
register struct BPS *pbps;
|
|||
|
|
|||
|
#ifdef DEBUG
|
|||
|
CheckIbp();
|
|||
|
#endif /* DEBUG */
|
|||
|
|
|||
|
/* Is the page currently in memory? */
|
|||
|
ibp = rgibpHash[IibpHash(fn,pn)];
|
|||
|
/* ibp is the first in a linked list of possible matches */
|
|||
|
/* resident in memory */
|
|||
|
|
|||
|
Scribble(3,'V');
|
|||
|
|
|||
|
while (ibp != ibpNil) /* while not end of linked list */
|
|||
|
{ /* check if any buffers in memory match */
|
|||
|
pbps = &mpibpbps[ibp];
|
|||
|
if (pbps->fn == fn && pbps->pn == pn)
|
|||
|
{ /* Found it */
|
|||
|
pbps->ts = ++tsMruBps; /* mark page as MRUsed */
|
|||
|
Scribble(3,' ');
|
|||
|
return ibp;
|
|||
|
}
|
|||
|
ibp = pbps->ibpHashNext;
|
|||
|
}
|
|||
|
|
|||
|
/* page is not currently in memory */
|
|||
|
|
|||
|
return IbpMakeValid( fn, pn );
|
|||
|
} /* end of I b p E n s u r e V a l i d */
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
CloseEveryRfn( fHardToo )
|
|||
|
{ /* Close all files we have open. Close only files on removable media
|
|||
|
if fHardToo is FALSE; ALL files if fHardToo is TRUE */
|
|||
|
int rfn;
|
|||
|
|
|||
|
for (rfn = 0; rfn < rfnMac; rfn++)
|
|||
|
{
|
|||
|
int fn = dnrfn [rfn].fn;
|
|||
|
|
|||
|
if (fn != fnNil)
|
|||
|
if ( fHardToo ||
|
|||
|
!((POFSTRUCT)((**hpfnfcb)[fn].rgbOpenFileBuf))->fFixedDisk )
|
|||
|
{
|
|||
|
CloseRfn( rfn );
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
typeFC FcWScratch(pch, cch)
|
|||
|
CHAR *pch;
|
|||
|
int cch;
|
|||
|
{ /*
|
|||
|
Description: Write chars at end of scratch file.
|
|||
|
Returns: first fc written.
|
|||
|
*/
|
|||
|
typeFC fc = (**hpfnfcb)[fnScratch].fcMac;
|
|||
|
#if 0
|
|||
|
extern BOOL bNo64KLimit;
|
|||
|
|
|||
|
if ((!bNo64KLimit) && (((long) fc) + ((long) cch) > 65536L)) /* scratch file to big */
|
|||
|
{
|
|||
|
DiskErrorWithMsg(IDPMTSFER, " FcWScratch"); /* session too long */
|
|||
|
|
|||
|
vfSysFull = fTrue;
|
|||
|
/* recovery is accomplished: all that happens is that a few
|
|||
|
characters do not get written to the scratch file - the
|
|||
|
user loses only a little bit of his work. */
|
|||
|
}
|
|||
|
else
|
|||
|
#endif
|
|||
|
WriteRgch(fnScratch, pch, cch);
|
|||
|
return fc;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
WriteRgch(fn, pch, cch)
|
|||
|
int fn;
|
|||
|
CHAR *pch;
|
|||
|
int cch;
|
|||
|
{ /*
|
|||
|
Description: Writes char string pch, length cch, to end of
|
|||
|
file fn.
|
|||
|
Returns: nothing
|
|||
|
*/
|
|||
|
extern vfDiskError;
|
|||
|
struct FCB *pfcb = &(**hpfnfcb)[fn];
|
|||
|
typePN pn = (typePN) (pfcb->fcMac / cfcPage);
|
|||
|
#ifdef WIN30
|
|||
|
/* Error checking was horrendous in these early days, right?
|
|||
|
Ha. It still is. In any case, don't know WHAT we can do
|
|||
|
if the page number has gotten too large, so just fake a
|
|||
|
disk error so that IbpEnsureValid() doesn't go off into
|
|||
|
never-never land! This catch effectively limits us to
|
|||
|
4M files ..pault 11/1/89 */
|
|||
|
|
|||
|
if (pn > pgnMax)
|
|||
|
#ifdef DEBUG
|
|||
|
DiskErrorWithMsg(IDPMTSDE2, "writergch");
|
|||
|
#else
|
|||
|
DiskError(IDPMTSDE2);
|
|||
|
#endif
|
|||
|
else
|
|||
|
#endif
|
|||
|
|
|||
|
while (cch > 0)
|
|||
|
{ /* One page at a time */
|
|||
|
int ibp = IbpEnsureValid(fn, pn++);
|
|||
|
struct BPS *pbps = &mpibpbps[ibp];
|
|||
|
int cchBp = pbps->cch;
|
|||
|
int cchBlt = min((int)cfcPage - cchBp, cch);
|
|||
|
|
|||
|
Assert( vfDiskError ||
|
|||
|
cchBp == pfcb->fcMac - (pn - 1) * cfcPage);
|
|||
|
|
|||
|
bltbyte(pch, &rgbp[ibp][cchBp], cchBlt);
|
|||
|
pbps->cch += cchBlt;
|
|||
|
pbps->fDirty = true;
|
|||
|
pfcb->fcMac += cchBlt;
|
|||
|
pfcb->pnMac = pn;
|
|||
|
pch += cchBlt;
|
|||
|
cch -= cchBlt;
|
|||
|
}
|
|||
|
} /* end of W r i t e R g c h */
|
|||
|
|
|||
|
|
|||
|
|
|||
|
|
|||
|
CloseRfn( rfn )
|
|||
|
int rfn;
|
|||
|
{/*
|
|||
|
Description: Close a file and delete its Rfn entry
|
|||
|
Returns: nothing
|
|||
|
*/
|
|||
|
struct ERFN *perfn = &dnrfn[rfn];
|
|||
|
int fn = perfn->fn;
|
|||
|
|
|||
|
Assert (rfn >= 0 &&
|
|||
|
rfn < rfnMac &&
|
|||
|
perfn->osfn != osfnNil &&
|
|||
|
fn != fnNil);
|
|||
|
|
|||
|
#ifdef DEBUG
|
|||
|
#ifdef DFILE
|
|||
|
CommSzSz( "Closing file: ", &(**(**hpfnfcb)[fn].hszFile)[0] );
|
|||
|
#endif
|
|||
|
#endif
|
|||
|
/* Close may fail if windows already closed the file for us,
|
|||
|
but that's OK */
|
|||
|
FCloseDoshnd( perfn->osfn );
|
|||
|
|
|||
|
{ /* Just like the statement below, but 28 bytes less
|
|||
|
under CMERGE V13 */
|
|||
|
REG1 struct FCB *pfcb = &(**hpfnfcb) [fn];
|
|||
|
pfcb->rfn = rfnNil;
|
|||
|
}
|
|||
|
/* (**hpfnfcb)[fn].rfn = rfnNil; */
|
|||
|
|
|||
|
|
|||
|
perfn->fn = fnNil;
|
|||
|
}
|
|||
|
|