windows-nt/Source/XPSP1/NT/base/mvdm/dpmi32/int21map.c

2993 lines
57 KiB
C
Raw Normal View History

2020-09-26 03:20:57 -05:00
/*++
Copyright (c) 1992 Microsoft Corporation
Module Name:
Int21Map.c
Abstract:
This module performs int 21 API translation for dpmi
Author:
Dave Hastings (daveh) 23-Nov-1992
Revision History:
Neil Sandlin (neilsa) 31-Jul-1995 - Updates for the 486 emulator
--*/
#include "precomp.h"
#pragma hdrstop
#include "int21map.h"
#include "softpc.h"
#include "xlathlp.h"
//
// Local constants
//
//#define Verbose 1
#define MAX_SUPPORTED_DOS_CALL 0x6d
#define DosxTranslated NULL
typedef VOID (*APIXLATFUNCTION)(VOID);
APIXLATFUNCTION ApiXlatTable[MAX_SUPPORTED_DOS_CALL] = {
DosxTranslated , // 00h - Terminate process
NoTranslation , // 01h - Char input with echo
NoTranslation , // 02h - Character output
NoTranslation , // 03h - Auxiliary input
NoTranslation , // 04h - Auxiliary output
NoTranslation , // 05h - Print character
NoTranslation , // 06h - Direct console I/O
NoTranslation , // 07h - Unfiltered char input
NoTranslation , // 08h - Char input w/o echo
DisplayString , // 09h - Display "$" term string
BufferedKeyboardInput , // 0Ah - Buffered keyboard input
NoTranslation , // 0Bh - Check keyboard status
FlushBuffReadKbd , // 0Ch - Flush buff, Read kbd
NoTranslation , // 0Dh - Disk reset
NoTranslation , // 0Eh - Select disk
NotSupportedFCB , // 0Fh * Open file with FCB
NotSupportedFCB , // 10h * Close file with FCB
FindFileFCB , // 11h - Find First File
FindFileFCB , // 12h - Find Next File
MapFCB , // 13h - Delete File
NotSupportedFCB , // 14h * Sequential Read
NotSupportedFCB , // 15h * Sequential Write
NotSupportedFCB , // 16h * Create File With FCB
RenameFCB , // 17h - Rename File With FCB
NoTranslation , // 18h - UNUSED
NoTranslation , // 19h - Get Current Disk
SetDTA , // 1Ah - Set DTA Address
GetDriveData , // 1Bh - Get Default Drive Data
GetDriveData , // 1Ch - Get Drive Data
NoTranslation , // 1Dh - UNUSED
NoTranslation , // 1Eh - UNUSED
GetDriveData , // 1Fh - Get Drive Parameter Blk
NoTranslation , // 20h - UNUSED
NotSupportedFCB , // 21h * Random Read
NotSupportedFCB , // 22h * Random Write
NotSupportedFCB , // 23h * Get File Size
NotSupportedFCB , // 24h * Set Relative Record
SetVector , // 25h - Set interrupt vector
CreatePSP , // 26h - Create PSP
NotSupportedFCB , // 27h * Random block read
NotSupportedFCB , // 28h * Random block write
ParseFilename , // 29h - Parse filename
GetDate , // 2Ah - Get date
NoTranslation , // 2Bh - Set date
NoTranslation , // 2Ch - Get time
NoTranslation , // 2Dh - Set time
NoTranslation , // 2Eh - Set/Reset verify flag
GetDTA , // 2Fh - Get DTA address
NoTranslation , // 30h - Get DOS version
TSR , // 31h - Terminate and Stay Res
GetDevParamBlock , // 32h - Get device parameter blk
NoTranslation , // 33h - Get/Set Control-C Flag
ReturnESBX , // 34h - Get Address of InDOS
GetVector , // 35h - Get Interrupt Vector
NoTranslation , // 36h - Get Disk Free Space
NoTranslation , // 37h - Char Oper
GetSetCountry , // 38h - Get/Set Current Country
MapASCIIZDSDX , // 39h - Create Directory
MapASCIIZDSDX , // 3Ah - Remove Directory
MapASCIIZDSDX , // 3Bh - Change Current Directory
MapASCIIZDSDX , // 3Ch - Create File with Handle
MapASCIIZDSDX , // 3Dh - Open File with Handle
NoTranslation , // 3Eh - Close File with Handle
ReadWriteFile , // 3Fh - Read File or Device
ReadWriteFile , // 40h - Write File or Device
MapASCIIZDSDX , // 41h - Delete File
NoTranslation , // 42h - Move file pointer
MapASCIIZDSDX , // 43h - Get/Set File Attributes
IOCTL , // 44h - IOCTL
NoTranslation , // 45h - Duplicate File Handle
NoTranslation , // 46h - Force Duplicate Handle
GetCurDir , // 47h - Get Current Directory
AllocateMemoryBlock , // 48h - Allocate Memory Block
FreeMemoryBlock , // 49h - Free Memory Block
ResizeMemoryBlock , // 4Ah - Resize Memory Block
LoadExec , // 4Bh - Load and Exec Program
DosxTranslated , // 4Ch - Terminate with Ret Code
NoTranslation , // 4Dh - Get Ret Code Child Proc
FindFirstFileHandle , // 4Eh - Find First File
FindNextFileHandle , // 4Fh - Find Next File
SetPSP , // 50h - Set PSP Segment
GetPSP , // 51h - Get PSP Segment
ReturnESBX , // 52h - Get List of Lists (invars)
TranslateBPB , // 53h - Translate BPB
NoTranslation , // 54h - Get Verify Flag
CreatePSP , // 55h - Create PSP
RenameFile , // 56h - Rename File
NoTranslation , // 57h - Get/Set Date/Time File
NoTranslation , // 58h - Get/Set Alloc Strategy
NoTranslation , // 59h - Get Extended Error Info
CreateTempFile , // 5Ah - Create Temporary File
MapASCIIZDSDX , // 5Bh - Create New File
NoTranslation , // 5Ch - Lock/Unlock File Region
Func5Dh , // 5Dh - Server DOS call
Func5Eh , // 5Eh - Net Name/Printer Setup
Func5Fh , // 5Fh - Network redir stuff
NotSupportedBad , // 60h - NameTrans
NoTranslation , // 61h - UNUSED
GetPSP , // 62h - Get PSP Address
#ifdef DBCS
ReturnDSSI , // 63h - Get DBCS Vector
#else
NotSupportedBetter , // 63h - Hangool call
#endif
NotSupportedBad , // 64h - Set Prn Flag
GetExtendedCountryInfo , // 65h - Get Extended Country Info
NoTranslation , // 66h - Get/Set Global Code Page
NoTranslation , // 67h - Set Handle Count
NoTranslation , // 68h - Commit File
NoTranslation , // 69h -
NoTranslation , // 6Ah -
NoTranslation , // 6Bh -
MapASCIIZDSSI // 6Ch - Extended open
};
VOID
DpmiXlatInt21Call(
VOID
)
/*++
Routine Description:
This routine dispatches to the appropriate routine to perform the
actual translation of the api
Arguments:
None
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
ULONG DosMajorCode;
PUCHAR StackPointer;
//
// Pop ds from stack
//
StackPointer = Sim32GetVDMPointer(
((ULONG)getSS() << 16),
1,
TRUE
);
StackPointer += (*GetSPRegister)();
setDS(*(PWORD16)StackPointer);
(*SetSPRegister)((*GetSPRegister)() + 2);
DosMajorCode = getAH();
if (DosMajorCode >= MAX_SUPPORTED_DOS_CALL) {
return; //bugbug find out what win31 does.
}
if (ApiXlatTable[DosMajorCode]) {
ULONG Eip, Esp;
USHORT Cs, Ss;
Eip = getEIP();
Esp = getESP();
Cs = getCS();
Ss = getSS();
(*ApiXlatTable[DosMajorCode])();
setEIP(Eip);
setESP(Esp);
setCS(Cs);
setSS(Ss);
SimulateIret(PASS_CARRY_FLAG_16);
}
// put this back in after beta 2.5
DpmiFreeAllBuffers();
}
VOID
NoTranslation(
VOID
)
/*++
Routine Description:
This routine handles int 21 functions which don't need any translation.
It simply executes the int 21 in real/v86 mode
Arguments:
None
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
DpmiSwitchToRealMode();
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
}
VOID
DisplayString(
VOID
)
/*++
Routine Description:
This routine translates the int 21 display string function.
N.B. Win 3.1 does this by calling int 21 function 2 repeatedly.
We do it this way because Win 3.1 does. If this function is
frequently used, we should actually buffer the string and
call the dos display string function.
Arguments:
None
Return Value:
None.
--*/
{ PUCHAR String;
DECLARE_LocalVdmContext;
USHORT ClientAX, ClientDX;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
String = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
//
// Repeatedly call int 21 function 2 to display the characters
//
ClientAX = getAX();
ClientDX = getDX();
setAH(2);
while (*(String) != '$') {
setDL(*(String));
DPMI_EXEC_INT(0x21);
String++;
}
setAX(ClientAX);
setDX(ClientDX);
DpmiSwitchToProtectedMode();
}
VOID
BufferedKeyboardInput(
VOID
)
/*++
Routine Description:
This routine performs the tranlation for the int 21 function Ah
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR PmInputBuffer, RmInputBuffer;
USHORT BufferSeg, BufferOff, BufferLen;
USHORT ClientDX;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
//
// Get a pointer to the PM buffer
//
PmInputBuffer = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
//
// Get the buffer length (there are two bytes in addition to the
// buffer)
//
BufferLen = *PmInputBuffer + 2;
//
// Move the buffer to low memory
//
RmInputBuffer = DpmiMapAndCopyBuffer(PmInputBuffer, BufferLen);
//
// Set up the registers for the int 21 call
//
DPMI_FLAT_TO_SEGMENTED(RmInputBuffer, &BufferSeg, &BufferOff);
setDS(BufferSeg);
setDX(BufferOff);
DPMI_EXEC_INT(0x21);
//
// Copy the data back
//
DpmiUnmapAndCopyBuffer(PmInputBuffer, RmInputBuffer, BufferLen);
setDX(ClientDX);
DpmiSwitchToProtectedMode();
setDS(ClientDS);
}
VOID
FlushBuffReadKbd(
VOID
)
/*++
Routine Description:
This routine performs translation for the flush keyboard routines.
It calls the appropriate xlat routine.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
if (getAL() == 0x0a) {
BufferedKeyboardInput();
} else {
NoTranslation();
}
}
VOID
NotSupportedFCB(
VOID
)
/*++
Routine Description:
This routine does not do any translation. It prints a warning to
the debugger
Arguments:
None.
Return Value:
None.
--*/
{
#if DBG
DECLARE_LocalVdmContext;
DbgPrint(
"WARNING: DOS INT 21 call AH = %x will not be translated.\n", getAH());
DbgPrint(
" Protected mode applications should not be using\n");
DbgPrint(
" this type of FCB call. Convert this application\n");
DbgPrint(
" to use the handle calls.\n"
);
#endif
NoTranslation();
}
VOID
FindFileFCB(
VOID
)
/*++
Routine Description:
This function translates the find first/find next FCB calls.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR FcbAddress, FcbBufferedAddress;
USHORT FcbSegment, FcbOffset;
USHORT ClientDX;
USHORT FcbLength;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
SetDTAPointers();
ClientDX = getDX();
FcbAddress = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
//
// Calculate the size of the FCB
//
FcbLength = DpmiCalcFcbLength(FcbAddress);
//
// Buffer the FCB
//
FcbBufferedAddress = DpmiMapAndCopyBuffer(FcbAddress, FcbLength);
//
// Check to see if we need to set the real dta
//
if (CurrentDosDta != CurrentDta)
SetDosDTA();
//
// Make the int 21 call
//
DPMI_FLAT_TO_SEGMENTED(FcbBufferedAddress, &FcbSegment, &FcbOffset);
setDS(FcbSegment);
setDX(FcbOffset);
DPMI_EXEC_INT(0x21);
//
// If the call was successful and the PM dta is in high memory
// copy the dta to high memory
//
if ((getAL() == 0) && (CurrentPmDtaAddress != CurrentDta)) {
MoveMemory(CurrentPmDtaAddress, CurrentDta, FcbLength);
DpmiUnmapAndCopyBuffer(FcbAddress, FcbBufferedAddress, FcbLength);
} else {
DpmiUnmapBuffer(FcbBufferedAddress, FcbLength);
}
setDX(ClientDX);
DpmiSwitchToProtectedMode();
setDS(ClientDS);
}
VOID
MapFCB(
VOID
)
/*++
Routine Description:
This routine translates the delete file fcb int 21 call
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT FcbLength, FcbSegment, FcbOffset;
PUCHAR FcbAddress, FcbBufferedAddress;
USHORT ClientDX;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
FcbAddress = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
//
// Get the length of the FCB
//
FcbLength = DpmiCalcFcbLength(FcbAddress);
//
// Copy the FCB
//
FcbBufferedAddress = DpmiMapAndCopyBuffer(FcbAddress, FcbLength);
//
// Make the int 21 call
//
DPMI_FLAT_TO_SEGMENTED(FcbBufferedAddress, &FcbSegment, &FcbOffset);
setDS(FcbSegment);
setDX(FcbOffset);
DPMI_EXEC_INT(0x21);
//
// Copy the FCB back
//
DpmiUnmapAndCopyBuffer(FcbAddress, FcbBufferedAddress, FcbLength);
//
// Clean up
//
setDX(ClientDX);
DpmiSwitchToProtectedMode();
setDS(ClientDS);
}
VOID
RenameFCB(
VOID
)
/*++
Routine Description:
This routine translates the rename fcb int 21 function
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT FcbSegment, FcbOffset;
PUCHAR FcbAddress, FcbBufferedAddress;
USHORT ClientDX;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
FcbAddress = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
//
// Copy the FCB (The fcb for rename is a special format, fixed size)
//
FcbBufferedAddress = DpmiMapAndCopyBuffer(FcbAddress, 0x25);
//
// Make the int 21 call
//
DPMI_FLAT_TO_SEGMENTED(FcbBufferedAddress, &FcbSegment, &FcbOffset);
setDS(FcbSegment);
setDX(FcbOffset);
DPMI_EXEC_INT(0x21);
//
// Copy the FCB back
//
DpmiUnmapAndCopyBuffer(FcbAddress, FcbAddress, 0x25);
//
// Clean up
//
setDX(ClientDX);
DpmiSwitchToProtectedMode();
setDS(ClientDS);
}
VOID
GetDriveData(
VOID
)
/*++
Routine Description:
Translates the drive data int 21 calls
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT Selector;
DpmiSwitchToRealMode();
DPMI_EXEC_INT(0x21);
Selector = getDS();
DpmiSwitchToProtectedMode();
setDS(SegmentToSelector(Selector, STD_DATA));
(*SetBXRegister)((ULONG)getBX());
}
VOID
SetVector(
VOID
)
/*++
Routine Description:
This function translates int 21 function 25
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
SetProtectedModeInterrupt(getAL(), getDS(), (*GetDXRegister)(),
(USHORT)(Frame32 ? VDM_INT_32 : VDM_INT_16));
}
VOID
CreatePSP(
VOID
)
/*++
Routine Description:
This routine translates the selector to a segment and reflects the
calls
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
ULONG Segment;
USHORT ClientDX;
DpmiSwitchToRealMode();
ClientDX = getDX();
Segment = SELECTOR_TO_INTEL_LINEAR_ADDRESS(ClientDX);
if (Segment > ONE_MB) {
//
// Create PSP doesn't do anything on error
//
} else {
setDX((USHORT) (Segment >> 4));
DPMI_EXEC_INT(0x21);
}
setDX(ClientDX);
DpmiSwitchToProtectedMode();
}
VOID
ParseFilename(
VOID
)
/*++
Routine Description:
This routine translates the int 21 parse filename api
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientSI, ClientDI, FcbLength, StringOff, Seg, Off;
PUCHAR Fcb, BufferedFcb, String, BufferedString;
USHORT ClientDS = getDS();
USHORT ClientES = getES();
DpmiSwitchToRealMode();
ClientSI = getSI();
ClientDI = getDI();
Fcb = Sim32GetVDMPointer(((ULONG)ClientES << 16), 1, TRUE)
+ (*GetDIRegister)();
FcbLength = DpmiCalcFcbLength(Fcb);
BufferedFcb = DpmiMapAndCopyBuffer(Fcb, FcbLength);
String = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetSIRegister)();
BufferedString = DpmiMapAndCopyBuffer(String, 20);
DPMI_FLAT_TO_SEGMENTED(BufferedFcb, &Seg, &Off);
setES(Seg);
setDI(Off);
DPMI_FLAT_TO_SEGMENTED(BufferedString, &Seg, &StringOff);
setDS(Seg);
setSI(StringOff);
DPMI_EXEC_INT(0x21);
DpmiUnmapAndCopyBuffer(Fcb, BufferedFcb, FcbLength);
DpmiUnmapAndCopyBuffer(String, BufferedString, 20);
setDI(ClientDI);
setSI(ClientSI + (getSI() - StringOff));
DpmiSwitchToProtectedMode();
setDS(ClientDS);
setES(ClientES);
}
VOID
GetDTA(
VOID
)
/*++
Routine Description:
This routine returns the current DTA.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
//
// Win31 compatibility:
//
// Some hosebag programs set the DTA to a selector that they later free!
// This test makes sure that this does not cause us to crash.
//
if (!SEGMENT_IS_WRITABLE(CurrentDtaSelector)) {
CurrentDtaSelector = 0;
CurrentDtaOffset = 0;
}
setES(CurrentDtaSelector);
setBX(CurrentDtaOffset);
setCF(0);
}
VOID
SetDTA(
VOID
)
/*++
Routine Description:
This routine sets the PM dta address. If the buffer is usable directly,
CurrentDta is set to the translated address. Otherwise, it is set to
the dosx dta.
N.B. The Set Dta call is not actually reflected to Dos until a function
call is made that actually uses the Dta, e.g. the Find First/Next
functions. This techique was implemented to match what is done in
Win 3.1.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientDX;
ClientDX = getDX();
CurrentDtaOffset = ClientDX;
CurrentDtaSelector = getDS();
//
// Make sure real DTA is updated later
//
CurrentDosDta = (PUCHAR) NULL;
}
VOID
SetDTAPointers(
VOID
)
/*++
Routine Description:
This routine sets up the flat address values used elsewhere to
reference the current DTA.
N.B. This must be done on every entry to handle function calls because
the PM dta may have been GlobalRealloc'd. This may change the base
address of the PM selector (which invalidates the flat addresses),
but not the selector/offset itself.
Arguments:
None.
Return Value:
None.
--*/
{
PUCHAR NewDta;
NewDta = Sim32GetVDMPointer(
(CurrentDtaSelector << 16),
1,
TRUE
);
NewDta += CurrentDtaOffset;
CurrentPmDtaAddress = NewDta;
//
// If the new dta is not accessible in v86 mode, use the one
// supplied by Dosx
//
if ((ULONG)(NewDta + 128 - IntelBase) > MAX_V86_ADDRESS) {
NewDta = DosxDtaBuffer;
}
//
// Update Current DTA information
//
CurrentDta = NewDta;
}
VOID
SetDosDTA(
VOID
)
/*++
Routine Description:
This routine is called internally by other functions in this module
to reflect a Set Dta call to Dos.
WARNING: The client must be in REAL mode
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientAX, ClientDX, ClientDS, NewDtaSegment, NewDtaOffset;
ASSERT((getMSW() & MSW_PE) == 0);
ClientAX = getAX();
ClientDX = getDX();
ClientDS = getDS();
DPMI_FLAT_TO_SEGMENTED(CurrentDta, &NewDtaSegment, &NewDtaOffset);
setDS(NewDtaSegment);
setDX(NewDtaOffset);
setAH(0x1a);
DPMI_EXEC_INT(0x21);
setAX(ClientAX);
setDX(ClientDX);
setDS(ClientDS);
CurrentDosDta = CurrentDta;
}
VOID
TSR(
VOID
)
/*++
Routine Description:
This function translates int 21h tsr. Win 3.1 does some
magic here. We didn't before and it worked fine. Maybe
we will later.
Arguments:
None.
Return Value:
None.
--*/
{
NoTranslation();
}
VOID
GetDevParamBlock(
VOID
)
/*++
Routine Description:
This routine translates the Device Parameter Block apis
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT Selector;
DpmiSwitchToRealMode();
DPMI_EXEC_INT(0x21);
Selector = getDS();
DpmiSwitchToProtectedMode();
setDS(SegmentToSelector(Selector, STD_DATA));
(*SetBXRegister)((ULONG)getBX());
}
VOID
ReturnESBX(
VOID
)
/*++
Routine Description:
This function translates api that return information in es:bx
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT Selector;
DpmiSwitchToRealMode();
DPMI_EXEC_INT(0x21);
Selector = getES();
DpmiSwitchToProtectedMode();
(*SetBXRegister)((ULONG)getBX());
setES(SegmentToSelector(Selector, STD_DATA));
}
VOID
GetVector(
VOID
)
/*++
Routine Description:
This function translates int 21 function 35
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PVDM_INTERRUPTHANDLER Handlers = DpmiInterruptHandlers;
USHORT IntNumber = getAL();
setES(Handlers[IntNumber].CsSelector);
(*SetBXRegister)(Handlers[IntNumber].Eip);
}
VOID
GetSetCountry(
VOID
)
/*++
Routine Description:
This function translates int 21 function 38
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
if (getDX() == 0xFFFF) {
NoTranslation();
} else {
USHORT ClientDX, Seg, Off;
PUCHAR Country, BufferedCountry;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
Country = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
BufferedCountry = DpmiMapAndCopyBuffer(Country, 34);
DPMI_FLAT_TO_SEGMENTED(BufferedCountry, &Seg, &Off);
setDS(Seg);
setDX(Off);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(Country, BufferedCountry, 34);
setDX(ClientDX);
setDS(ClientDS);
}
}
VOID
MapASCIIZDSDX(
VOID
)
/*++
Routine Description:
This routine maps the int 21 functions that pass file names in
ds:dx
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR BufferedString;
USHORT ClientDX, StringSeg, StringOff, Length;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
BufferedString = DpmiMapString(ClientDS, (GetDXRegister)(), &Length);
DPMI_FLAT_TO_SEGMENTED(BufferedString, &StringSeg, &StringOff);
setDS(StringSeg);
setDX(StringOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapString(BufferedString, Length);
setDX(ClientDX);
setDS(ClientDS);
}
VOID
ReadWriteFile(
VOID
)
/*++
Routine Description:
This routine translates the read/write file int 21 calls. Large reads
are broken down into 4k chunks.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientCX, ClientDX, ClientAX, Function, DataSeg, DataOff, BytesToRead;
ULONG BytesRead, TotalBytesToRead;
PUCHAR ReadWriteData, BufferedData;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientCX = getCX();
ClientDX = getDX();
Function = getAH();
ClientAX = getAX();
TotalBytesToRead = (*GetCXRegister)();
BytesRead = 0;
ReadWriteData = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
// while (TotalBytesToRead != BytesRead) {
do {
if ((TotalBytesToRead - BytesRead) > 1024 * 4) {
BytesToRead = 4 * 1024;
} else {
BytesToRead = (USHORT)(TotalBytesToRead - BytesRead);
}
BufferedData = DpmiMapAndCopyBuffer(ReadWriteData, BytesToRead);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &DataSeg, &DataOff);
setDS(DataSeg);
setDX(DataOff);
setCX(BytesToRead);
setAX(ClientAX);
DPMI_EXEC_INT(0x21);
if (getCF()) {
DpmiUnmapBuffer(BufferedData, BytesToRead);
break;
}
if (getAX() == 0) {
DpmiUnmapBuffer(BufferedData, BytesToRead);
break;
} else if (getAX() < BytesToRead) {
CopyMemory(ReadWriteData, BufferedData, getAX());
DpmiUnmapBuffer(BufferedData, BytesToRead);
BytesRead += getAX();
break;
} else {
DpmiUnmapAndCopyBuffer(
ReadWriteData,
BufferedData,
BytesToRead
);
}
ReadWriteData += getAX();
BytesRead += getAX();
// }
} while (TotalBytesToRead != BytesRead);
setDX(ClientDX);
setCX(ClientCX);
if (!getCF()) {
(*SetAXRegister)(BytesRead);
}
DpmiSwitchToProtectedMode();
setDS(ClientDS);
}
#define MAX_SUPPORTED_DOS_IOCTL_CALL 0x10
//
// Note: The translations here differ from those in the windows dpmi vxd,
// because we do not have to lock memory, and we don't support block
// device drivers.
//
APIXLATFUNCTION IOCTLXlatTable[MAX_SUPPORTED_DOS_IOCTL_CALL] = {
NoTranslation , // 00 - Get Device Data
NoTranslation , // 01 - Set Device Data
MapDSDXLenCX , // 02 - Receive Ctrl Chr Data
MapDSDXLenCX , // 03 - Send Ctrl Chr Data
MapDSDXLenCX , // 04 - Receive Ctrl Block Data
MapDSDXLenCX , // 05 - Send Ctrl Block Data
NoTranslation , // 06 - Check Input Status
NoTranslation , // 07 - Check Output Status
NoTranslation , // 08 - Check Block Dev Removable
NoTranslation , // 09 - Check Block Dev Remote
NoTranslation , // 0A - Check if Handle Remote
NoTranslation , // 0B - Change sharing retry cnt
IOCTLMap2Bytes , // 0C - MAP DS:DX LENGTH 2!
IOCTLBlockDevs , // 0D - Generic IOCTL to blk dev
NoTranslation , // 0E - Get Logical Drive Map
NoTranslation // 0F - Set Logical Drive Map
};
VOID
IOCTL(
VOID
)
/*++
Routine Description:
This function translates the int 21 ioctl function.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT IoctlMinor;
IoctlMinor = getAL();
if (IoctlMinor >= MAX_SUPPORTED_DOS_IOCTL_CALL) {
#if DBG
OutputDebugString("DPMI: IOCTL DOS CALL NOT SUPPORTED\n");
#endif
NoTranslation();
return;
}
(*IOCTLXlatTable[IoctlMinor])();
}
VOID
GetCurDir(
VOID
)
/*++
Routine Description:
This routine translates the get current directory call
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR DirInfo, BufferedDirInfo;
USHORT ClientSI, Seg, Off;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientSI = getSI();
DirInfo = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetSIRegister)();
BufferedDirInfo = DpmiMapAndCopyBuffer(DirInfo, 64);
DPMI_FLAT_TO_SEGMENTED(BufferedDirInfo, &Seg, &Off);
setDS(Seg);
setSI(Off);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(DirInfo, BufferedDirInfo, 64);
setSI(ClientSI);
setDS(ClientDS);
}
VOID
AllocateMemoryBlock(
VOID
)
/*++
Routine Description:
This routine translates the AllocateMemory Int 21 api
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PMEM_DPMI pMem;
ULONG MemSize = ((ULONG)getBX()) << 4;
pMem = DpmiAllocateXmem(MemSize);
if (pMem) {
pMem->SelCount = (USHORT) ((MemSize>>16) + 1);
pMem->Sel = ALLOCATE_SELECTORS(pMem->SelCount);
if (!pMem->Sel) {
pMem->SelCount = 0;
DpmiFreeXmem(pMem);
pMem = NULL;
} else {
SetDescriptorArray(pMem->Sel, (ULONG)pMem->Address, MemSize);
}
}
if (!pMem) {
setCF(1);
setAX(8);
setBX(0);
} else {
setCF(0);
setAX(pMem->Sel);
}
}
VOID
FreeMemoryBlock(
VOID
)
/*++
Routine Description:
This routine translates the ResizeMemory Int 21 api
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PMEM_DPMI pMem;
USHORT Sel = getES();
if (pMem = DpmiFindXmem(Sel)) {
while(pMem->SelCount--) {
FreeSelector(Sel);
Sel+=8;
}
DpmiFreeXmem(pMem);
setCF(0);
} else {
setCF(1);
setAX(9);
}
}
VOID
ResizeMemoryBlock(
VOID
)
/*++
Routine Description:
This routine translates the ResizeMemory int 21 api
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PMEM_DPMI pMem;
ULONG MemSize = ((ULONG)getBX()) << 4;
USHORT Sel = getES();
if (pMem = DpmiFindXmem(Sel)) {
if (DpmiReallocateXmem(pMem, MemSize)) {
SetDescriptorArray(pMem->Sel, (ULONG)pMem->Address, MemSize);
setCF(0);
} else {
// not enough memory
setCF(1);
setAX(8);
}
} else {
// invalid block
setCF(1);
setAX(9);
}
}
VOID
LoadExec(
VOID
)
/*++
Routine Description:
This function translates the int 21 load exec function. Load overlay
is not supported.
The child always inherits the environment, and the fcbs in the parameter
block are ignored. (win 3.1 does this)
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR CommandTail, BufferedString, Environment;
USHORT ClientDX, ClientBX, Seg, Off, Length, i, EnvironmentSel;
USHORT ClientDS = getDS();
USHORT ClientES = getES();
DpmiSwitchToRealMode();
ClientDX = getDX();
ClientBX = getBX();
if (getAL() == 3) {
setCF(1);
setAX(1);
} else {
//
// Make sure real DTA is updated later
//
CurrentDosDta = (PUCHAR) NULL;
//
// Map the command string
//
BufferedString = DpmiMapString(ClientDS, (*GetDXRegister)(), &Length);
//
// Set up the Parameter block
//
// We use the large xlat buffer. The parameter block comes
// first, and we fill in the command tail after
//
ZeroMemory(LargeXlatBuffer, 512);
//
// The environment segment address is now set
//
//
// Set the command tail address, and copy the command tail (all
// 128 bytes
//
DPMI_FLAT_TO_SEGMENTED((LargeXlatBuffer + 0x10), &Seg, &Off)
*(PWORD16)(LargeXlatBuffer + 2) = Off;
*(PWORD16)(LargeXlatBuffer + 4) = Seg;
//
// CommandTail = FLAT(es:bx)
//
CommandTail = Sim32GetVDMPointer(((ULONG)ClientES << 16), 1, TRUE)
+ (*GetBXRegister)();
if (CurrentAppFlags & DPMI_32BIT) {
//
// CommandTail -> string
//
CommandTail = Sim32GetVDMPointer((*(PWORD16)(CommandTail + 4)) << 16, 1, TRUE)
+ *(PDWORD16)(CommandTail);
} else {
//
// CommandTail -> string
//
CommandTail = Sim32GetVDMPointer(*(PDWORD16)(CommandTail + 2), 1, TRUE);
}
CopyMemory((LargeXlatBuffer + 0x10), CommandTail, 128);
//
// Set FCB pointers and put spaces in the file names
//
DPMI_FLAT_TO_SEGMENTED((LargeXlatBuffer + 144), &Seg, &Off)
*(PWORD16)(LargeXlatBuffer + 6) = Off;
*(PWORD16)(LargeXlatBuffer + 8) = Seg;
for (i = 0; i < 11; i++) {
(LargeXlatBuffer + 144 + 1)[i] = ' ';
}
DPMI_FLAT_TO_SEGMENTED((LargeXlatBuffer + 188), &Seg, &Off)
*(PWORD16)(LargeXlatBuffer + 0xA) = Off;
*(PWORD16)(LargeXlatBuffer + 0xC) = Seg;
for (i = 0; i < 11; i++) {
(LargeXlatBuffer + 188 + 1)[i] = ' ';
}
//
// Save the environment selector, and make it a segment
//
Environment = Sim32GetVDMPointer(((ULONG)CurrentPSPSelector << 16) | 0x2C, 1, TRUE);
EnvironmentSel = *(PWORD16)Environment;
*(PWORD16)Environment =
(USHORT)(SELECTOR_TO_INTEL_LINEAR_ADDRESS(EnvironmentSel) >> 4);
//
// Set up registers for the exec
//
DPMI_FLAT_TO_SEGMENTED(BufferedString, &Seg, &Off);
setDS(Seg);
setDX(Off);
DPMI_FLAT_TO_SEGMENTED(LargeXlatBuffer, &Seg, &Off);
setES(Seg);
setBX(Off);
DPMI_EXEC_INT(0x21);
//
// Restore the environment selector
//
Environment = Sim32GetVDMPointer(((ULONG)CurrentPSPSelector << 16) | 0x2C, 1, TRUE);
*(PWORD16)Environment = EnvironmentSel;
//
// Free translation buffer
//
DpmiUnmapString(BufferedString, Length);
}
setDX(ClientDX);
setBX(ClientBX);
DpmiSwitchToProtectedMode();
setES(ClientES);
setDS(ClientDS);
}
VOID
Terminate(
VOID
)
/*++
Routine Description:
description-of-function.
Arguments:
argument-name - Supplies | Returns description of argument.
.
.
Return Value:
return-value - Description of conditions needed to return value. - or -
None.
--*/
{
// bugbug We're currently mapping this one in the dos extender
}
VOID
FindFirstFileHandle(
VOID
)
/*++
Routine Description:
This routine translates the find first api.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientDX, Seg, Off, StringLength;
PUCHAR BufferedString;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
SetDTAPointers();
ClientDX = getDX();
//
// Copy the DTA (if necessary)
//
if (CurrentDta != CurrentPmDtaAddress) {
CopyMemory(CurrentDta, CurrentPmDtaAddress, 43);
}
//
// Check to see if we need to set the real dta
//
if (CurrentDosDta != CurrentDta)
SetDosDTA();
//
// map the string
//
BufferedString = DpmiMapString(ClientDS, (GetDXRegister)(), &StringLength);
DPMI_FLAT_TO_SEGMENTED(BufferedString, &Seg, &Off);
setDS(Seg);
setDX(Off);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
setDS(ClientDS);
DpmiUnmapString(BufferedString, StringLength);
//
// Copy the DTA back (if necessary)
//
if (CurrentDta != CurrentPmDtaAddress) {
CopyMemory(CurrentPmDtaAddress, CurrentDta, 43);
}
setDX(ClientDX);
}
VOID
FindNextFileHandle(
VOID
)
/*++
Routine Description:
This routine translates the find next api
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
DpmiSwitchToRealMode();
SetDTAPointers();
//
// Copy the DTA (if necessary)
//
if (CurrentDta != CurrentPmDtaAddress) {
CopyMemory(CurrentDta, CurrentPmDtaAddress, 43);
}
//
// Check to see if we need to set the real dta
//
if (CurrentDosDta != CurrentDta)
SetDosDTA();
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
//
// Copy the DTA back (if necessary)
//
if (CurrentDta != CurrentPmDtaAddress) {
CopyMemory(CurrentPmDtaAddress, CurrentDta, 43);
}
}
VOID
SetPSP(
VOID
)
/*++
Routine Description:
This routine translates the set psp api. This is substantially the
same as CreatePSP, except that this can fail (and return carry). It
also remembers the PSP selector, so we can return it on request.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
ULONG Segment;
USHORT ClientBX;
DpmiSwitchToRealMode();
ClientBX = getBX();
if (ClientBX == 0) {
CurrentPSPSelector = ClientBX;
} else {
Segment = SELECTOR_TO_INTEL_LINEAR_ADDRESS(ClientBX);
if (Segment > ONE_MB) {
setCF(1);
} else {
setBX((USHORT) (Segment >> 4));
DPMI_EXEC_INT(0x21);
CurrentPSPSelector = ClientBX;
}
}
setBX(ClientBX);
DpmiSwitchToProtectedMode();
}
VOID
GetPSP(
VOID
)
/*++
Routine Description:
This routine returns the current PSP selector
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
DpmiSwitchToRealMode();
//
// Get the current psp segment to see if it changed
//
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
//
// If it changed get a new selector for the psp
//
if (getBX() !=
(USHORT)(SELECTOR_TO_INTEL_LINEAR_ADDRESS(CurrentPSPSelector) >> 4)
){
CurrentPSPSelector = SegmentToSelector(getBX(), STD_DATA);
}
setBX(CurrentPSPSelector);
setCF(0);
}
VOID
TranslateBPB(
VOID
)
/*++
Routine Description:
This function fails and returns. On NT we do not support this dos
call.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
#if DBG
OutputDebugString("DPMI: Int 21 function 53 is not supported\n");
#endif
setCF(1);
}
VOID
RenameFile(
VOID
)
/*++
Routine Description:
This routine translates the rename int 21 api
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR SourceString, DestinationString;
USHORT ClientDX, ClientDI, Seg, Off, SourceLength, DestinationLength;
USHORT ClientDS = getDS();
USHORT ClientES = getES();
DpmiSwitchToRealMode();
ClientDX = getDX();
ClientDI = getDI();
SourceString = DpmiMapString(ClientDS, (*GetDXRegister)(), &SourceLength);
DestinationString = DpmiMapString(ClientES, (*GetDIRegister)(), &DestinationLength);
DPMI_FLAT_TO_SEGMENTED(SourceString, &Seg, &Off);
setDX(Off);
setDS(Seg);
DPMI_FLAT_TO_SEGMENTED(DestinationString, &Seg, &Off);
setDI(Off);
setES(Seg);
DPMI_EXEC_INT(0x21);
setDX(ClientDX);
setDI(ClientDI);
DpmiSwitchToProtectedMode();
setDS(ClientDS);
setES(ClientES);
}
VOID
CreateTempFile(
VOID
)
/*++
Routine Description:
This function maps the int 21 create temp file api
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR String, BufferedString;
USHORT ClientDX, Seg, Off, Length;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
String = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
Length = 0;
while (String[Length] != '\0') {
Length++;
}
Length += 13;
BufferedString = DpmiMapAndCopyBuffer(String, Length);
DPMI_FLAT_TO_SEGMENTED(BufferedString, &Seg, &Off);
setDS(Seg);
setDX(Off);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(String, BufferedString, Length);
setDX(ClientDX);
setDS(ClientDS);
}
#define MAX_SUPPORTED_DOS_5D_CALL 12
APIXLATFUNCTION Func5DXlatTable[MAX_SUPPORTED_DOS_5D_CALL] = {
NotSupportedBad , // 0
MapDPL , // 1
NotSupportedBad , // 2
MapDPL , // 3
MapDPL , // 4
NotSupportedBad , // 5
NotSupportedBad , // 6
NoTranslation , // 7
NoTranslation , // 8
NoTranslation , // 9
MapDPL , // 10
NotSupportedBad // 11
};
VOID
Func5Dh(
VOID
)
/*++
Routine Description:
This function translates dos call 5d
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT Func5DhMinor;
Func5DhMinor = getAL();
if (Func5DhMinor >= MAX_SUPPORTED_DOS_5D_CALL) {
#if DBG
OutputDebugString("DPMI: DOS FUNCTION 5D UNSUPPORTED\n");
#endif
NoTranslation();
return;
}
(*Func5DXlatTable[Func5DhMinor])();
}
#define MAX_SUPPORTED_DOS_5E_CALL 4
APIXLATFUNCTION Func5EXlatTable[MAX_SUPPORTED_DOS_5E_CALL] = {
GetMachineName,
MapASCIIZDSDX,
GetPrinterSetup,
SetPrinterSetup
};
VOID
Func5Eh(
VOID
)
/*++
Routine Description:
description-of-function.
Arguments:
argument-name - Supplies | Returns description of argument.
.
.
Return Value:
return-value - Description of conditions needed to return value. - or -
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT Func5EhMinor;
Func5EhMinor = getAL();
if (Func5EhMinor >= MAX_SUPPORTED_DOS_5E_CALL) {
#if DBG
OutputDebugString("DPMI: DOS FUNCTION 5E UNSUPPORTED\n");
#endif
NoTranslation();
return;
}
(*Func5EXlatTable[Func5EhMinor])();
}
VOID
Func5Fh(
VOID
)
/*++
Routine Description:
description-of-function.
Arguments:
argument-name - Supplies | Returns description of argument.
.
.
Return Value:
return-value - Description of conditions needed to return value. - or -
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT Func5FMinor;
Func5FMinor = getAL();
if (Func5FMinor == 4) {
MapASCIIZDSSI();
return;
} else if ((Func5FMinor == 2) || (Func5FMinor == 3) ||
(Func5FMinor == 5)
){
USHORT ClientSI, ClientDI, DataOff, DataSeg;
PUCHAR Data16, BufferedData16, Data128, BufferedData128;
USHORT ClientDS = getDS();
USHORT ClientES = getES();
DpmiSwitchToRealMode();
ClientDI = getDI();
ClientSI = getSI();
Data16 = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetSIRegister)();
BufferedData16 = DpmiMapAndCopyBuffer(Data16, 16);
Data128 = Sim32GetVDMPointer(((ULONG)ClientES << 16), 1, TRUE)
+ (*GetDIRegister)();
BufferedData128 = DpmiMapAndCopyBuffer(Data128, 128);
DPMI_FLAT_TO_SEGMENTED(BufferedData16, &DataSeg, &DataOff);
setDS(DataSeg);
setSI(DataOff);
DPMI_FLAT_TO_SEGMENTED(BufferedData128, &DataSeg, &DataOff);
setES(DataSeg);
setDI(DataOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(Data16, BufferedData16, 16);
DpmiUnmapAndCopyBuffer(Data128, BufferedData128, 128);
setDS(ClientDS);
setES(ClientES);
setSI(ClientSI);
setDI(ClientDI);
} else {
#if DBG
OutputDebugString("DPMI: UNSUPPORTED INT 21 FUNCTION 5F\n");
#endif
NoTranslation();
}
}
VOID
NotSupportedBad(
VOID
)
/*++
Routine Description:
description-of-function.
Arguments:
argument-name - Supplies | Returns description of argument.
.
.
Return Value:
return-value - Description of conditions needed to return value. - or -
None.
--*/
{
#if DBG
DECLARE_LocalVdmContext;
DbgPrint("WARNING: DOS INT 21 call AX= %x will not be translated.\n", getAH());
DbgPrint(" Use of this call is not supported from Prot\n");
DbgPrint(" mode applications.\n");
#endif
NoTranslation();
}
VOID
ReturnDSSI(
VOID
)
/*++
Routine Description:
This function translates api that return information in ds:si
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT Selector;
DpmiSwitchToRealMode();
DPMI_EXEC_INT(0x21);
Selector = getDS();
DpmiSwitchToProtectedMode();
(*SetSIRegister)((ULONG)getSI());
setDS(SegmentToSelector(Selector, STD_DATA));
}
VOID
NotSupportedBetter(
VOID
)
/*++
Routine Description:
description-of-function.
Arguments:
argument-name - Supplies | Returns description of argument.
.
.
Return Value:
return-value - Description of conditions needed to return value. - or -
None.
--*/
{
#if DBG
DECLARE_LocalVdmContext;
DbgPrint("WARNING: DOS INT 21 call AX= %x will not be translated.", getAH());
DbgPrint(" Use of this call by a Prot Mode app is not");
DbgPrint(" appropriate. There is a better INT 21 call, or a");
DbgPrint(" Windows call which should be used instead of this.");
#endif
NoTranslation();
}
VOID
GetExtendedCountryInfo(
VOID
)
/*++
Routine Description:
This routine translates the get extended country info int 21 api
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR Country, BufferedCountry;
USHORT ClientDI, Seg, Off, Length;
USHORT ClientES = getES();
DpmiSwitchToRealMode();
ClientDI = getDI();
Length = getCX();
Country = Sim32GetVDMPointer(((ULONG)ClientES << 16), 1, TRUE)
+ (*GetDIRegister)();
BufferedCountry = DpmiMapAndCopyBuffer(Country, Length);
DPMI_FLAT_TO_SEGMENTED(BufferedCountry, &Seg, &Off);
setES(Seg);
setDI(Off);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
setES(ClientES);
DpmiUnmapAndCopyBuffer(Country, BufferedCountry, Length);
setDI(ClientDI);
}
VOID
MapASCIIZDSSI(
VOID
)
/*++
Routine Description:
This function translates the int 21 extended open call
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
PUCHAR BufferedString;
USHORT ClientSI, StringSeg, StringOff, Length;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientSI = getSI();
BufferedString = DpmiMapString(ClientDS, (*GetSIRegister)(), &Length);
DPMI_FLAT_TO_SEGMENTED(BufferedString, &StringSeg, &StringOff);
setDS(StringSeg);
setSI(StringOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapString(BufferedString, Length);
setSI(ClientSI);
setDS(ClientDS);
}
VOID
MapDSDXLenCX(
VOID
)
/*++
Routine Description:
This function maps the ioctl calls that pass data in DS:DX, with
length in cx
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientDX, ClientCX, DataSeg, DataOff;
PUCHAR Data, BufferedData;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
ClientCX = getCX();
Data = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
BufferedData = DpmiMapAndCopyBuffer(Data, ClientCX);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &DataSeg, &DataOff);
setDS(DataSeg);
setDX(DataOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(Data, BufferedData, ClientCX);
setDS(ClientDS);
setDX(ClientDX);
}
VOID
IOCTLMap2Bytes(
VOID
)
/*++
Routine Description:
description-of-function.
Arguments:
argument-name - Supplies | Returns description of argument.
.
.
Return Value:
return-value - Description of conditions needed to return value. - or -
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientDX, DataSeg, DataOff;
PUCHAR Data, BufferedData;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
Data = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE);
BufferedData = DpmiMapAndCopyBuffer(Data, 2);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &DataSeg, &DataOff);
setDS(DataSeg);
setDX(DataOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(Data, BufferedData, 2);
setDS(ClientDS);
setDX(ClientDX);
}
VOID
IOCTLBlockDevs(
VOID
)
/*++
Routine Description:
This function fails the block device ioctls
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT IoctlSubFunction, Seg, Off, ClientDX;
PUCHAR Data, BufferedData;
USHORT Length;
USHORT ClientDS = getDS();
IoctlSubFunction = getCL();
if ((IoctlSubFunction < 0x40) || (IoctlSubFunction > 0x42) &&
(IoctlSubFunction < 0x60) || (IoctlSubFunction > 0x62) &&
(IoctlSubFunction != 0x68)
) {
#if DBG
OutputDebugString("DPMI: IOCTL DOS CALL NOT SUPPORTED\n");
#endif
NoTranslation();
return;
}
//
// Read and write track are special (and a pain)
//
if ((IoctlSubFunction == 0x41) || (IoctlSubFunction == 0x61)) {
IoctlReadWriteTrack();
return;
}
DpmiSwitchToRealMode();
ClientDX = getDX();
Data = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
switch (IoctlSubFunction) {
case 0x40:
//
// Map set device params
//
Length = (*(PWORD16)(Data + 0x26));
Length <<= 2;
Length += 0x28;
break;
case 0x60:
//
// Map get device params
//
Length = 38;
break;
case 0x62:
//
// Map format verify
//
Length = 5;
break;
case 0x68:
//
// Map Media sense
//
Length = 4;
break;
}
BufferedData = DpmiMapAndCopyBuffer(Data, Length);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &Seg, &Off);
setDS(Seg);
setDX(Off);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(Data, BufferedData, Length);
setDS(ClientDS);
setDX(ClientDX);
}
VOID
IoctlReadWriteTrack(
VOID
)
/*++
Routine Description:
This routine maps the read/write track ioctl.
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientDX, ClientDS, ClientCX, ClientAX;
USHORT Seg, Off, NumberOfSectors, BytesPerSector;
USHORT SectorsRead, SectorsToRead;
PUCHAR ParameterBlock, BufferedPBlock, Data, BufferedData;
ClientAX = getAX();
ClientDX = getDX();
ClientCX = getCX();
ClientDS = getDS();
DpmiSwitchToRealMode();
//
// Find out how many bytes/sector
//
BufferedData = DpmiAllocateBuffer(0x40);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &Seg, &Off);
setDS(Seg);
setDX(Off);
setAX(0x440D);
setCX(0x860);
DPMI_EXEC_INT(0x21);
if (getCF()) {
//
// Failed, we don't know how much data to buffer,
// so fail read/write track
//
DpmiFreeBuffer(BufferedData, 0x40);
setDX(ClientDX);
setCX(ClientCX);
DpmiSwitchToProtectedMode();
setDS(ClientDS);
return;
}
//
// Get the number of bytes/sector
//
BytesPerSector = *(PWORD16)(BufferedData + 0x7);
DpmiFreeBuffer(BufferedData, 0x40);
setDX(ClientDX);
//
// First map the parameter block
//
ParameterBlock = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
BufferedPBlock = DpmiMapAndCopyBuffer(ParameterBlock, 13);
//
// Get the segment and offset of the parameter block
//
DPMI_FLAT_TO_SEGMENTED(BufferedPBlock, &Seg, &Off);
setDS(Seg);
setDX(Off);
if (CurrentAppFlags & DPMI_32BIT) {
Data = Sim32GetVDMPointer(
(*((PWORD16)(BufferedPBlock + 0xd)) << 16),
1,
TRUE
);
Data += *((PDWORD16)(BufferedPBlock + 0x9));
} else {
Data = Sim32GetVDMPointer(
(*((PWORD16)(BufferedPBlock + 0xb)) << 16),
1,
TRUE
);
Data += *((PWORD16)(BufferedPBlock + 0x9));
}
NumberOfSectors = *((PWORD16)(BufferedPBlock + 7));
SectorsRead = 0;
while (NumberOfSectors != SectorsRead) {
if ((NumberOfSectors - SectorsRead) * BytesPerSector > 1024 * 4) {
SectorsToRead = 4 * 1024 / BytesPerSector;
} else {
SectorsToRead = (USHORT)(NumberOfSectors - SectorsRead);
}
BufferedData = DpmiMapAndCopyBuffer(
Data,
(USHORT) (SectorsToRead * BytesPerSector)
);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &Seg, &Off);
*((PWORD16)(BufferedPBlock + 9)) = Off;
*((PWORD16)(BufferedPBlock + 11)) = Seg;
*((PWORD16)(BufferedPBlock + 7)) = SectorsToRead;
setAX(ClientAX);
setCX(ClientCX);
DPMI_EXEC_INT(0x21);
if (getCF()) {
DpmiUnmapBuffer(
BufferedData,
(USHORT) (SectorsToRead * BytesPerSector)
);
break;
}
DpmiUnmapAndCopyBuffer(
Data,
BufferedData,
(USHORT) (SectorsToRead * BytesPerSector)
);
Data += SectorsToRead * BytesPerSector;
*((PWORD16)(BufferedPBlock + 5)) += SectorsToRead;
SectorsRead += SectorsToRead;
}
DpmiUnmapBuffer(BufferedPBlock,13);
setDX(ClientDX);
DpmiSwitchToProtectedMode();
setDS(ClientDS);
}
VOID
MapDPL(
VOID
)
/*++
Routine Description:
This routine maps a DPL for the server call
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientDX, DataSeg, DataOff;
PUCHAR Data, BufferedData;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
Data = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
BufferedData = DpmiMapAndCopyBuffer(Data, 22);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &DataSeg, &DataOff);
setDS(DataSeg);
setDX(DataOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(Data, BufferedData, 22);
setDX(ClientDX);
setDS(ClientDS);
}
VOID
GetMachineName(
VOID
)
/*++
Routine Description:
This routine maps a machine name for int 21 function 5e
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientDX, DataSeg, DataOff;
PUCHAR Data, BufferedData;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientDX = getDX();
Data = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetDXRegister)();
BufferedData = DpmiMapAndCopyBuffer(Data, 16);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &DataSeg, &DataOff);
setDS(DataSeg);
setDX(DataOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(Data, BufferedData, 16);
setDX(ClientDX);
setDS(ClientDS);
}
VOID
GetPrinterSetup(
VOID
)
/*++
Routine Description:
This routine maps printer setup data for int 21 function 5e
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientSI, ClientCX, DataSeg, DataOff;
PUCHAR Data, BufferedData;
USHORT ClientDS = getDS();
DpmiSwitchToRealMode();
ClientSI = getSI();
ClientCX = getCX();
Data = Sim32GetVDMPointer(((ULONG)ClientDS << 16), 1, TRUE)
+ (*GetSIRegister)();
BufferedData = DpmiMapAndCopyBuffer(Data, ClientCX);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &DataSeg, &DataOff);
setDS(DataSeg);
setSI(DataOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
DpmiUnmapAndCopyBuffer(Data, BufferedData, ClientCX);
setSI(ClientSI);
setDS(ClientDS);
}
VOID
SetPrinterSetup(
VOID
)
/*++
Routine Description:
This routine maps printer setup data for int 21 function 5e
Arguments:
None.
Return Value:
None.
--*/
{
DECLARE_LocalVdmContext;
USHORT ClientDI, DataSeg, DataOff;
PUCHAR Data, BufferedData;
USHORT ClientES = getES();
DpmiSwitchToRealMode();
ClientDI = getDI();
Data = Sim32GetVDMPointer(((ULONG)ClientES << 16), 1, TRUE)
+ (*GetDIRegister)();
BufferedData = DpmiMapAndCopyBuffer(Data, 64);
DPMI_FLAT_TO_SEGMENTED(BufferedData, &DataSeg, &DataOff);
setES(DataSeg);
setDI(DataOff);
DPMI_EXEC_INT(0x21);
DpmiSwitchToProtectedMode();
setES(ClientES);
DpmiUnmapAndCopyBuffer(Data, BufferedData, 64);
setDI(ClientDI);
}
VOID
GetDate(
VOID
)
/*++
Routine Description:
This routine maps int21 func 2A GetDate
Arguments:
None.
Return Value:
Client (DH) - month
Client (DL) - Day
Client (CX) - Year
Client (AL) - WeekDay
--*/
{
DECLARE_LocalVdmContext;
SYSTEMTIME TimeDate;
GetLocalTime(&TimeDate);
setDH((UCHAR)TimeDate.wMonth);
setDL((UCHAR)TimeDate.wDay);
setCX(TimeDate.wYear);
setAL((UCHAR)TimeDate.wDayOfWeek);
}