2020-09-30 16:53:55 +02:00

497 lines
12 KiB
C

/* demhndl.c - SVC handlers for calls where file handle is provided.
*
* demClose
* demRead
* demWrite
* demChgFilePtr
* demFileTimes
*
* Modification History:
*
* Sudeepb 02-Apr-1991 Created
* rfirth 25-Sep-1991 Added Vdm Redir stuff for named pipes
*/
#include "dem.h"
#include "demmsg.h"
#include <softpc.h>
#include <io.h>
#include <fcntl.h>
#include <vrnmpipe.h>
#include <exterr.h>
#include <mvdm.h>
#include "dpmtbls.h"
BOOL (*VrInitialized)(VOID); // POINTER TO FUNCTION
extern BOOL IsVdmRedirLoaded(VOID);
/* demClose - Close a file
*
*
* Entry - Client (AX:BP) File Handle
* Client (CX:DX) File position (if -1 no seek needed before closing
* the handle.
* (VadimB)
* Client (es:di) SFT ptr - this is implied in abort.asm code
*
* Exit
* SUCCESS
* Client (CY) = 0
*
* FAILURE
* Client (CY) = 1
* Client (AX) = system status code
*
*/
VOID demClose (VOID)
{
HANDLE hFile;
LONG lLoc;
USHORT usDX,usCX;
hFile = GETHANDLE (getAX(),getBP());
if (hFile == 0) {
setCF (0);
return;
}
usCX = getCX();
usDX = getDX();
if (!((usCX == (USHORT)-1) && (usDX == (USHORT)-1))) {
lLoc = (LONG)((((int)usCX) << 16) + (int)usDX);
//
// Note that we don't check for failure in this case as edlin,
// for instance, can have the file position be negative and
// we still need to do the cleanup below. Note that we are not
// even sure why seeking on close matter, but the DOS code does it...
//
DPM_SetFilePointer (hFile,
lLoc,
NULL,
FILE_BEGIN);
}
if (DPM_CloseHandle (hFile) == FALSE){
demClientError(hFile, (CHAR)-1);
}
//
// if the redir TSR is being run in this VDM session, check if the handle
// being closed references a named pipe - we have to delete some info
// that we keep for the open named pipe
//
if (IsVdmRedirLoaded()) {
VrRemoveOpenNamedPipeInfo(hFile);
}
setCF(0);
return;
}
/* demRead - Read a file
*
*
* Entry - Client (AX:BP) File Handle
* Client (CX) Count to read
* Client (DS:DX) Buffer Address
* Client (BX:SI) = current file pointer location.
* ZF = 1 if seek is not needed prior to read.
*
* Exit
* SUCCESS
* Client (CY) = 0
* Client (AX) = Count of bytes read
*
* FAILURE
* Client (CY) = 1
* Client (AX) = system status code
*
*/
VOID demRead (VOID)
{
HANDLE hFile;
LPVOID lpBuf;
DWORD dwBytesRead;
USHORT usDS,usDX;
DWORD dwReadError;
BOOL ok;
UCHAR locus, action, class;
LONG lLoc;
hFile = GETHANDLE (getAX(),getBP());
usDS = getDS();
usDX = getDX();
lpBuf = (LPVOID) GetVDMAddr (usDS,usDX);
//
// if this handle is a named pipe then use VrReadNamedPipe since we have
// to perform an overlapped read, and wait on the event handle for completion
// even though we're still doing synchronous read
//
if (IsVdmRedirLoaded()) {
if (VrIsNamedPipeHandle(hFile)) {
//
// named pipe read always sets the extended error information in the
// DOS data segment. This is the only way we can return bytes read
// information and a more data indication
//
ok = VrReadNamedPipe(hFile,
lpBuf,
(DWORD)getCX(),
&dwBytesRead,
&dwReadError
);
switch (dwReadError) {
case NO_ERROR:
locus = action = class = 0;
break;
case ERROR_NO_DATA:
case ERROR_MORE_DATA:
locus = errLOC_Net;
class = errCLASS_TempSit;
action = errACT_Retry;
break;
default:
//
// any error other than the specific ones we handle here should be
// correctly handled by DOS
//
goto readFailureExit;
}
pExtendedError->ExtendedErrorLocus = locus;
STOREWORD(pExtendedError->ExtendedError, (WORD)dwReadError);
pExtendedError->ExtendedErrorAction = action;
pExtendedError->ExtendedErrorClass = class;
if (ok) {
goto readSuccessExit;
} else {
goto readFailureExit;
}
}
}
//
// if the redir TSR is not loaded, or the handle is not a named pipe then
// perform normal file read
//
if (!getZF()) {
ULONG Zero = 0;
lLoc = (LONG)((((int)getBX()) << 16) + (int)getSI());
if ((DPM_SetFilePointer (hFile,
lLoc,
&Zero,
FILE_BEGIN) == -1L) &&
(GetLastError() != NO_ERROR)) {
goto readFailureExit;
}
}
if (DPM_ReadFile (hFile,
lpBuf,
(DWORD)getCX(),
&dwBytesRead,
NULL) == FALSE){
readFailureExit:
Sim32FlushVDMPointer (((ULONG)(usDS << 16)) | usDX, getCX(),
(PBYTE )lpBuf, FALSE);
Sim32FreeVDMPointer (((ULONG)(usDS << 16)) | usDX, getCX(),
(PBYTE )lpBuf, FALSE);
if (GetLastError() == ERROR_BROKEN_PIPE) {
setAX(0);
setCF(0);
return;
}
demClientError(hFile, (CHAR)-1);
return ;
}
readSuccessExit:
Sim32FlushVDMPointer (((ULONG)(usDS << 16)) | usDX, getCX(),
(PBYTE )lpBuf, FALSE);
Sim32FreeVDMPointer (((ULONG)(usDS << 16)) | usDX, getCX(),
(PBYTE )lpBuf, FALSE);
setCF(0);
setAX((USHORT)dwBytesRead);
return;
}
/* demWrite - Write to a file
*
*
* Entry - Client (AX:BP) File Handle
* Client (CX) Count to write
* Client (DS:DX) Buffer Address
* Client (BX:SI) = current file pointer location.
* ZF = 1 if seek is not needed prior to write.
*
* Exit
* SUCCESS
* Client (CY) = 0
* Client (AX) = Count of bytes written
*
* FAILURE
* Client (CY) = 1
* Client (AX) = system status code
*
*/
VOID demWrite (VOID)
{
HANDLE hFile;
DWORD dwBytesWritten;
LPVOID lpBuf;
LONG lLoc;
DWORD dwErrCode;
hFile = GETHANDLE (getAX(),getBP());
lpBuf = (LPVOID) GetVDMAddr (getDS(),getDX());
//
// if this handle is a named pipe then use VrWriteNamedPipe since we have
// to perform an overlapped write, and wait on the event handle for completion
// even though we're still doing synchronous write
//
if (IsVdmRedirLoaded()) {
if (VrIsNamedPipeHandle(hFile)) {
if (VrWriteNamedPipe(hFile, lpBuf, (DWORD)getCX(), &dwBytesWritten)) {
goto writeSuccessExit;
} else {
goto writeFailureExit;
}
}
}
//
// if the redir TSR is not loaded, or the handle is not a named pipe then
// perform normal file write
//
if (!getZF()) {
ULONG Zero = 0;
lLoc = (LONG)((((int)getBX()) << 16) + (int)getSI());
if ((DPM_SetFilePointer (hFile,
lLoc,
&Zero,
FILE_BEGIN) == -1L) &&
(GetLastError() != NO_ERROR)) {
demClientError(hFile, (CHAR)-1);
return ;
}
}
// In DOS CX=0 truncates or extends the file to current file pointer.
if (getCX() == 0){
if (DPM_SetEndOfFile(hFile) == FALSE){
demClientError(hFile, (CHAR)-1);
return;
}
setCF (0);
return;
}
if (DPM_WriteFile (hFile,
lpBuf,
(DWORD)getCX(),
&dwBytesWritten,
NULL) == FALSE){
// If disk is full then we should return 0 byte written and CF is clear
dwErrCode = GetLastError();
if(dwErrCode == ERROR_DISK_FULL) {
setCF(0);
setAX(0);
return;
}
SetLastError(dwErrCode);
writeFailureExit:
demClientError(hFile, (CHAR)-1);
return ;
}
writeSuccessExit:
setCF(0);
setAX((USHORT)dwBytesWritten);
return;
}
/* demChgFilePtr - Change File Pointer
*
*
* Entry - Client (AX:BP) File Handle
* Client (CX:DX) New Location
* Client (BL) Positioning Method
* 0 - File Absolute
* 1 - Relative to Current Position
* 2 - Relative to end of file
*
* Exit
* SUCCESS
* Client (CY) = 0
* Client (DX:AX) = New Location
*
* FAILURE
* Client (CY) = 1
* Client (AX) = system status code
*
*/
VOID demChgFilePtr (VOID)
{
HANDLE hFile;
LONG lLoc;
DWORD dwLoc;
#if (FILE_BEGIN != 0 || FILE_CURRENT != 1 || FILE_END !=2)
#error "Win32 values not DOS compatible"
#
#endif
hFile = GETHANDLE (getAX(),getBP());
lLoc = (LONG)((((int)getCX()) << 16) + (int)getDX());
if ((dwLoc = DPM_SetFilePointer (hFile,
lLoc,
NULL,
(DWORD)getBL())) == -1L){
demClientError(hFile, (CHAR)-1);
return ;
}
setCF(0);
setAX((USHORT)dwLoc);
setDX((USHORT)(dwLoc >> 16));
return;
}
/* DemCommit -- Commit File(Flush file buffers)
*
* Entry - Client (AX:BP) File Handle
*
* Exit
* SUCCESS
* Client (CY) = 0
* buffer flushed
*
* FAILURE
* Client (CY) = 1
*
*/
VOID demCommit(VOID)
{
HANDLE hFile;
BOOL bRet;
hFile = GETHANDLE(getAX(),getBP());
bRet = DPM_FlushFileBuffers(hFile);
#if DBG
if (!bRet) {
//
// FlushFileBuffers fails with access denied if the handle
// is open for read-only access, however it's not an error
// for DOS.
//
DWORD LastError;
LastError = GetLastError();
if (LastError != ERROR_ACCESS_DENIED) {
sprintf(demDebugBuffer,
"ntvdm demCommit warning: FlushFileBuffers error %d\n",
LastError);
OutputDebugStringOem(demDebugBuffer);
}
}
#endif
setCF(0);
}
/* function to check if new data has been written to the file or
if the file has been marked EOF
Input: Client (AX:BP) = 32bits NT file handle
Output: Client ZF = 1 if new data or EOF
CF = 1 if EOF
*/
VOID demPipeFileDataEOF(VOID)
{
HANDLE hFile;
BOOL fEOF;
BOOL DataEOF;
DWORD FileSizeLow;
DWORD FileSizeHigh;
hFile = GETHANDLE(getAX(), getBP());
DataEOF = cmdPipeFileDataEOF(hFile, &fEOF);
if (fEOF) {
//EOF, get file size, max size = 32bits
FileSizeLow = GetFileSize(hFile, &FileSizeHigh);
setAX((WORD)(FileSizeLow / 0x10000));
setBP((WORD)FileSizeLow);
setCF(1); // EOF is encountered
}
else
setCF(0);
setZF(DataEOF ? 0 : 1);
}
/* function to check if the file has been marked EOF
Input: Client(AX:BP) = 32bits NT file handle
Output: Client CY = 1 if EOF
*/
VOID demPipeFileEOF(VOID)
{
HANDLE hFile;
DWORD FileSizeLow;
DWORD FileSizeHigh;
hFile = GETHANDLE(getAX(), getBP());
if (cmdPipeFileEOF(hFile)) {
FileSizeLow = GetFileSize(hFile, &FileSizeHigh);
setAX((WORD)(FileSizeLow / 0x10000)); // file size in 32bits
setBP((WORD)FileSizeLow);
setCF(1); //EOF is encountered
}
else
setCF(0);
}