6287 lines
170 KiB
C
6287 lines
170 KiB
C
/*****************************************************************************
|
|
*
|
|
* (C) Copyright MICROSOFT Corp., 1988-1990
|
|
*
|
|
* Title: VMM.H - Include file for Virtual Machine Manager
|
|
*
|
|
* Version: 1.00
|
|
*
|
|
* Date: 05-May-1988
|
|
*
|
|
* Author: RAL
|
|
*
|
|
*-----------------------------------------------------------------------------
|
|
*
|
|
* Change log:
|
|
*
|
|
* DATE REV DESCRIPTION
|
|
* ----------- --- -----------------------------------------------------------
|
|
* 05-May-1988 RAL Original
|
|
* 13-Nov-1991 PBS C version
|
|
* 17-Dec-1993 Adds Far East VxDs identifiers
|
|
*/
|
|
|
|
#ifndef _VMM_
|
|
#define _VMM_
|
|
|
|
|
|
/*
|
|
* NON Windows/386 Virtual Device sources can include this file to get
|
|
* some useful equates by declaring the symbol "Not_VxD" If this symbol
|
|
* is defined, then everything that has to do with the specifics of the
|
|
* 32 bit environment for virtual devices is removed. Useful equates
|
|
* include: device ID's, pushad structure, BeginDoc, EndDoc, BeginMsg,
|
|
* EndMsg, page table equates, etc.
|
|
*/
|
|
|
|
#define FALSE 0 // False
|
|
#define VMM_TRUE (~FALSE) // The opposite of False!
|
|
|
|
#define DEBLEVELRETAIL 0
|
|
#define DEBLEVELNORMAL 1
|
|
#define DEBLEVELMAX 2
|
|
|
|
#ifndef DEBLEVEL
|
|
#ifdef DEBUG
|
|
#define DEBLEVEL DEBLEVELNORMAL
|
|
#else
|
|
#define DEBLEVEL DEBLEVELRETAIL
|
|
#endif
|
|
#endif
|
|
|
|
#ifndef WIN31COMPAT
|
|
#define WIN40SERVICES
|
|
#define WIN403SERVICES /*OPK-3 Services*/
|
|
#endif
|
|
|
|
#ifndef WIN40COMPAT
|
|
#define WIN41SERVICES
|
|
#endif
|
|
|
|
/* ASM
|
|
ifdef MASM6
|
|
ifndef NO_MASM6_OPTIONS
|
|
;
|
|
; option switches necessary to build VMM/VxD sources with MASM 6
|
|
;
|
|
option oldmacros
|
|
ifndef NEWSTRUCTS ; define NEWSTRUCTS for MASM6 struct semantics
|
|
option oldstructs
|
|
endif
|
|
option noscoped
|
|
option segment:flat
|
|
option offset:flat
|
|
option proc:private
|
|
endif
|
|
endif
|
|
;
|
|
; These null macros are recognized by a utility program that produces
|
|
; documentation files.
|
|
;
|
|
IFDEF MASM6
|
|
BeginDoc MACRO
|
|
ENDM
|
|
EndDoc MACRO
|
|
ENDM
|
|
|
|
BeginMsg MACRO
|
|
ENDM
|
|
EndMsg MACRO
|
|
ENDM
|
|
ELSE
|
|
BeginDoc EQU <>
|
|
EndDoc EQU <>
|
|
|
|
BeginMsg EQU <>
|
|
EndMsg EQU <>
|
|
ENDIF
|
|
*/
|
|
|
|
|
|
/******************************************************************************
|
|
*
|
|
* EQUATES FOR REQUIRED DEVICES
|
|
*
|
|
* Device ID formulation note:
|
|
*
|
|
* The high bit of the device ID is reserved for future use.
|
|
* Microsoft reserves the device ID's 0-1FFh for standard devices. If
|
|
* an OEM VxD is a replacement for a standard VxD, then it must use the
|
|
* standard VxD ID.
|
|
*
|
|
* OEMS WHO WANT A VXD DEVICE ID ASSIGNED TO THEM,
|
|
* PLEASE CONTACT MICROSOFT PRODUCT SUPPORT. ID's are only required for
|
|
* devices which provide services, V86 API's or PM API's. Also, calling
|
|
* services or API's by VxD name is now supported in version 4.0, so an
|
|
* ID may not be necessary as long as a unique 8 character name is used.
|
|
*
|
|
*****************************************************************************/
|
|
|
|
#define UNDEFINED_DEVICE_ID 0x00000
|
|
#define VMM_DEVICE_ID 0x00001 /* Used for dynalink table */
|
|
#define DEBUG_DEVICE_ID 0x00002
|
|
#define VPICD_DEVICE_ID 0x00003
|
|
#define VDMAD_DEVICE_ID 0x00004
|
|
#define VTD_DEVICE_ID 0x00005
|
|
#define V86MMGR_DEVICE_ID 0x00006
|
|
#define PAGESWAP_DEVICE_ID 0x00007
|
|
#define PARITY_DEVICE_ID 0x00008
|
|
#define REBOOT_DEVICE_ID 0x00009
|
|
#define VDD_DEVICE_ID 0x0000A
|
|
#define VSD_DEVICE_ID 0x0000B
|
|
#define VMD_DEVICE_ID 0x0000C
|
|
#define VKD_DEVICE_ID 0x0000D
|
|
#define VCD_DEVICE_ID 0x0000E
|
|
#define VPD_DEVICE_ID 0x0000F
|
|
#define BLOCKDEV_DEVICE_ID 0x00010
|
|
#define VMCPD_DEVICE_ID 0x00011
|
|
#define EBIOS_DEVICE_ID 0x00012
|
|
#define BIOSXLAT_DEVICE_ID 0x00013
|
|
#define VNETBIOS_DEVICE_ID 0x00014
|
|
#define DOSMGR_DEVICE_ID 0x00015
|
|
#define WINLOAD_DEVICE_ID 0x00016
|
|
#define SHELL_DEVICE_ID 0x00017
|
|
#define VMPOLL_DEVICE_ID 0x00018
|
|
#define VPROD_DEVICE_ID 0x00019
|
|
#define DOSNET_DEVICE_ID 0x0001A
|
|
#define VFD_DEVICE_ID 0x0001B
|
|
#define VDD2_DEVICE_ID 0x0001C /* Secondary display adapter */
|
|
#define WINDEBUG_DEVICE_ID 0x0001D
|
|
#define TSRLOAD_DEVICE_ID 0x0001E /* TSR instance utility ID */
|
|
#define BIOSHOOK_DEVICE_ID 0x0001F /* Bios interrupt hooker VxD */
|
|
#define INT13_DEVICE_ID 0x00020
|
|
#define PAGEFILE_DEVICE_ID 0x00021 /* Paging File device */
|
|
#define SCSI_DEVICE_ID 0x00022 /* SCSI device */
|
|
#define MCA_POS_DEVICE_ID 0x00023 /* MCA_POS device */
|
|
#define SCSIFD_DEVICE_ID 0x00024 /* SCSI FastDisk device */
|
|
#define VPEND_DEVICE_ID 0x00025 /* Pen device */
|
|
#define APM_DEVICE_ID 0x00026 /* Power Management device */
|
|
#define VPOWERD_DEVICE_ID APM_DEVICE_ID /* We overload APM since we replace it */
|
|
#define VXDLDR_DEVICE_ID 0x00027 /* VxD Loader device */
|
|
#define NDIS_DEVICE_ID 0x00028 /* NDIS wrapper */
|
|
#define BIOS_EXT_DEVICE_ID 0x00029 /* Fix Broken BIOS device */
|
|
#define VWIN32_DEVICE_ID 0x0002A /* for new WIN32-VxD */
|
|
#define VCOMM_DEVICE_ID 0x0002B /* New COMM device driver */
|
|
#define SPOOLER_DEVICE_ID 0x0002C /* Local Spooler */
|
|
#define WIN32S_DEVICE_ID 0x0002D /* Win32S on Win 3.1 driver */
|
|
#define DEBUGCMD_DEVICE_ID 0x0002E /* Debug command extensions */
|
|
/* #define RESERVED_DEVICE_ID 0x0002F /* Not currently in use */
|
|
/* #define ATI_HELPER_DEVICE_ID 0x00030 /* grabbed by ATI */
|
|
|
|
/* 31-32 USED BY WFW NET COMPONENTS */
|
|
/* #define VNB_DEVICE_ID 0x00031 /* Netbeui of snowball */
|
|
/* #define SERVER_DEVICE_ID 0x00032 /* Server of snowball */
|
|
|
|
#define CONFIGMG_DEVICE_ID 0x00033 /* Configuration manager (Plug&Play) */
|
|
#define DWCFGMG_DEVICE_ID 0x00034 /* Configuration manager for win31 and DOS */
|
|
#define SCSIPORT_DEVICE_ID 0x00035 /* Dragon miniport loader/driver */
|
|
#define VFBACKUP_DEVICE_ID 0x00036 /* allows backup apps to work with NEC */
|
|
#define ENABLE_DEVICE_ID 0x00037 /* for access VxD */
|
|
#define VCOND_DEVICE_ID 0x00038 /* Virtual Console Device - check vcond.inc */
|
|
/* 39 used by WFW VFat Helper device */
|
|
|
|
/* 3A used by WFW E-FAX */
|
|
/* #define EFAX_DEVICE_ID 0x0003A /* EFAX VxD ID */
|
|
|
|
/* 3B used by MS-DOS 6.1 for the DblSpace VxD which has APIs */
|
|
/* #define DSVXD_DEVICE_ID 0x0003B /* Dbl Space VxD ID */
|
|
|
|
#define ISAPNP_DEVICE_ID 0x0003C /* ISA P&P Enumerator */
|
|
#define BIOS_DEVICE_ID 0x0003D /* BIOS P&P Enumerator */
|
|
/* #define WINSOCK_DEVICE_ID 0x0003E /* WinSockets */
|
|
/* #define WSIPX_DEVICE_ID 0x0003F /* WinSockets for IPX */
|
|
|
|
#define IFSMgr_Device_ID 0x00040 /* Installable File System Manager */
|
|
#define VCDFSD_DEVICE_ID 0x00041 /* Static CDFS ID */
|
|
#define MRCI2_DEVICE_ID 0x00042 /* DrvSpace compression engine */
|
|
#define PCI_DEVICE_ID 0x00043 /* PCI P&P Enumerator */
|
|
#define PELOADER_DEVICE_ID 0x00044 /* PE Image Loader */
|
|
#define EISA_DEVICE_ID 0x00045 /* EISA P&P Enumerator */
|
|
#define DRAGCLI_DEVICE_ID 0x00046 /* Dragon network client */
|
|
#define DRAGSRV_DEVICE_ID 0x00047 /* Dragon network server */
|
|
#define PERF_DEVICE_ID 0x00048 /* Config/stat info */
|
|
|
|
#define AWREDIR_DEVICE_ID 0x00049 /* AtWork Network FSD */
|
|
#define DDS_DEVICE_ID 0x0004A /* Device driver services */
|
|
#define NTKERN_DEVICE_ID 0x0004B /* NT kernel device id */
|
|
#define VDOSKEYD_DEVICE_ID 0x0004B /* DOSKEY device id */
|
|
|
|
/*
|
|
* Far East DOS support VxD ID
|
|
*/
|
|
|
|
#define ETEN_Device_ID 0x00060 /* ETEN DOS (Taiwan) driver */
|
|
#define CHBIOS_Device_ID 0x00061 /* CHBIOS DOS (Korean) driver */
|
|
#define VMSGD_Device_ID 0x00062 /* DBCS Message Mode driver */
|
|
#define VPPID_Device_ID 0x00063 /* PC-98 System Control PPI */
|
|
#define VIME_Device_ID 0x00064 /* Virtual DOS IME */
|
|
#define VHBIOSD_Device_ID 0x00065 /* HBIOS (Korean) for HWin31 driver */
|
|
|
|
#define BASEID_FOR_NAMEBASEDVXD 0xf000 /* Name based VxD IDs start here */
|
|
#define BASEID_FOR_NAMEBASEDVXD_MASK 0x0fff /* Mask to get the real vxd id */
|
|
/*
|
|
* Initialization order equates. Devices are initialized in order from
|
|
* LOWEST to HIGHEST. If 2 or more devices have the same initialization
|
|
* order value, then they are initialized in order of occurance, so a
|
|
* specific order is not guaranteed. Holes have been left to allow maximum
|
|
* flexibility in ordering devices.
|
|
*/
|
|
|
|
#define VMM_INIT_ORDER 0x000000000
|
|
#define DEBUG_INIT_ORDER 0x000000000 /* normally using 0 is bad */
|
|
#define DEBUGCMD_INIT_ORDER 0x000000000 /* but debug must be first */
|
|
#define PERF_INIT_ORDER 0x000900000
|
|
#define APM_INIT_ORDER 0x001000000
|
|
#define VPOWERD_INIT_ORDER APM_INIT_ORDER /* We overload APM since we replace it */
|
|
#define BIOSHOOK_INIT_ORDER 0x006000000
|
|
#define VPROD_INIT_ORDER 0x008000000
|
|
#define VPICD_INIT_ORDER 0x00C000000
|
|
#define VTD_INIT_ORDER 0x014000000
|
|
#define VWIN32_INIT_ORDER 0x014100000
|
|
#define NTKERN_INIT_ORDER 0x015000000 /* Must be before VxDLdr (so that it is ready for devnodes) */
|
|
#define VXDLDR_INIT_ORDER 0x016000000
|
|
|
|
#define ENUMERATOR_INIT_ORDER 0x016800000 /* Should be before IOS */
|
|
#define ISAPNP_INIT_ORDER ENUMERATOR_INIT_ORDER
|
|
#define EISA_INIT_ORDER ENUMERATOR_INIT_ORDER
|
|
#define PCI_INIT_ORDER ENUMERATOR_INIT_ORDER
|
|
#define BIOS_INIT_ORDER ENUMERATOR_INIT_ORDER+1 /* To simplify reenumeration */
|
|
#define CONFIGMG_INIT_ORDER ENUMERATOR_INIT_ORDER+0xFFFF /* After all enumerators */
|
|
|
|
#define VCDFSD_INIT_ORDER 0x016F00000
|
|
#define IOS_INIT_ORDER 0x017000000
|
|
#define PAGEFILE_INIT_ORDER 0x018000000
|
|
#define PAGESWAP_INIT_ORDER 0x01C000000
|
|
#define PARITY_INIT_ORDER 0x020000000
|
|
#define REBOOT_INIT_ORDER 0x024000000
|
|
#define EBIOS_INIT_ORDER 0x026000000
|
|
#define VDD_INIT_ORDER 0x028000000
|
|
#define VSD_INIT_ORDER 0x02C000000
|
|
|
|
#define VCD_INIT_ORDER 0x030000000
|
|
#define COMMDRVR_INIT_ORDER (VCD_INIT_ORDER - 1)
|
|
#define PRTCL_INIT_ORDER (COMMDRVR_INIT_ORDER - 2)
|
|
#define MODEM_INIT_ORDER (COMMDRVR_INIT_ORDER - 3)
|
|
#define PORT_INIT_ORDER (COMMDRVR_INIT_ORDER - 4)
|
|
|
|
#define VMD_INIT_ORDER 0x034000000
|
|
#define VKD_INIT_ORDER 0x038000000
|
|
#define VPD_INIT_ORDER 0x03C000000
|
|
#define BLOCKDEV_INIT_ORDER 0x040000000
|
|
#define MCA_POS_INIT_ORDER 0x041000000
|
|
#define SCSIFD_INIT_ORDER 0x041400000
|
|
#define SCSIMASTER_INIT_ORDER 0x041800000
|
|
#define INT13_INIT_ORDER 0x042000000
|
|
#define VMCPD_INIT_ORDER 0x048000000
|
|
#define BIOSXLAT_INIT_ORDER 0x050000000
|
|
#define VNETBIOS_INIT_ORDER 0x054000000
|
|
#define DOSMGR_INIT_ORDER 0x058000000
|
|
#define DOSNET_INIT_ORDER 0x05C000000
|
|
#define WINLOAD_INIT_ORDER 0x060000000
|
|
#define VMPOLL_INIT_ORDER 0x064000000
|
|
|
|
#define UNDEFINED_INIT_ORDER 0x080000000
|
|
#define VCOND_INIT_ORDER UNDEFINED_INIT_ORDER
|
|
|
|
#define WINDEBUG_INIT_ORDER 0x081000000
|
|
#define VDMAD_INIT_ORDER 0x090000000
|
|
#define V86MMGR_INIT_ORDER 0x0A0000000
|
|
|
|
#define IFSMgr_Init_Order 0x10000 + V86MMGR_Init_Order
|
|
#define FSD_Init_Order 0x00100 + IFSMgr_Init_Order
|
|
#define VFD_INIT_ORDER 0x50000 + IFSMgr_Init_Order
|
|
|
|
/* Device that must touch memory in 1st Mb at crit init (after V86mmgr) */
|
|
#define UNDEF_TOUCH_MEM_INIT_ORDER 0x0A8000000
|
|
#define SHELL_INIT_ORDER 0x0B0000000
|
|
|
|
/* ASM
|
|
;******************************************************************************
|
|
;
|
|
; Macro to cause a delay in between I/O accesses to the same device.
|
|
;
|
|
;------------------------------------------------------------------------------
|
|
|
|
IO_Delay macro
|
|
jmp $+2
|
|
ENDM
|
|
*/
|
|
|
|
#define VXD_FAILURE 0
|
|
#define VXD_SUCCESS 1
|
|
|
|
typedef ULONG HVM; /* VM handle typedef */
|
|
|
|
/*
|
|
* Registers as they appear on the stack after a PUSHAD.
|
|
*/
|
|
|
|
struct Pushad_Struc {
|
|
ULONG Pushad_EDI; /* Client's EDI */
|
|
ULONG Pushad_ESI; /* Client's ESI */
|
|
ULONG Pushad_EBP; /* Client's EBP */
|
|
ULONG Pushad_ESP; /* ESP before pushad */
|
|
ULONG Pushad_EBX; /* Client's EBX */
|
|
ULONG Pushad_EDX; /* Client's EDX */
|
|
ULONG Pushad_ECX; /* Client's ECX */
|
|
ULONG Pushad_EAX; /* Client's EAX */
|
|
};
|
|
|
|
/* XLATOFF */
|
|
|
|
#ifdef RC_INVOKED
|
|
#define NOBASEDEFS
|
|
#endif
|
|
|
|
#ifndef NOBASEDEFS
|
|
|
|
#pragma warning (disable:4209) // turn off redefinition warning
|
|
|
|
typedef unsigned char UCHAR;
|
|
typedef unsigned short USHORT;
|
|
|
|
#pragma warning (default:4209) // turn off redefinition warning
|
|
|
|
#endif
|
|
|
|
#define GetVxDServiceOrdinal(service) __##service
|
|
|
|
#define Begin_Service_Table(device, seg) \
|
|
enum device##_SERVICES { \
|
|
device##_dummy = (device##_DEVICE_ID << 16) - 1,
|
|
|
|
#define Declare_Service(service, local) \
|
|
GetVxDServiceOrdinal(service),
|
|
|
|
#define Declare_SCService(service, args, local) \
|
|
GetVxDServiceOrdinal(service),
|
|
|
|
#define End_Service_Table(device, seg) \
|
|
Num_##device##_Services};
|
|
|
|
#define VXDINLINE static __inline
|
|
/* XLATON */
|
|
|
|
#ifndef Not_VxD
|
|
|
|
/* XLATOFF */
|
|
#define VxD_LOCKED_CODE_SEG code_seg("_LTEXT", "LCODE")
|
|
#define VxD_LOCKED_DATA_SEG data_seg("_LDATA", "LCODE")
|
|
#define VxD_INIT_CODE_SEG code_seg("_ITEXT", "ICODE")
|
|
#define VxD_INIT_DATA_SEG data_seg("_IDATA", "ICODE")
|
|
#define VxD_ICODE_SEG code_seg("_ITEXT", "ICODE")
|
|
#define VxD_IDATA_SEG data_seg("_IDATA", "ICODE")
|
|
#define VxD_PAGEABLE_CODE_SEG code_seg("_PTEXT", "PCODE")
|
|
#define VxD_PAGEABLE_DATA_SEG data_seg("_PDATA", "PDATA")
|
|
#define VxD_STATIC_CODE_SEG code_seg("_STEXT", "SCODE")
|
|
#define VxD_STATIC_DATA_SEG data_seg("_SDATA", "SCODE")
|
|
#define VxD_DEBUG_ONLY_CODE_SEG code_seg("_DBOCODE", "DBOCODE")
|
|
#define VxD_DEBUG_ONLY_DATA_SEG data_seg("_DBODATA", "DBOCODE")
|
|
|
|
#define VxD_SYSEXIT_CODE_SEG code_seg("SYSEXIT", "SYSEXITCODE")
|
|
#define VxD_INT21_CODE_SEG code_seg("INT21", "INT21CODE")
|
|
#define VxD_RARE_CODE_SEG code_seg("RARE", "RARECODE")
|
|
#define VxD_W16_CODE_SEG code_seg("W16", "W16CODE")
|
|
#define VxD_W32_CODE_SEG code_seg("W32", "W32CODE")
|
|
#define VxD_VMCREATE_CODE_SEG code_seg("VMCREATE", "VMCREATECODE")
|
|
#define VxD_VMDESTROY_CODE_SEG code_seg("VMDESTROY", "VMDESTROYCODE")
|
|
#define VxD_THCREATE_CODE_SEG code_seg("THCREATE", "THCREATECODE")
|
|
#define VxD_THDESTROY_CODE_SEG code_seg("THDESTROY", "THDESTROYCODE")
|
|
#define VxD_VMSUSPEND_CODE_SEG code_seg("VMSUSPEND", "VMSUSPENDCODE")
|
|
#define VxD_VMRESUME_CODE_SEG code_seg("VMRESUME", "VMRESUMECODE")
|
|
#define VxD_PNP_CODE_SEG code_seg("PNP", "PNPCODE")
|
|
#define VxD_DOSVM_CODE_SEG code_seg("DOSVM", "DOSVMCODE")
|
|
#define VxD_LOCKABLE_CODE_SEG code_seg("LOCKABLE", "LOCKABLECODE")
|
|
/* XLATON */
|
|
|
|
/* ASM
|
|
??_CUR_CODE_SEG = 0
|
|
|
|
??_LCODE = 1
|
|
??_ICODE = 2
|
|
??_PCODE = 3
|
|
??_SCODE = 4
|
|
??_DBOCODE = 5
|
|
??_16ICODE = 6
|
|
??_RCODE = 7
|
|
??_LOCKABLECODE = 8
|
|
|
|
?_LCODE equ <(??_CUR_CODE_SEG MOD 16) - ??_LCODE>
|
|
?_ICODE equ <(??_CUR_CODE_SEG MOD 16) - ??_ICODE>
|
|
?_PCODE equ <(??_CUR_CODE_SEG MOD 16) - ??_PCODE>
|
|
?_SCODE equ <(??_CUR_CODE_SEG MOD 16) - ??_SCODE>
|
|
?_DBOCODE equ <(??_CUR_CODE_SEG MOD 16) - ??_DBOCODE>
|
|
?_16ICODE equ <(??_CUR_CODE_SEG MOD 16) - ??_16ICODE>
|
|
?_RCODE equ <(??_CUR_CODE_SEG MOD 16) - ??_RCODE>
|
|
?_LOCKABLECODE equ <(??_CUR_CODE_SEG MOD 16) - ??_LOCKABLECODE>
|
|
|
|
ifndef NO_SEGMENTS
|
|
|
|
;
|
|
; SEGMENT definitions and order
|
|
;
|
|
|
|
IFDEF MASM6
|
|
_FLAT EQU FLAT
|
|
ELSE
|
|
_FLAT EQU USE32
|
|
ENDIF
|
|
|
|
;* 32 bit locked code
|
|
_LTEXT SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_LTEXT ENDS
|
|
|
|
_TEXT SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_TEXT ENDS
|
|
|
|
;* 32 bit pageable code
|
|
_PTEXT SEGMENT DWORD PUBLIC _FLAT 'PCODE'
|
|
_PTEXT ENDS
|
|
|
|
|
|
|
|
MakeCodeSeg MACRO seglist, classname, grpname, iseg
|
|
|
|
IRP segname,<seglist> ;; For each name in the list
|
|
|
|
IFNB <classname>
|
|
segname SEGMENT DWORD PUBLIC _FLAT "&classname&CODE"
|
|
ELSE
|
|
segname SEGMENT DWORD PUBLIC _FLAT "&segname&CODE"
|
|
ENDIF
|
|
|
|
IFB <iseg>
|
|
VxD_&&segname&&_CODE_SEG MACRO
|
|
segname SEGMENT
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + ??_PCODE
|
|
ASSUME cs:FLAT, ds:FLAT, es:FLAT, ss:FLAT
|
|
|
|
ENDM
|
|
ELSE
|
|
VxD_&&segname&&_CODE_SEG MACRO
|
|
segname SEGMENT
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + iseg
|
|
ASSUME cs:FLAT, ds:FLAT, es:FLAT, ss:FLAT
|
|
|
|
ENDM
|
|
ENDIF
|
|
|
|
VxD_&&segname&&_CODE_ENDS MACRO
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHR 4
|
|
segname ENDS
|
|
ENDM
|
|
|
|
segname ENDS
|
|
|
|
IFNDEF BLD_COFF
|
|
IFNB <grpname>
|
|
_&grpname GROUP segname
|
|
ELSE
|
|
_&&segname GROUP segname
|
|
ENDIF
|
|
ENDIF
|
|
|
|
ENDM ;; End for each segment
|
|
|
|
ENDM
|
|
|
|
MakeCodeSeg <LOCKABLE_BEGIN, LOCKABLE, LOCKABLE_END>, \
|
|
LOCKABLE, LOCKABLE, ??_LOCKABLECODE
|
|
MakeCodeSeg INT21
|
|
MakeCodeSeg SYSEXIT
|
|
MakeCodeSeg RARE
|
|
MakeCodeSeg W16
|
|
MakeCodeSeg W32
|
|
MakeCodeSeg VMCREATE
|
|
MakeCodeSeg VMDESTROY
|
|
MakeCodeSeg THCREATE
|
|
MakeCodeSeg THDESTROY
|
|
MakeCodeSeg VMSUSPEND
|
|
MakeCodeSeg VMRESUME
|
|
MakeCodeSeg PNP
|
|
MakeCodeSeg DOSVM
|
|
|
|
|
|
;*** DefLockableCodeBegin - define beginning of lockable code
|
|
;
|
|
; Defines a label with the given name to mark the beginning
|
|
; of the lockable code area for this VxD. In the debug version,
|
|
; also defines a DWORD containing DFS_TEST_BLOCK so that
|
|
; procedures in the lockable code segment defined with
|
|
; BeginProc may call _Debug_Flags_Service with flags appropriate
|
|
; to the code's current state.
|
|
|
|
DefLockableCodeBegin MACRO name, private
|
|
VxD_LOCKABLE_BEGIN_CODE_SEG
|
|
IFB <private>
|
|
PUBLIC name
|
|
ENDIF
|
|
name LABEL NEAR
|
|
VxD_LOCKABLE_BEGIN_CODE_ENDS
|
|
ifndef WIN31COMPAT
|
|
if DEBLEVEL
|
|
VxD_LOCKED_DATA_SEG
|
|
PUBLIC name&_Debug_Flags
|
|
name&_Debug_Flags DD DFS_TEST_BLOCK
|
|
VxD_LOCKED_DATA_ENDS
|
|
??_debug_flags equ <name&_Debug_Flags>
|
|
endif
|
|
endif
|
|
ENDM
|
|
|
|
;*** DefLockableCodeEnd - define end of lockable code
|
|
;
|
|
; Defines a label with the given name to mark the end
|
|
; of the lockable code area for this VxD. By subtracting
|
|
; the offset of the beginning label from the offset of
|
|
; the ending label, the VxD may determine how many bytes
|
|
; of memory to lock or unlock.
|
|
|
|
DefLockableCodeEnd MACRO name, private
|
|
VxD_LOCKABLE_END_CODE_SEG
|
|
IFB <private>
|
|
PUBLIC name
|
|
ENDIF
|
|
name LABEL NEAR
|
|
VxD_LOCKABLE_END_CODE_ENDS
|
|
ENDM
|
|
|
|
;*** CodeLockFlags - declare locked code debug flags
|
|
;
|
|
; This macro declares the locked code debug flags.
|
|
|
|
CodeLockFlags MACRO name
|
|
ifndef WIN31COMPAT
|
|
if DEBLEVEL
|
|
ifndef name&_Debug_Flags
|
|
VxD_LOCKED_DATA_SEG
|
|
extrn name&_Debug_Flags:dword
|
|
VxD_LOCKED_DATA_ENDS
|
|
??_debug_flags equ <name&_Debug_Flags>
|
|
endif
|
|
endif
|
|
endif
|
|
ENDM
|
|
|
|
;*** MarkCodeLocked - signify that lockable code is locked
|
|
;
|
|
; This macro clears DFS_TEST_BLOCK in the debug flags
|
|
; DWORD.
|
|
|
|
MarkCodeLocked MACRO
|
|
ifndef WIN31COMPAT
|
|
if DEBLEVEL
|
|
ifdef ??_debug_flags
|
|
pushfd
|
|
and ??_debug_flags,NOT DFS_TEST_BLOCK
|
|
popfd
|
|
endif
|
|
endif
|
|
endif
|
|
ENDM
|
|
|
|
;*** MarkCodeUnlocked - signify that lockable code is unlocked
|
|
;
|
|
; This macro sets DFS_TEST_BLOCK in the debug flags
|
|
; DWORD.
|
|
|
|
MarkCodeUnlocked MACRO
|
|
ifndef WIN31COMPAT
|
|
if DEBLEVEL
|
|
ifdef ??_debug_flags
|
|
pushfd
|
|
or ??_debug_flags,DFS_TEST_BLOCK
|
|
popfd
|
|
endif
|
|
endif
|
|
endif
|
|
ENDM
|
|
|
|
|
|
;* 32 bit initialization code
|
|
_ITEXT SEGMENT DWORD PUBLIC _FLAT 'ICODE'
|
|
_ITEXT ENDS
|
|
|
|
;* 32 bit locked data
|
|
_LDATA SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_LDATA ENDS
|
|
|
|
_DATA SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_DATA ENDS
|
|
|
|
;* 32 bit pageable data
|
|
_PDATA SEGMENT DWORD PUBLIC _FLAT 'PDATA'
|
|
_PDATA ENDS
|
|
|
|
;* 32 Bit initialization data
|
|
_IDATA SEGMENT DWORD PUBLIC _FLAT 'ICODE'
|
|
_IDATA ENDS
|
|
|
|
;* Created by C8
|
|
_BSS SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_BSS ENDS
|
|
|
|
CONST SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
CONST ENDS
|
|
|
|
_TLS SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_TLS ENDS
|
|
|
|
;* 32 Bit static code for DL-VxDs
|
|
_STEXT SEGMENT DWORD PUBLIC _FLAT 'SCODE'
|
|
_STEXT ENDS
|
|
|
|
;* 32 Bit static data for DL-VxDs
|
|
_SDATA SEGMENT DWORD PUBLIC _FLAT 'SCODE'
|
|
_SDATA ENDS
|
|
|
|
;* dummy segment for IsDebugOnlyLoaded
|
|
_DBOSTART SEGMENT DWORD PUBLIC _FLAT 'DBOCODE'
|
|
_DBOSTART ENDS
|
|
|
|
;* 32 bit debug only code; loaded only if debugger is present
|
|
_DBOCODE SEGMENT DWORD PUBLIC _FLAT 'DBOCODE'
|
|
_DBOCODE ENDS
|
|
|
|
;* 32 bit debug only data; loaded only if debugger is present
|
|
_DBODATA SEGMENT DWORD PUBLIC _FLAT 'DBOCODE'
|
|
_DBODATA ENDS
|
|
|
|
if DEBLEVEL
|
|
;* Start of 32 bit path coverage data
|
|
_PATHSTART SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_PATHSTART ENDS
|
|
|
|
;* 32 bit path coverage data
|
|
_PATHDATA SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_PATHDATA ENDS
|
|
|
|
;* End of 32 bit path coverage data
|
|
_PATHEND SEGMENT DWORD PUBLIC _FLAT 'LCODE'
|
|
_PATHEND ENDS
|
|
endif
|
|
|
|
;* 16 bit code/data that is put into IGROUP automaticly
|
|
_16ICODE SEGMENT WORD USE16 PUBLIC '16ICODE'
|
|
_16ICODE ENDS
|
|
|
|
;* Real Mode initialization code/data for devices
|
|
_RCODE SEGMENT WORD USE16 PUBLIC 'RCODE'
|
|
_RCODE ENDS
|
|
|
|
IFNDEF BLD_COFF
|
|
_LGROUP GROUP _LTEXT, _TEXT, _LDATA, _DATA, _BSS, CONST, _TLS
|
|
_IGROUP GROUP _ITEXT, _IDATA
|
|
_SGROUP GROUP _STEXT, _SDATA
|
|
_DBOGROUP GROUP _DBOSTART, _DBOCODE, _DBODATA
|
|
IF DEBLEVEL
|
|
_PGROUP GROUP _PATHSTART, _PATHDATA, _PATHEND
|
|
ENDIF
|
|
ENDIF
|
|
|
|
endif ; NO_SEGMENTS
|
|
|
|
ASSUME CS:FLAT, DS:FLAT, ES:FLAT, SS:FLAT
|
|
|
|
OFFSET32 EQU <OFFSET FLAT:>
|
|
|
|
|
|
BeginDoc
|
|
;==============================================================================
|
|
; The following macros are used in defining the routines
|
|
; in a VxD which are going to be registered with VMM as callable entry
|
|
; points. Once registered, the entry points can be called by any other
|
|
; devices via the "VxDCall" macro, defined below. In the comments below,
|
|
; replace "VxD" with the appropriate device name.
|
|
;
|
|
;*******
|
|
; In the VxD.INC file, put the following lines, replacing <function_name>
|
|
; with an appropriate name describing the function of the routine.
|
|
;
|
|
; Begin_Service_Table VxD[,<segname>]
|
|
; VxD_Service <function_name>[,<local segname>]
|
|
; VxD_Service <function_name>[,<local segname>]
|
|
; . . .
|
|
; VxD_Service <function_name>[,<local segname>]
|
|
; End_Service_Table VxD[,<segname>]
|
|
;
|
|
; Note that <segname> is an optional argument and, if specified, the
|
|
; table is put in the segment defined by the macro "yyy_Data_Seg",
|
|
; where yyy=segname. Otherwise the segment is defined by the
|
|
; "VxD_Data_Seg" macro, defined below.
|
|
; Note that <local segname> is an optional argument and, if specified,
|
|
; the procedure's segment is defined by the macro "zzz_Code_Seg",
|
|
; where zzz=segname. Otherwise the segment is defined by the
|
|
; "VxD_Code_Seg" macro, defined below.
|
|
;
|
|
;*******
|
|
; One VxD module should have the following in order to define the entry points:
|
|
;Create_VxD_Service_Table = 1 ; Only in module where table is
|
|
; INCLUDE VxD.INC ; Include the table definition
|
|
;
|
|
;*******
|
|
; All modules that want to call the services defined in the table should include
|
|
; VxD.INC, but not define the label "Create_VxD_Service_Table". This
|
|
; will define the service names to be used with the VxDCall macro.
|
|
;
|
|
EndDoc
|
|
|
|
Begin_Service_Table MACRO Device_Name, Def_Segment
|
|
|
|
IFDEF Device_Name&_Name_Based
|
|
IFNDEF @@NextInternalID
|
|
@@NextInternalID = 0
|
|
ENDIF
|
|
@@NextInternalID = (@@NextInternalID + 1)
|
|
Device_Name&_Internal_ID = @@NextInternalID + BASEID_FOR_NAMEBASEDVXD
|
|
DefineVxDName Device_Name, %Device_Name&_Internal_ID
|
|
ENDIF
|
|
|
|
IFB <Def_Segment>
|
|
BST2 Device_Name, VxD
|
|
ELSE
|
|
BST2 Device_Name, Def_Segment
|
|
ENDIF
|
|
ENDM
|
|
|
|
DefineVxDName MACRO Device_Name, InternalID
|
|
@@VxDName&InternalID EQU <___&Device_Name&STable>
|
|
ENDM
|
|
|
|
|
|
BST2 MACRO Device_Name, Def_Segment
|
|
|
|
Num_&Device_Name&_Services = 0
|
|
|
|
IFDEF Create_&Device_Name&_Service_Table
|
|
|
|
|
|
Def_Segment&_LOCKED_DATA_SEG
|
|
|
|
Device_Name&_Service_Table LABEL DWORD
|
|
|
|
Device_Name&_Service MACRO Procedure, Local_Seg, Condition, StdCallBytes, fastcall
|
|
LOCAL $$&Procedure, extrnproc, tableproc
|
|
|
|
extrnproc MACRO
|
|
IFNB <fastcall>
|
|
IFB <StdCallBytes>
|
|
.err ;StdCallBytes required
|
|
ENDIF
|
|
EXTRN @&&Procedure&&@&&StdCallBytes:NEAR
|
|
ELSE
|
|
IFNB <StdCallBytes>
|
|
EXTRN _&&Procedure&&@&&StdCallBytes:NEAR
|
|
ELSE
|
|
EXTRN Procedure:NEAR
|
|
ENDIF
|
|
ENDIF
|
|
ENDM
|
|
|
|
tableproc MACRO
|
|
IFNB <fastcall>
|
|
dd OFFSET32 @&&Procedure&&@&&StdCallBytes
|
|
ELSE
|
|
IFNB <StdCallBytes>
|
|
dd OFFSET32 _&&Procedure&&@&&StdCallBytes
|
|
ELSE
|
|
dd OFFSET32 Procedure
|
|
ENDIF
|
|
ENDIF
|
|
ENDM
|
|
|
|
IFNB <Condition>
|
|
$$&&Procedure MACRO extern
|
|
IFDEF &Condition
|
|
IFNB <extern>
|
|
extrnproc
|
|
ELSE
|
|
tableproc
|
|
ENDIF
|
|
ELSE
|
|
IFB <extern>
|
|
dd 0
|
|
ENDIF
|
|
ENDIF
|
|
ENDM
|
|
ENDIF
|
|
|
|
IFDIFI <Procedure>, <RESERVED>
|
|
PUBLIC _&&Procedure
|
|
IF1
|
|
_&&Procedure LABEL DWORD
|
|
IFNB <fastcall>
|
|
PUBLIC __&&Procedure
|
|
__&&Procedure LABEL DWORD
|
|
ENDIF
|
|
ENDIF
|
|
IFDIFI <Local_Seg>, <LOCAL>
|
|
IFNB <Local_Seg>
|
|
Local_Seg&&_SEG
|
|
ELSE
|
|
Def_Segment&_CODE_SEG
|
|
ENDIF
|
|
IFNB <Condition>
|
|
$$&&Procedure extern
|
|
ELSE
|
|
extrnproc
|
|
ENDIF
|
|
IFNB <Local_Seg>
|
|
Local_Seg&&_ENDS
|
|
ELSE
|
|
Def_Segment&_CODE_ENDS
|
|
ENDIF
|
|
ENDIF
|
|
IFNB <Condition>
|
|
$$&&Procedure
|
|
ELSE
|
|
tableproc
|
|
ENDIF
|
|
|
|
IFDEF Device_Name&_Name_Based
|
|
@@&&Procedure = (Device_Name&_Internal_ID SHL 16) + Num_&Device_Name&_Services
|
|
ELSE
|
|
@@&&Procedure = (Device_Name&_Device_ID SHL 16) + Num_&Device_Name&_Services
|
|
ENDIF
|
|
ELSE
|
|
dd 0
|
|
ENDIF
|
|
Num_&Device_Name&_Services = Num_&Device_Name&_Services + 1
|
|
IFNB <Condition>
|
|
Purge $$&&Procedure
|
|
ENDIF
|
|
Purge extrnproc
|
|
Purge tableproc
|
|
ENDM
|
|
|
|
Device_Name&_StdCall_Service MACRO Procedure, Args, Local_Seg, Condition
|
|
Device_Name&_Service Procedure, Local_Seg, Condition, %Args*4
|
|
??_standardccall&&_Procedure = Args
|
|
ENDM
|
|
|
|
Device_Name&_FastCall_Service MACRO Procedure, Args, Local_Seg, Condition
|
|
Device_Name&_Service Procedure, Local_Seg, Condition, %Args*4, TRUE
|
|
??_fastcall&&_Procedure = Args
|
|
ENDM
|
|
|
|
ELSE
|
|
|
|
; Local_Seg and Condition are placeholders only in this form
|
|
|
|
IFDEF Device_Name&_Name_Based
|
|
|
|
Device_Name&_Service MACRO Procedure, Local_Seg, Condition
|
|
|
|
|
|
IFDIFI <Procedure>, <RESERVED>
|
|
@@&&Procedure = (Device_Name&_Internal_ID SHL 16) + Num_&Device_Name&_Services
|
|
ENDIF
|
|
Num_&Device_Name&_Services = Num_&Device_Name&_Services + 1
|
|
|
|
ENDM
|
|
ELSE
|
|
|
|
Device_Name&_Service MACRO Procedure, Local_Seg, Condition
|
|
|
|
IFDIFI <Procedure>, <RESERVED>
|
|
@@&&Procedure = (Device_Name&_Device_ID SHL 16) + Num_&Device_Name&_Services
|
|
ENDIF
|
|
Num_&Device_Name&_Services = Num_&Device_Name&_Services + 1
|
|
|
|
ENDM
|
|
|
|
ENDIF
|
|
|
|
Device_Name&_StdCall_Service MACRO Procedure, Args, Local_Seg, Condition
|
|
Device_Name&_Service Procedure, Local_Seg, Condition
|
|
??_standardccall_&&Procedure = Args
|
|
ENDM
|
|
|
|
Device_Name&_FastCall_Service MACRO Procedure, Args, Local_Seg, Condition
|
|
Device_Name&_Service Procedure, Local_Seg, Condition
|
|
??_fastcall_&&Procedure = Args
|
|
ENDM
|
|
|
|
ENDIF
|
|
|
|
ENDM
|
|
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
End_Service_Table MACRO Device_Name, Def_Segment
|
|
|
|
PURGE Device_Name&_Service
|
|
|
|
IFDEF Create_&Device_Name&_Service_Table
|
|
|
|
IFB <Def_Segment>
|
|
VxD_LOCKED_DATA_ENDS
|
|
ELSE
|
|
Def_Segment&_LOCKED_DATA_ENDS
|
|
ENDIF
|
|
|
|
ENDIF
|
|
|
|
ENDM
|
|
|
|
GetVxDServiceOrdinal macro reg,service
|
|
mov reg,@@&service
|
|
endm
|
|
|
|
GetVxDServiceAddress macro reg,service
|
|
mov reg,OFFSET32 service
|
|
endm
|
|
|
|
|
|
;*** Begin_Win32_Services - begin defining Win32 Service Table
|
|
;
|
|
; This macro is used to begin the definition of the Win32
|
|
; Service table. It is modelled after, but not identical
|
|
; to, the Begin_Service_Table macro. If the the special
|
|
; symbol Create_Win32_Services is defined to be true, then
|
|
; the actual table is emitted. Otherwise, only the service
|
|
; numbers are defined.
|
|
;
|
|
; ENTRY VxDName - the name of the VxD; it is assumed
|
|
; that a corresponding Device_ID is
|
|
; also defined.
|
|
; EXIT The macro VxDName&_Win32_Sevice is defined; it
|
|
; accepts a service name as its only parameter.
|
|
; This macro is then used to define each service.
|
|
|
|
Begin_Win32_Services MACRO VxDName
|
|
ifndef Create_Win32_Services
|
|
Create_Win32_Services = 0
|
|
endif
|
|
.errb <VxDName>, <VxD name missing>
|
|
??w32svcno = 0
|
|
if Create_Win32_Services
|
|
VxDName&_Win32_Services label dword
|
|
dd csvc&VxDName, 0
|
|
endif
|
|
??inw32svc = 1
|
|
|
|
VxDName&_Win32_Service MACRO Name
|
|
.erre ??inw32svc, <Missing Begin_Win32_Services>
|
|
if Create_Win32_Services
|
|
dd OFFSET32 Name,cparm&&Name
|
|
endif
|
|
@32&&Name equ ((VxDName&_Device_ID SHL 16) + ??w32svcno)
|
|
??w32svcno = ??w32svcno + 1
|
|
ENDM
|
|
ENDM
|
|
|
|
|
|
;*** End_Win32_Services - mark end of Win32 Service Table
|
|
;
|
|
; This macro completes initialization of the Win32
|
|
; Service table.
|
|
;
|
|
; ENTRY VxDName - the same name passed to
|
|
; Begin_Win32_services
|
|
|
|
End_Win32_Services MACRO VxDName
|
|
.errb <VxDName>, <VxD name misssing>
|
|
if Create_Win32_Services
|
|
csvc&VxDName equ ($ - VxDName&_Win32_Services)/8 - 1
|
|
endif
|
|
??inw32svc = 0
|
|
PURGE VxDName&_Win32_Service
|
|
ENDM
|
|
|
|
|
|
;*** Declare_Win32_Service - declare an external Win32 Service
|
|
;
|
|
; This macro is used to declare a Win32 service that
|
|
; is defined elsewhere, perhaps in a C module.
|
|
;
|
|
; ENTRY Name - the service name
|
|
; cParms - the number of DWORD parameters
|
|
; EXIT The name is defined as external
|
|
|
|
Declare_Win32_Service MACRO Name, cParms
|
|
ifndef Create_Win32_Services
|
|
Create_Win32_Services = 0
|
|
endif
|
|
if Create_Win32_Services
|
|
?merge <Name>,,,,<EQU>,<_>,<Name>,<@>,%(cParms*4 + 8)
|
|
?merge <cparm>,<Name>,,,<EQU>,<cParms>
|
|
VxD_CODE_SEG
|
|
?merge <EXTRN>,,,,,<_>,<Name>,<@>,%(cParms*4 + 8),<:NEAR>
|
|
VxD_CODE_ENDS
|
|
endif
|
|
ENDM
|
|
|
|
|
|
;*** Win32call - call a Win32 service from a ring 3 thunk
|
|
;
|
|
; This macro is used to call a Win32 service from
|
|
; a ring 3 thunk. Note that control will not return
|
|
; to the instruction following the call, but to the
|
|
; instruction following the call to the thunk.
|
|
;
|
|
; ENTRY Service - the name of the service
|
|
; CallBack - the fword containing the callback
|
|
|
|
Win32call MACRO Service, CallBack
|
|
ifndef Create_Win32_Services
|
|
Create_Win32_Services = 0
|
|
endif
|
|
ife Create_Win32_Services
|
|
mov eax,@32&Service
|
|
ifdef IS_16
|
|
movzx esp,sp
|
|
endif
|
|
call fword ptr [CallBack]
|
|
ifdef DEBUG
|
|
int 3
|
|
endif
|
|
endif
|
|
ENDM
|
|
*/
|
|
|
|
/*XLATOFF*/
|
|
#define GetVxDServiceAddress(service) service
|
|
|
|
#define VxDCall(service) \
|
|
_asm _emit 0xcd \
|
|
_asm _emit 0x20 \
|
|
_asm _emit (GetVxDServiceOrdinal(service) & 0xff) \
|
|
_asm _emit (GetVxDServiceOrdinal(service) >> 8) & 0xff \
|
|
_asm _emit (GetVxDServiceOrdinal(service) >> 16) & 0xff \
|
|
_asm _emit (GetVxDServiceOrdinal(service) >> 24) & 0xff \
|
|
|
|
#define VMMCall VxDCall
|
|
|
|
#define VxDJmp(service) \
|
|
_asm _emit 0xcd \
|
|
_asm _emit 0x20 \
|
|
_asm _emit (GetVxDServiceOrdinal(service) & 0xff) \
|
|
_asm _emit ((GetVxDServiceOrdinal(service) >> 8) & 0xff) | 0x80 \
|
|
_asm _emit (GetVxDServiceOrdinal(service) >> 16) & 0xff \
|
|
_asm _emit (GetVxDServiceOrdinal(service) >> 24) & 0xff \
|
|
|
|
#define VMMJmp VxDJmp
|
|
|
|
#define SERVICE __cdecl
|
|
#define ASYNC_SERVICE __cdecl
|
|
#define WIN32_SERVICE void __stdcall
|
|
|
|
#ifndef FASTCALL
|
|
#define FASTCALL __fastcall
|
|
#endif
|
|
/*XLATON*/
|
|
|
|
/* ASM
|
|
;******************************************************************************
|
|
;
|
|
; Dword_Align -- Aligns code to dword boundry by inserting nops
|
|
;
|
|
;------------------------------------------------------------------------------
|
|
|
|
Dword_Align MACRO Seg_Name
|
|
LOCAL segn
|
|
IFDEF MASM6
|
|
align 4
|
|
ELSE
|
|
IFNB <Seg_Name>
|
|
segn equ Seg_Name
|
|
ELSE
|
|
IFE ?_LCODE
|
|
segn equ <_LTEXT>
|
|
ELSE
|
|
IFE ?_ICODE
|
|
segn equ <_ITEXT>
|
|
ELSE
|
|
IFE ?_PCODE
|
|
segn equ <_PTEXT>
|
|
ELSE
|
|
IFE ?_SCODE
|
|
segn equ <_STEXT>
|
|
ELSE
|
|
.err <Dword_Align not supported>
|
|
ENDIF
|
|
ENDIF
|
|
ENDIF
|
|
ENDIF
|
|
ENDIF
|
|
IF (($-OFFSET segn:0) MOD 4)
|
|
db 4 - (($-OFFSET segn:0) MOD 4) DUP (90h)
|
|
ENDIF
|
|
ENDIF
|
|
ENDM
|
|
|
|
|
|
BeginDoc
|
|
;******************************************************************************
|
|
;
|
|
; Fatal_Error
|
|
;
|
|
; DESCRIPTION:
|
|
; This macro is used to crash Windows/386 when an unrecoverable error
|
|
; is detected. If Msg_Ptr is ommitted then no error message will be
|
|
; displayed, otherwise Msg_Ptr is the address
|
|
; when the
|
|
;
|
|
; PARAMETERS:
|
|
; Msg_Ptr (OPTIONAL) - Points to an ASCIIZ string to display.
|
|
;
|
|
; EXIT:
|
|
; To DOS (hopefully). This macro never returns.
|
|
;
|
|
;==============================================================================
|
|
EndDoc
|
|
|
|
Fatal_Error MACRO Msg_Ptr, Exit_Flags
|
|
pushad
|
|
IFB <Msg_Ptr>
|
|
xor esi, esi
|
|
ELSE
|
|
mov esi, Msg_Ptr
|
|
IFB <Exit_Flags>
|
|
xor eax, eax
|
|
ELSE
|
|
mov eax, Exit_Flags
|
|
ENDIF
|
|
ENDIF
|
|
VMMCall Fatal_Error_Handler
|
|
ENDM
|
|
|
|
EF_Hang_On_Exit EQU 1h
|
|
*/
|
|
|
|
|
|
/******************************************************************************
|
|
*
|
|
* The following are control block headers and flags of interest to VxDs.
|
|
*
|
|
*****************************************************************************/
|
|
|
|
struct cb_s {
|
|
ULONG CB_VM_Status; /* VM status flags */
|
|
ULONG CB_High_Linear; /* Address of VM mapped high */
|
|
ULONG CB_Client_Pointer;
|
|
ULONG CB_VMID;
|
|
ULONG CB_Signature;
|
|
};
|
|
|
|
#define VMCB_ID 0x62634D56 /* VMcb */
|
|
|
|
/*
|
|
* VM status indicates globally interesting VM states
|
|
*/
|
|
|
|
#define VMSTAT_EXCLUSIVE_BIT 0x00 /* VM is exclusive mode */
|
|
#define VMSTAT_EXCLUSIVE (1L << VMSTAT_EXCLUSIVE_BIT)
|
|
#define VMSTAT_BACKGROUND_BIT 0x01 /* VM runs in background */
|
|
#define VMSTAT_BACKGROUND (1L << VMSTAT_BACKGROUND_BIT)
|
|
#define VMSTAT_CREATING_BIT 0x02 /* In process of creating */
|
|
#define VMSTAT_CREATING (1L << VMSTAT_CREATING_BIT)
|
|
#define VMSTAT_SUSPENDED_BIT 0x03 /* VM not scheduled */
|
|
#define VMSTAT_SUSPENDED (1L << VMSTAT_SUSPENDED_BIT)
|
|
#define VMSTAT_NOT_EXECUTEABLE_BIT 0x04 /* VM partially destroyed */
|
|
#define VMSTAT_NOT_EXECUTEABLE (1L << VMSTAT_NOT_EXECUTEABLE_BIT)
|
|
#define VMSTAT_PM_EXEC_BIT 0x05 /* Currently in PM app */
|
|
#define VMSTAT_PM_EXEC (1L << VMSTAT_PM_EXEC_BIT)
|
|
#define VMSTAT_PM_APP_BIT 0x06 /* PM app present in VM */
|
|
#define VMSTAT_PM_APP (1L << VMSTAT_PM_APP_BIT)
|
|
#define VMSTAT_PM_USE32_BIT 0x07 /* PM app is 32-bit */
|
|
#define VMSTAT_PM_USE32 (1L << VMSTAT_PM_USE32_BIT)
|
|
#define VMSTAT_VXD_EXEC_BIT 0x08 /* Call from VxD */
|
|
#define VMSTAT_VXD_EXEC (1L << VMSTAT_VXD_EXEC_BIT)
|
|
#define VMSTAT_HIGH_PRI_BACK_BIT 0x09 /* High pri background */
|
|
#define VMSTAT_HIGH_PRI_BACK (1L << VMSTAT_HIGH_PRI_BACK_BIT)
|
|
#define VMSTAT_BLOCKED_BIT 0x0A /* Blocked on semaphore */
|
|
#define VMSTAT_BLOCKED (1L << VMSTAT_BLOCKED_BIT)
|
|
#define VMSTAT_AWAKENING_BIT 0x0B /* Woke up after blocked */
|
|
#define VMSTAT_AWAKENING (1L << VMSTAT_AWAKENING_BIT)
|
|
#define VMSTAT_PAGEABLEV86BIT 0x0C /* part of V86 is pageable (PM app) */
|
|
#define VMSTAT_PAGEABLEV86_BIT VMSTAT_PAGEABLEV86BIT
|
|
#define VMSTAT_PAGEABLEV86 (1L << VMSTAT_PAGEABLEV86BIT)
|
|
#define VMSTAT_V86INTSLOCKEDBIT 0x0D /* Locked regardless of pager type */
|
|
#define VMSTAT_V86INTSLOCKED_BIT VMSTAT_V86INTSLOCKEDBIT
|
|
#define VMSTAT_V86INTSLOCKED (1L << VMSTAT_V86INTSLOCKEDBIT)
|
|
#define VMSTAT_IDLE_TIMEOUT_BIT 0x0E /* Scheduled by time-slicer */
|
|
#define VMSTAT_IDLE_TIMEOUT (1L << VMSTAT_IDLE_TIMEOUT_BIT)
|
|
#define VMSTAT_IDLE_BIT 0x0F /* VM has released time slice */
|
|
#define VMSTAT_IDLE (1L << VMSTAT_IDLE_BIT)
|
|
#define VMSTAT_CLOSING_BIT 0x10 /* Close_VM called for VM */
|
|
#define VMSTAT_CLOSING (1L << VMSTAT_CLOSING_BIT)
|
|
#define VMSTAT_TS_SUSPENDED_BIT 0x11 /* VM suspended by */
|
|
#define VMSTAT_TS_SUSPENDED (1L << VMSTAT_TS_SUSPENDED_BIT)
|
|
#define VMSTAT_TS_MAXPRI_BIT 0x12 /* this is fgd_pri 10,000 internally*/
|
|
#define VMSTAT_TS_MAXPRI (1L << VMSTAT_TS_MAXPRI_BIT)
|
|
|
|
#define VMSTAT_USE32_MASK (VMSTAT_PM_USE32 | VMSTAT_VXD_EXEC)
|
|
|
|
struct tcb_s {
|
|
ULONG TCB_Flags; /* Thread status flags */
|
|
ULONG TCB_Reserved1; /* Used internally by VMM */
|
|
ULONG TCB_Reserved2; /* Used internally by VMM */
|
|
ULONG TCB_Signature;
|
|
ULONG TCB_ClientPtr; /* Client registers of thread */
|
|
ULONG TCB_VMHandle; /* VM that thread is part of */
|
|
USHORT TCB_ThreadId; /* Unique Thread ID */
|
|
USHORT TCB_PMLockOrigSS; /* Original SS:ESP before lock stack */
|
|
ULONG TCB_PMLockOrigESP;
|
|
ULONG TCB_PMLockOrigEIP; /* Original CS:EIP before lock stack */
|
|
ULONG TCB_PMLockStackCount;
|
|
USHORT TCB_PMLockOrigCS;
|
|
USHORT TCB_PMPSPSelector;
|
|
ULONG TCB_ThreadType; /* dword passed to VMMCreateThread */
|
|
USHORT TCB_pad1; /* reusable; for dword align */
|
|
UCHAR TCB_pad2; /* reusable; for dword align */
|
|
UCHAR TCB_extErrLocus; /* extended error Locus */
|
|
USHORT TCB_extErr; /* extended error Code */
|
|
UCHAR TCB_extErrAction; /* " " Action */
|
|
UCHAR TCB_extErrClass; /* " " Class */
|
|
ULONG TCB_extErrPtr; /* " pointer */
|
|
|
|
};
|
|
|
|
typedef struct tcb_s TCB;
|
|
typedef TCB *PTCB;
|
|
|
|
#define SCHED_OBJ_ID_THREAD 0x42434854 // THCB in ASCII
|
|
|
|
/*
|
|
* Thread status indicates globally interesting thread states.
|
|
* Flags are for information only and must not be modified.
|
|
*/
|
|
|
|
#define THFLAG_SUSPENDED_BIT 0x03 // Thread not scheduled
|
|
#define THFLAG_SUSPENDED (1L << THFLAG_SUSPENDED_BIT)
|
|
#define THFLAG_NOT_EXECUTEABLE_BIT 0x04 // Thread partially destroyed
|
|
#define THFLAG_NOT_EXECUTEABLE (1L << THFLAG_NOT_EXECUTEABLE_BIT)
|
|
#define THFLAG_THREAD_CREATION_BIT 0x08 // Thread in status nascendi
|
|
#define THFLAG_THREAD_CREATION (1L << THFLAG_THREAD_CREATION_BIT)
|
|
#define THFLAG_THREAD_BLOCKED_BIT 0x0A // Blocked on semaphore
|
|
#define THFLAG_THREAD_BLOCKED (1L << THFLAG_THREAD_BLOCKED_BIT)
|
|
#define THFLAG_RING0_THREAD_BIT 0x1C // thread runs only at ring 0
|
|
#define THFLAG_RING0_THREAD (1L << THFLAG_RING0_THREAD_BIT)
|
|
#define THFLAG_ASYNC_THREAD_BIT 0x1F // thread is asynchronous
|
|
#define THFLAG_ASYNC_THREAD (1L << THFLAG_ASYNC_THREAD_BIT)
|
|
#define THFLAG_CHARSET_BITS 0x10 // Default character set
|
|
#define THFLAG_CHARSET_MASK (3L << THFLAG_CHARSET_BITS)
|
|
#define THFLAG_ANSI (0L << THFLAG_CHARSET_BITS)
|
|
#define THFLAG_OEM (1L << THFLAG_CHARSET_BITS)
|
|
#define THFLAG_UNICODE (2L << THFLAG_CHARSET_BITS)
|
|
#define THFLAG_RESERVED (3L << THFLAG_CHARSET_BITS)
|
|
#define THFLAG_EXTENDED_HANDLES_BIT 0x12 // Thread uses extended file handles
|
|
#define THFLAG_EXTENDED_HANDLES (1L << THFLAG_EXTENDED_HANDLES_BIT)
|
|
/* the win32 loader opens win32 exes with this bit set to notify IFS
|
|
* so a defragger won't move these files
|
|
* the bit is turned off once the open completes.
|
|
* file open flags are overloaded which is why this is here
|
|
*/
|
|
#define THFLAG_OPEN_AS_IMMOVABLE_FILE_BIT 0x13 // File thus opened not moved
|
|
#define THFLAG_OPEN_AS_IMMOVABLE_FILE (1L << THFLAG_OPEN_AS_IMMOVABLE_FILE_BIT)
|
|
|
|
/*
|
|
* Protected mode application control blocks
|
|
*/
|
|
struct pmcb_s {
|
|
ULONG PMCB_Flags;
|
|
ULONG PMCB_Parent;
|
|
};
|
|
|
|
/*
|
|
* The reference data for fault error codes 1-5 (GSDVME_PRIVINST through
|
|
* GSDVME_INVALFLT) is a pointer to the following fault information structure.
|
|
*/
|
|
struct VMFaultInfo {
|
|
ULONG VMFI_EIP; // faulting EIP
|
|
WORD VMFI_CS; // faulting CS
|
|
WORD VMFI_Ints; // interrupts in service, if any
|
|
};
|
|
|
|
typedef struct VMFaultInfo *PVMFaultInfo;
|
|
|
|
/******************************************************************************
|
|
* V M M S E R V I C E S
|
|
******************************************************************************/
|
|
|
|
/*XLATOFF*/
|
|
#define VMM_Service Declare_Service
|
|
#define VMM_StdCall_Service Declare_SCService
|
|
#define VMM_FastCall_Service Declare_SCService
|
|
#pragma warning (disable:4003) // turn off not enough params warning
|
|
/*XLATON*/
|
|
|
|
/*MACROS*/
|
|
Begin_Service_Table(VMM, VMM)
|
|
|
|
VMM_Service (Get_VMM_Version, LOCAL) // MUST REMAIN SERVICE 0!
|
|
|
|
VMM_Service (Get_Cur_VM_Handle)
|
|
VMM_Service (Test_Cur_VM_Handle)
|
|
VMM_Service (Get_Sys_VM_Handle)
|
|
VMM_Service (Test_Sys_VM_Handle)
|
|
VMM_Service (Validate_VM_Handle)
|
|
|
|
VMM_Service (Get_VMM_Reenter_Count)
|
|
VMM_Service (Begin_Reentrant_Execution)
|
|
VMM_Service (End_Reentrant_Execution)
|
|
|
|
VMM_Service (Install_V86_Break_Point)
|
|
VMM_Service (Remove_V86_Break_Point)
|
|
VMM_Service (Allocate_V86_Call_Back)
|
|
VMM_Service (Allocate_PM_Call_Back)
|
|
|
|
VMM_Service (Call_When_VM_Returns)
|
|
|
|
VMM_Service (Schedule_Global_Event)
|
|
VMM_Service (Schedule_VM_Event)
|
|
VMM_Service (Call_Global_Event)
|
|
VMM_Service (Call_VM_Event)
|
|
VMM_Service (Cancel_Global_Event)
|
|
VMM_Service (Cancel_VM_Event)
|
|
VMM_Service (Call_Priority_VM_Event)
|
|
VMM_Service (Cancel_Priority_VM_Event)
|
|
|
|
VMM_Service (Get_NMI_Handler_Addr)
|
|
VMM_Service (Set_NMI_Handler_Addr)
|
|
VMM_Service (Hook_NMI_Event)
|
|
|
|
VMM_Service (Call_When_VM_Ints_Enabled)
|
|
VMM_Service (Enable_VM_Ints)
|
|
VMM_Service (Disable_VM_Ints)
|
|
|
|
VMM_Service (Map_Flat)
|
|
VMM_Service (Map_Lin_To_VM_Addr)
|
|
|
|
// Scheduler services
|
|
|
|
VMM_Service (Adjust_Exec_Priority)
|
|
VMM_Service (Begin_Critical_Section)
|
|
VMM_Service (End_Critical_Section)
|
|
VMM_Service (End_Crit_And_Suspend)
|
|
VMM_Service (Claim_Critical_Section)
|
|
VMM_Service (Release_Critical_Section)
|
|
VMM_Service (Call_When_Not_Critical)
|
|
VMM_Service (Create_Semaphore)
|
|
VMM_Service (Destroy_Semaphore)
|
|
VMM_Service (Wait_Semaphore)
|
|
VMM_Service (Signal_Semaphore)
|
|
VMM_Service (Get_Crit_Section_Status)
|
|
VMM_Service (Call_When_Task_Switched)
|
|
VMM_Service (Suspend_VM)
|
|
VMM_Service (Resume_VM)
|
|
VMM_Service (No_Fail_Resume_VM)
|
|
VMM_Service (Nuke_VM)
|
|
VMM_Service (Crash_Cur_VM)
|
|
|
|
VMM_Service (Get_Execution_Focus)
|
|
VMM_Service (Set_Execution_Focus)
|
|
VMM_Service (Get_Time_Slice_Priority)
|
|
VMM_Service (Set_Time_Slice_Priority)
|
|
VMM_Service (Get_Time_Slice_Granularity)
|
|
VMM_Service (Set_Time_Slice_Granularity)
|
|
VMM_Service (Get_Time_Slice_Info)
|
|
VMM_Service (Adjust_Execution_Time)
|
|
VMM_Service (Release_Time_Slice)
|
|
VMM_Service (Wake_Up_VM)
|
|
VMM_Service (Call_When_Idle)
|
|
|
|
VMM_Service (Get_Next_VM_Handle)
|
|
|
|
// Time-out and system timer services
|
|
|
|
VMM_Service (Set_Global_Time_Out)
|
|
VMM_Service (Set_VM_Time_Out)
|
|
VMM_Service (Cancel_Time_Out)
|
|
VMM_Service (Get_System_Time)
|
|
VMM_Service (Get_VM_Exec_Time)
|
|
|
|
VMM_Service (Hook_V86_Int_Chain)
|
|
VMM_Service (Get_V86_Int_Vector)
|
|
VMM_Service (Set_V86_Int_Vector)
|
|
VMM_Service (Get_PM_Int_Vector)
|
|
VMM_Service (Set_PM_Int_Vector)
|
|
|
|
VMM_Service (Simulate_Int)
|
|
VMM_Service (Simulate_Iret)
|
|
VMM_Service (Simulate_Far_Call)
|
|
VMM_Service (Simulate_Far_Jmp)
|
|
VMM_Service (Simulate_Far_Ret)
|
|
VMM_Service (Simulate_Far_Ret_N)
|
|
VMM_Service (Build_Int_Stack_Frame)
|
|
|
|
VMM_Service (Simulate_Push)
|
|
VMM_Service (Simulate_Pop)
|
|
|
|
// Heap Manager
|
|
|
|
VMM_Service (_HeapAllocate)
|
|
VMM_Service (_HeapReAllocate)
|
|
VMM_Service (_HeapFree)
|
|
VMM_Service (_HeapGetSize)
|
|
|
|
/*ENDMACROS*/
|
|
|
|
/****************************************************
|
|
*
|
|
* Flags for heap allocator calls
|
|
*
|
|
* NOTE: HIGH 8 BITS (bits 24-31) are reserved
|
|
*
|
|
***************************************************/
|
|
|
|
//
|
|
// Flags affecting the returned block
|
|
//
|
|
|
|
#define HEAPZEROINIT 0x00000001
|
|
#define HEAPZEROREINIT 0x00000002
|
|
#define HEAPNOCOPY 0x00000004
|
|
|
|
//
|
|
// Alignment flags
|
|
//
|
|
|
|
#define HEAPALIGN_SHIFT 16
|
|
#define HEAPALIGN_MASK 0x000F0000
|
|
|
|
#define HEAPALIGN_4 0x00000000 // dword aligned
|
|
#define HEAPALIGN_8 0x00000000 // quadword aligned
|
|
#define HEAPALIGN_16 0x00000000 // paragraph aligned
|
|
#define HEAPALIGN_32 0x00010000 // etc.
|
|
#define HEAPALIGN_64 0x00020000
|
|
#define HEAPALIGN_128 0x00030000
|
|
#define HEAPALIGN_256 0x00040000
|
|
#define HEAPALIGN_512 0x00050000
|
|
#define HEAPALIGN_1K 0x00060000
|
|
#define HEAPALIGN_2K 0x00070000
|
|
#define HEAPALIGN_4K 0x00080000
|
|
#define HEAPALIGN_8K 0x00090000
|
|
#define HEAPALIGN_16K 0x000A0000
|
|
#define HEAPALIGN_32K 0x000B0000
|
|
#define HEAPALIGN_64K 0x000C0000
|
|
#define HEAPALIGN_128K 0x000D0000
|
|
|
|
//
|
|
// Flags indicating which system heap to use. There are four bits reserved
|
|
// to identify the heap to use. Four are currently defined by the system.
|
|
//
|
|
|
|
#define HEAPTYPESHIFT 8
|
|
#define HEAPTYPEMASK 0x00000700
|
|
|
|
#define HEAPLOCKEDHIGH 0x00000000
|
|
#define HEAPLOCKEDIFDP 0x00000100
|
|
#define HEAPSWAP 0x00000200
|
|
#define HEAPLOCKEDLOW 0x00000300
|
|
#define HEAPINIT 0x00000400 // will be automatically freed after
|
|
// init complete
|
|
#define HEAPSYSVM 0x00000500
|
|
|
|
//
|
|
// other flags
|
|
//
|
|
|
|
#define HEAPCLEAN 0x00000800
|
|
#define HEAPCONTIG 0x00001000 // memory must be physically contiguous
|
|
#define HEAPFORGET 0x00002000 // this memory will never be freed
|
|
|
|
// Page Manager
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_PageAllocate)
|
|
VMM_Service (_PageReAllocate)
|
|
VMM_Service (_PageFree)
|
|
VMM_Service (_PageLock)
|
|
VMM_Service (_PageUnLock)
|
|
VMM_Service (_PageGetSizeAddr)
|
|
VMM_Service (_PageGetAllocInfo)
|
|
VMM_Service (_GetFreePageCount)
|
|
VMM_Service (_GetSysPageCount)
|
|
VMM_Service (_GetVMPgCount)
|
|
VMM_Service (_MapIntoV86)
|
|
VMM_Service (_PhysIntoV86)
|
|
VMM_Service (_TestGlobalV86Mem)
|
|
VMM_Service (_ModifyPageBits)
|
|
VMM_Service (_CopyPageTable)
|
|
VMM_Service (_LinMapIntoV86)
|
|
VMM_Service (_LinPageLock)
|
|
VMM_Service (_LinPageUnLock)
|
|
VMM_Service (_SetResetV86Pageable)
|
|
VMM_Service (_GetV86PageableArray)
|
|
VMM_Service (_PageCheckLinRange)
|
|
VMM_Service (_PageOutDirtyPages)
|
|
VMM_Service (_PageDiscardPages)
|
|
/*ENDMACROS*/
|
|
|
|
/****************************************************
|
|
*
|
|
* Flags for other page allocator calls
|
|
*
|
|
* NOTE: HIGH 8 BITS (bits 24-31) are reserved
|
|
*
|
|
***************************************************/
|
|
|
|
#define PAGEZEROINIT 0x00000001
|
|
#define PAGEUSEALIGN 0x00000002
|
|
#define PAGECONTIG 0x00000004
|
|
#define PAGEFIXED 0x00000008
|
|
#define PAGEDEBUGNULFAULT 0x00000010
|
|
#define PAGEZEROREINIT 0x00000020
|
|
#define PAGENOCOPY 0x00000040
|
|
#define PAGELOCKED 0x00000080
|
|
#define PAGELOCKEDIFDP 0x00000100
|
|
#define PAGESETV86PAGEABLE 0x00000200
|
|
#define PAGECLEARV86PAGEABLE 0x00000400
|
|
#define PAGESETV86INTSLOCKED 0x00000800
|
|
#define PAGECLEARV86INTSLOCKED 0x00001000
|
|
#define PAGEMARKPAGEOUT 0x00002000
|
|
#define PAGEPDPSETBASE 0x00004000
|
|
#define PAGEPDPCLEARBASE 0x00008000
|
|
#define PAGEDISCARD 0x00010000
|
|
#define PAGEPDPQUERYDIRTY 0x00020000
|
|
#define PAGEMAPFREEPHYSREG 0x00040000
|
|
#define PAGEPHYSONLY 0x04000000
|
|
//efine PAGEDONTUSE 0x08000000 // ;Internal
|
|
#define PAGENOMOVE 0x10000000
|
|
#define PAGEMAPGLOBAL 0x40000000
|
|
#define PAGEMARKDIRTY 0x80000000
|
|
|
|
/****************************************************
|
|
*
|
|
* Flags for _PhysIntoV86,
|
|
* _MapIntoV86, and _LinMapIntoV86
|
|
*
|
|
***************************************************/
|
|
|
|
#define MAPV86_IGNOREWRAP 0x00000001
|
|
|
|
|
|
// Informational services
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_GetNulPageHandle)
|
|
VMM_Service (_GetFirstV86Page)
|
|
VMM_Service (_MapPhysToLinear)
|
|
VMM_Service (_GetAppFlatDSAlias)
|
|
VMM_Service (_SelectorMapFlat)
|
|
VMM_Service (_GetDemandPageInfo)
|
|
VMM_Service (_GetSetPageOutCount)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flags bits for _GetSetPageOutCount
|
|
*/
|
|
#define GSPOC_F_GET 0x00000001
|
|
|
|
// Device VM page manager
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Hook_V86_Page)
|
|
VMM_Service (_Assign_Device_V86_Pages)
|
|
VMM_Service (_DeAssign_Device_V86_Pages)
|
|
VMM_Service (_Get_Device_V86_Pages_Array)
|
|
VMM_Service (MMGR_SetNULPageAddr)
|
|
|
|
// GDT/LDT management
|
|
|
|
VMM_Service (_Allocate_GDT_Selector)
|
|
VMM_Service (_Free_GDT_Selector)
|
|
VMM_Service (_Allocate_LDT_Selector)
|
|
VMM_Service (_Free_LDT_Selector)
|
|
VMM_Service (_BuildDescriptorDWORDs)
|
|
VMM_Service (_GetDescriptor)
|
|
VMM_Service (_SetDescriptor)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flag equates for _BuildDescriptorDWORDs
|
|
*/
|
|
#define BDDEXPLICITDPL 0x00000001
|
|
|
|
/*
|
|
* Flag equates for _Allocate_LDT_Selector
|
|
*/
|
|
#define ALDTSPECSEL 0x00000001
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_MMGR_Toggle_HMA)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flag equates for _MMGR_Toggle_HMA
|
|
*/
|
|
#define MMGRHMAPHYSICAL 0x00000001
|
|
#define MMGRHMAENABLE 0x00000002
|
|
#define MMGRHMADISABLE 0x00000004
|
|
#define MMGRHMAQUERY 0x00000008
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Get_Fault_Hook_Addrs)
|
|
VMM_Service (Hook_V86_Fault)
|
|
VMM_Service (Hook_PM_Fault)
|
|
VMM_Service (Hook_VMM_Fault)
|
|
VMM_Service (Begin_Nest_V86_Exec)
|
|
VMM_Service (Begin_Nest_Exec)
|
|
VMM_Service (Exec_Int)
|
|
VMM_Service (Resume_Exec)
|
|
VMM_Service (End_Nest_Exec)
|
|
|
|
VMM_Service (Allocate_PM_App_CB_Area, VMM_ICODE)
|
|
VMM_Service (Get_Cur_PM_App_CB)
|
|
|
|
VMM_Service (Set_V86_Exec_Mode)
|
|
VMM_Service (Set_PM_Exec_Mode)
|
|
|
|
VMM_Service (Begin_Use_Locked_PM_Stack)
|
|
VMM_Service (End_Use_Locked_PM_Stack)
|
|
|
|
VMM_Service (Save_Client_State)
|
|
VMM_Service (Restore_Client_State)
|
|
|
|
VMM_Service (Exec_VxD_Int)
|
|
|
|
VMM_Service (Hook_Device_Service)
|
|
|
|
VMM_Service (Hook_Device_V86_API)
|
|
VMM_Service (Hook_Device_PM_API)
|
|
|
|
VMM_Service (System_Control)
|
|
|
|
// I/O and software interrupt hooks
|
|
|
|
VMM_Service (Simulate_IO)
|
|
VMM_Service (Install_Mult_IO_Handlers)
|
|
VMM_Service (Install_IO_Handler)
|
|
VMM_Service (Enable_Global_Trapping)
|
|
VMM_Service (Enable_Local_Trapping)
|
|
VMM_Service (Disable_Global_Trapping)
|
|
VMM_Service (Disable_Local_Trapping)
|
|
|
|
// Linked List Abstract Data Type Services
|
|
|
|
VMM_Service (List_Create)
|
|
VMM_Service (List_Destroy)
|
|
VMM_Service (List_Allocate)
|
|
VMM_Service (List_Attach)
|
|
VMM_Service (List_Attach_Tail)
|
|
VMM_Service (List_Insert)
|
|
VMM_Service (List_Remove)
|
|
VMM_Service (List_Deallocate)
|
|
VMM_Service (List_Get_First)
|
|
VMM_Service (List_Get_Next)
|
|
VMM_Service (List_Remove_First)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flags used by List_Create
|
|
*/
|
|
#define LF_ASYNC_BIT 0
|
|
#define LF_ASYNC (1 << LF_ASYNC_BIT)
|
|
#define LF_USE_HEAP_BIT 1
|
|
#define LF_USE_HEAP (1 << LF_USE_HEAP_BIT)
|
|
#define LF_ALLOC_ERROR_BIT 2
|
|
#define LF_ALLOC_ERROR (1 << LF_ALLOC_ERROR_BIT)
|
|
/*
|
|
* Swappable lists must use the heap.
|
|
*/
|
|
#define LF_SWAP (LF_USE_HEAP + (1 << 3))
|
|
|
|
/******************************************************************************
|
|
* I N I T I A L I Z A T I O N P R O C E D U R E S
|
|
******************************************************************************/
|
|
|
|
// Instance data manager
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_AddInstanceItem)
|
|
|
|
// System structure data manager
|
|
|
|
VMM_Service (_Allocate_Device_CB_Area)
|
|
VMM_Service (_Allocate_Global_V86_Data_Area, VMM_ICODE)
|
|
VMM_Service (_Allocate_Temp_V86_Data_Area, VMM_ICODE)
|
|
VMM_Service (_Free_Temp_V86_Data_Area, VMM_ICODE)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flag bits for _Allocate_Global_V86_Data_Area
|
|
*/
|
|
#define GVDAWordAlign 0x00000001
|
|
#define GVDADWordAlign 0x00000002
|
|
#define GVDAParaAlign 0x00000004
|
|
#define GVDAPageAlign 0x00000008
|
|
#define GVDAInstance 0x00000100
|
|
#define GVDAZeroInit 0x00000200
|
|
#define GVDAReclaim 0x00000400
|
|
#define GVDAInquire 0x00000800
|
|
#define GVDAHighSysCritOK 0x00001000
|
|
#define GVDAOptInstance 0x00002000
|
|
#define GVDAForceLow 0x00004000
|
|
|
|
/*
|
|
* Flag bits for _Allocate_Temp_V86_Data_Area
|
|
*/
|
|
#define TVDANeedTilInitComplete 0x00000001
|
|
|
|
// Initialization information calls (win.ini and environment parameters)
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Get_Profile_Decimal_Int, VMM_ICODE)
|
|
VMM_Service (Convert_Decimal_String, VMM_ICODE)
|
|
VMM_Service (Get_Profile_Fixed_Point, VMM_ICODE)
|
|
VMM_Service (Convert_Fixed_Point_String, VMM_ICODE)
|
|
VMM_Service (Get_Profile_Hex_Int, VMM_ICODE)
|
|
VMM_Service (Convert_Hex_String, VMM_ICODE)
|
|
VMM_Service (Get_Profile_Boolean, VMM_ICODE)
|
|
VMM_Service (Convert_Boolean_String, VMM_ICODE)
|
|
VMM_Service (Get_Profile_String, VMM_ICODE)
|
|
VMM_Service (Get_Next_Profile_String, VMM_ICODE)
|
|
VMM_Service (Get_Environment_String, VMM_ICODE)
|
|
VMM_Service (Get_Exec_Path, VMM_ICODE)
|
|
VMM_Service (Get_Config_Directory, VMM_ICODE)
|
|
VMM_Service (OpenFile, VMM_ICODE)
|
|
/*ENDMACROS*/
|
|
|
|
// OpenFile, if called after init, must point EDI to a buffer of at least
|
|
// this size.
|
|
|
|
#define VMM_OPENFILE_BUF_SIZE 260
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Get_PSP_Segment, VMM_ICODE)
|
|
VMM_Service (GetDOSVectors, VMM_ICODE)
|
|
VMM_Service (Get_Machine_Info)
|
|
/*ENDMACROS*/
|
|
|
|
#define GMIF_80486_BIT 0x10
|
|
#define GMIF_80486 (1 << GMIF_80486_BIT)
|
|
#define GMIF_PCXT_BIT 0x11
|
|
#define GMIF_PCXT (1 << GMIF_PCXT_BIT)
|
|
#define GMIF_MCA_BIT 0x12
|
|
#define GMIF_MCA (1 << GMIF_MCA_BIT)
|
|
#define GMIF_EISA_BIT 0x13
|
|
#define GMIF_EISA (1 << GMIF_EISA_BIT)
|
|
#define GMIF_CPUID_BIT 0x14
|
|
#define GMIF_CPUID (1 << GMIF_CPUID_BIT)
|
|
|
|
// Following service is not restricted to initialization
|
|
|
|
/*MACROS*/
|
|
VMM_Service (GetSet_HMA_Info)
|
|
VMM_Service (Set_System_Exit_Code)
|
|
|
|
VMM_Service (Fatal_Error_Handler)
|
|
VMM_Service (Fatal_Memory_Error)
|
|
|
|
// Called by VTD only
|
|
|
|
VMM_Service (Update_System_Clock)
|
|
|
|
/******************************************************************************
|
|
* D E B U G G I N G E X T E R N S
|
|
******************************************************************************/
|
|
|
|
VMM_Service (Test_Debug_Installed) // Valid call in retail also
|
|
|
|
VMM_Service (Out_Debug_String)
|
|
VMM_Service (Out_Debug_Chr)
|
|
VMM_Service (In_Debug_Chr)
|
|
VMM_Service (Debug_Convert_Hex_Binary)
|
|
VMM_Service (Debug_Convert_Hex_Decimal)
|
|
|
|
VMM_Service (Debug_Test_Valid_Handle)
|
|
VMM_Service (Validate_Client_Ptr)
|
|
VMM_Service (Test_Reenter)
|
|
VMM_Service (Queue_Debug_String)
|
|
VMM_Service (Log_Proc_Call)
|
|
VMM_Service (Debug_Test_Cur_VM)
|
|
|
|
VMM_Service (Get_PM_Int_Type)
|
|
VMM_Service (Set_PM_Int_Type)
|
|
|
|
VMM_Service (Get_Last_Updated_System_Time)
|
|
VMM_Service (Get_Last_Updated_VM_Exec_Time)
|
|
|
|
VMM_Service (Test_DBCS_Lead_Byte) // for DBCS Enabling
|
|
/*ENDMACROS*/
|
|
|
|
/* ASM
|
|
.errnz @@Test_DBCS_Lead_Byte - 100D1h ; VMM service table changed above this service
|
|
*/
|
|
|
|
/*************************************************************************
|
|
*************************************************************************
|
|
* END OF 3.00 SERVICE TABLE MUST NOT SHUFFLE SERVICES BEFORE THIS POINT
|
|
* FOR COMPATIBILITY.
|
|
*************************************************************************
|
|
*************************************************************************/
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_AddFreePhysPage, VMM_ICODE)
|
|
VMM_Service (_PageResetHandlePAddr)
|
|
VMM_Service (_SetLastV86Page, VMM_ICODE)
|
|
VMM_Service (_GetLastV86Page)
|
|
VMM_Service (_MapFreePhysReg)
|
|
VMM_Service (_UnmapFreePhysReg)
|
|
VMM_Service (_XchgFreePhysReg)
|
|
VMM_Service (_SetFreePhysRegCalBk, VMM_ICODE)
|
|
VMM_Service (Get_Next_Arena, VMM_ICODE)
|
|
VMM_Service (Get_Name_Of_Ugly_TSR, VMM_ICODE)
|
|
VMM_Service (Get_Debug_Options, VMM_ICODE)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flags for AddFreePhysPage
|
|
*/
|
|
#define AFPP_SWAPOUT 0x0001 // physical memory that must be swapped out
|
|
// and subsequently restored at system exit
|
|
/*
|
|
* Flags for PageChangePager
|
|
*/
|
|
#define PCP_CHANGEPAGER 0x1 // change the pager for the page range
|
|
#define PCP_CHANGEPAGERDATA 0x2 // change the pager data dword for the pages
|
|
#define PCP_VIRGINONLY 0x4 // make the above changes to virgin pages only
|
|
|
|
|
|
/*
|
|
* Bits for the ECX return of Get_Next_Arena
|
|
*/
|
|
#define GNA_HIDOSLINKED 0x0002 // High DOS arenas linked when WIN386 started
|
|
#define GNA_ISHIGHDOS 0x0004 // High DOS arenas do exist
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Set_Physical_HMA_Alias, VMM_ICODE)
|
|
VMM_Service (_GetGlblRng0V86IntBase, VMM_ICODE)
|
|
VMM_Service (_Add_Global_V86_Data_Area, VMM_ICODE)
|
|
|
|
VMM_Service (GetSetDetailedVMError)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Error code values for the GetSetDetailedVMError service. PLEASE NOTE
|
|
* that all of these error code values need to have bits set in the high
|
|
* word. This is to prevent collisions with other VMDOSAPP standard errors.
|
|
* Also, the low word must be non-zero.
|
|
*
|
|
* First set of errors (high word = 0001) are intended to be used
|
|
* when a VM is CRASHED (VNE_Crashed or VNE_Nuked bit set on
|
|
* VM_Not_Executeable).
|
|
*
|
|
* PLEASE NOTE that each of these errors (high word == 0001) actually
|
|
* has two forms:
|
|
*
|
|
* 0001xxxxh
|
|
* 8001xxxxh
|
|
*
|
|
* The device which sets the error initially always sets the error with
|
|
* the high bit CLEAR. The system will then optionally set the high bit
|
|
* depending on the result of the attempt to "nicely" crash the VM. This
|
|
* bit allows the system to tell the user whether the crash is likely or
|
|
* unlikely to destabalize the system.
|
|
*/
|
|
#define GSDVME_PRIVINST 0x00010001 /* Privledged instruction */
|
|
#define GSDVME_INVALINST 0x00010002 /* Invalid instruction */
|
|
#define GSDVME_INVALPGFLT 0x00010003 /* Invalid page fault */
|
|
#define GSDVME_INVALGPFLT 0x00010004 /* Invalid GP fault */
|
|
#define GSDVME_INVALFLT 0x00010005 /* Unspecified invalid fault */
|
|
#define GSDVME_USERNUKE 0x00010006 /* User requested NUKE of VM */
|
|
#define GSDVME_DEVNUKE 0x00010007 /* Device specific problem */
|
|
#define GSDVME_DEVNUKEHDWR 0x00010008 /* Device specific problem:
|
|
* invalid hardware fiddling
|
|
* by VM (invalid I/O)
|
|
*/
|
|
#define GSDVME_NUKENOMSG 0x00010009 /* Supress standard messages:
|
|
* SHELL_Message used for
|
|
* custom msg.
|
|
*/
|
|
#define GSDVME_OKNUKEMASK 0x80000000 /* "Nice nuke" bit */
|
|
|
|
/*
|
|
* Second set of errors (high word = 0002) are intended to be used
|
|
* when a VM start up is failed (VNE_CreateFail, VNE_CrInitFail, or
|
|
* VNE_InitFail bit set on VM_Not_Executeable).
|
|
*/
|
|
#define GSDVME_INSMEMV86 0x00020001 /* base V86 mem - V86MMGR */
|
|
#define GSDVME_INSV86SPACE 0x00020002 /* Kb Req too large - V86MMGR */
|
|
#define GSDVME_INSMEMXMS 0x00020003 /* XMS Kb Req - V86MMGR */
|
|
#define GSDVME_INSMEMEMS 0x00020004 /* EMS Kb Req - V86MMGR */
|
|
#define GSDVME_INSMEMV86HI 0x00020005 /* Hi DOS V86 mem - DOSMGR
|
|
* V86MMGR
|
|
*/
|
|
#define GSDVME_INSMEMVID 0x00020006 /* Base Video mem - VDD */
|
|
#define GSDVME_INSMEMVM 0x00020007 /* Base VM mem - VMM
|
|
* CB, Inst Buffer
|
|
*/
|
|
#define GSDVME_INSMEMDEV 0x00020008 /* Couldn't alloc base VM
|
|
* memory for device.
|
|
*/
|
|
#define GSDVME_CRTNOMSG 0x00020009 /* Supress standard messages:
|
|
* SHELL_Message used for
|
|
* custom msg.
|
|
*/
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Is_Debug_Chr)
|
|
|
|
// Mono_Out services
|
|
|
|
VMM_Service (Clear_Mono_Screen)
|
|
VMM_Service (Out_Mono_Chr)
|
|
VMM_Service (Out_Mono_String)
|
|
VMM_Service (Set_Mono_Cur_Pos)
|
|
VMM_Service (Get_Mono_Cur_Pos)
|
|
VMM_Service (Get_Mono_Chr)
|
|
|
|
// Service locates a byte in ROM
|
|
|
|
VMM_Service (Locate_Byte_In_ROM, VMM_ICODE)
|
|
|
|
VMM_Service (Hook_Invalid_Page_Fault)
|
|
VMM_Service (Unhook_Invalid_Page_Fault)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flag bits of IPF_Flags
|
|
*/
|
|
#define IPF_PGDIR 0x00000001 /* Page directory entry not-present */
|
|
#define IPF_V86PG 0x00000002 /* Unexpected not present Page in V86 */
|
|
#define IPF_V86PGH 0x00000004 /* Like IPF_V86PG at high linear */
|
|
#define IPF_INVTYP 0x00000008 /* page has invalid not present type */
|
|
#define IPF_PGERR 0x00000010 /* pageswap device failure */
|
|
#define IPF_REFLT 0x00000020 /* re-entrant page fault */
|
|
#define IPF_VMM 0x00000040 /* Page fault caused by a VxD */
|
|
#define IPF_PM 0x00000080 /* Page fault by VM in Prot Mode */
|
|
#define IPF_V86 0x00000100 /* Page fault by VM in V86 Mode */
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Set_Delete_On_Exit_File)
|
|
|
|
VMM_Service (Close_VM)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flags for Close_VM service
|
|
*/
|
|
|
|
#define CVF_CONTINUE_EXEC_BIT 0
|
|
#define CVF_CONTINUE_EXEC (1 << CVF_CONTINUE_EXEC_BIT)
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Enable_Touch_1st_Meg) // Debugging only
|
|
VMM_Service (Disable_Touch_1st_Meg) // Debugging only
|
|
|
|
VMM_Service (Install_Exception_Handler)
|
|
VMM_Service (Remove_Exception_Handler)
|
|
|
|
VMM_Service (Get_Crit_Status_No_Block)
|
|
/*ENDMACROS*/
|
|
|
|
/* ASM
|
|
; Check if VMM service table has changed above this service
|
|
.errnz @@Get_Crit_Status_No_Block - 100F1h
|
|
*/
|
|
|
|
#ifdef WIN40SERVICES
|
|
|
|
/*************************************************************************
|
|
*************************************************************************
|
|
*
|
|
* END OF 3.10 SERVICE TABLE MUST NOT SHUFFLE SERVICES BEFORE THIS POINT
|
|
* FOR COMPATIBILITY.
|
|
*************************************************************************
|
|
*************************************************************************/
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_GetLastUpdatedThreadExecTime)
|
|
|
|
VMM_Service (_Trace_Out_Service)
|
|
VMM_Service (_Debug_Out_Service)
|
|
VMM_Service (_Debug_Flags_Service)
|
|
/*ENDMACROS*/
|
|
|
|
#endif /* WIN40SERVICES */
|
|
|
|
|
|
/*
|
|
* Flags for _Debug_Flags_Service service.
|
|
*
|
|
* Don't change these unless you really really know what you're doing.
|
|
* We need to define these even if we are in WIN31COMPAT mode.
|
|
*/
|
|
|
|
#define DFS_LOG_BIT 0
|
|
#define DFS_LOG (1 << DFS_LOG_BIT)
|
|
#define DFS_PROFILE_BIT 1
|
|
#define DFS_PROFILE (1 << DFS_PROFILE_BIT)
|
|
#define DFS_TEST_CLD_BIT 2
|
|
#define DFS_TEST_CLD (1 << DFS_TEST_CLD_BIT)
|
|
#define DFS_NEVER_REENTER_BIT 3
|
|
#define DFS_NEVER_REENTER (1 << DFS_NEVER_REENTER_BIT)
|
|
#define DFS_TEST_REENTER_BIT 4
|
|
#define DFS_TEST_REENTER (1 << DFS_TEST_REENTER_BIT)
|
|
#define DFS_NOT_SWAPPING_BIT 5
|
|
#define DFS_NOT_SWAPPING (1 << DFS_NOT_SWAPPING_BIT)
|
|
#define DFS_TEST_BLOCK_BIT 6
|
|
#define DFS_TEST_BLOCK (1 << DFS_TEST_BLOCK_BIT)
|
|
|
|
#define DFS_RARE_SERVICES 0xFFFFFF80
|
|
|
|
#define DFS_EXIT_NOBLOCK (DFS_RARE_SERVICES+0)
|
|
#define DFS_ENTER_NOBLOCK (DFS_RARE_SERVICES+DFS_TEST_BLOCK)
|
|
|
|
#define DFS_TEST_NEST_EXEC (DFS_RARE_SERVICES+1)
|
|
|
|
#ifdef WIN40SERVICES
|
|
|
|
/*MACROS*/
|
|
VMM_Service (VMMAddImportModuleName)
|
|
|
|
VMM_Service (VMM_Add_DDB)
|
|
VMM_Service (VMM_Remove_DDB)
|
|
|
|
VMM_Service (Test_VM_Ints_Enabled)
|
|
VMM_Service (_BlockOnID)
|
|
|
|
VMM_Service (Schedule_Thread_Event)
|
|
VMM_Service (Cancel_Thread_Event)
|
|
VMM_Service (Set_Thread_Time_Out)
|
|
VMM_Service (Set_Async_Time_Out)
|
|
|
|
VMM_Service (_AllocateThreadDataSlot)
|
|
VMM_Service (_FreeThreadDataSlot)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flag equates for _CreateMutex
|
|
*/
|
|
#define MUTEX_MUST_COMPLETE 1L
|
|
#define MUTEX_NO_CLEANUP_THREAD_STATE 2L
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_CreateMutex)
|
|
|
|
VMM_Service (_DestroyMutex)
|
|
VMM_Service (_GetMutexOwner)
|
|
VMM_Service (Call_When_Thread_Switched)
|
|
|
|
VMM_Service (VMMCreateThread)
|
|
VMM_Service (_GetThreadExecTime)
|
|
VMM_Service (VMMTerminateThread)
|
|
|
|
VMM_Service (Get_Cur_Thread_Handle)
|
|
VMM_Service (Test_Cur_Thread_Handle)
|
|
VMM_Service (Get_Sys_Thread_Handle)
|
|
VMM_Service (Test_Sys_Thread_Handle)
|
|
VMM_Service (Validate_Thread_Handle)
|
|
VMM_Service (Get_Initial_Thread_Handle)
|
|
VMM_Service (Test_Initial_Thread_Handle)
|
|
VMM_Service (Debug_Test_Valid_Thread_Handle)
|
|
VMM_Service (Debug_Test_Cur_Thread)
|
|
|
|
VMM_Service (VMM_GetSystemInitState)
|
|
|
|
VMM_Service (Cancel_Call_When_Thread_Switched)
|
|
VMM_Service (Get_Next_Thread_Handle)
|
|
VMM_Service (Adjust_Thread_Exec_Priority)
|
|
|
|
VMM_Service (_Deallocate_Device_CB_Area)
|
|
VMM_Service (Remove_IO_Handler)
|
|
VMM_Service (Remove_Mult_IO_Handlers)
|
|
VMM_Service (Unhook_V86_Int_Chain)
|
|
VMM_Service (Unhook_V86_Fault)
|
|
VMM_Service (Unhook_PM_Fault)
|
|
VMM_Service (Unhook_VMM_Fault)
|
|
VMM_Service (Unhook_Device_Service)
|
|
|
|
VMM_Service (_PageReserve)
|
|
VMM_Service (_PageCommit)
|
|
VMM_Service (_PageDecommit)
|
|
VMM_Service (_PagerRegister)
|
|
VMM_Service (_PagerQuery)
|
|
VMM_Service (_PagerDeregister)
|
|
VMM_Service (_ContextCreate)
|
|
VMM_Service (_ContextDestroy)
|
|
VMM_Service (_PageAttach)
|
|
VMM_Service (_PageFlush)
|
|
VMM_Service (_SignalID)
|
|
VMM_Service (_PageCommitPhys)
|
|
|
|
VMM_Service (_Register_Win32_Services)
|
|
|
|
VMM_Service (Cancel_Call_When_Not_Critical)
|
|
VMM_Service (Cancel_Call_When_Idle)
|
|
VMM_Service (Cancel_Call_When_Task_Switched)
|
|
|
|
VMM_Service (_Debug_Printf_Service)
|
|
VMM_Service (_EnterMutex)
|
|
VMM_Service (_LeaveMutex)
|
|
VMM_Service (Simulate_VM_IO)
|
|
VMM_Service (Signal_Semaphore_No_Switch)
|
|
|
|
VMM_Service (_ContextSwitch)
|
|
VMM_Service (_PageModifyPermissions)
|
|
VMM_Service (_PageQuery)
|
|
|
|
VMM_Service (_EnterMustComplete)
|
|
VMM_Service (_LeaveMustComplete)
|
|
VMM_Service (_ResumeExecMustComplete)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Flag equates for _GetThreadTerminationStatus
|
|
*/
|
|
#define THREAD_TERM_STATUS_CRASH_PEND 1L
|
|
#define THREAD_TERM_STATUS_NUKE_PEND 2L
|
|
#define THREAD_TERM_STATUS_SUSPEND_PEND 4L
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_GetThreadTerminationStatus)
|
|
VMM_Service (_GetInstanceInfo)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Return values for _GetInstanceInfo
|
|
*/
|
|
#define INSTINFO_NONE 0 /* no data instanced in range */
|
|
#define INSTINFO_SOME 1 /* some data instanced in range */
|
|
#define INSTINFO_ALL 2 /* all data instanced in range */
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_ExecIntMustComplete)
|
|
VMM_Service (_ExecVxDIntMustComplete)
|
|
|
|
VMM_Service (Begin_V86_Serialization)
|
|
|
|
VMM_Service (Unhook_V86_Page)
|
|
VMM_Service (VMM_GetVxDLocationList)
|
|
VMM_Service (VMM_GetDDBList)
|
|
VMM_Service (Unhook_NMI_Event)
|
|
|
|
VMM_Service (Get_Instanced_V86_Int_Vector)
|
|
VMM_Service (Get_Set_Real_DOS_PSP)
|
|
/*ENDMACROS*/
|
|
|
|
#define GSRDP_Set 0x0001
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Call_Priority_Thread_Event)
|
|
VMM_Service (Get_System_Time_Address)
|
|
VMM_Service (Get_Crit_Status_Thread)
|
|
|
|
VMM_Service (Get_DDB)
|
|
VMM_Service (Directed_Sys_Control)
|
|
/*ENDMACROS*/
|
|
|
|
// Registry APIs for VxDs
|
|
/*MACROS*/
|
|
VMM_Service (_RegOpenKey)
|
|
VMM_Service (_RegCloseKey)
|
|
VMM_Service (_RegCreateKey)
|
|
VMM_Service (_RegDeleteKey)
|
|
VMM_Service (_RegEnumKey)
|
|
VMM_Service (_RegQueryValue)
|
|
VMM_Service (_RegSetValue)
|
|
VMM_Service (_RegDeleteValue)
|
|
VMM_Service (_RegEnumValue)
|
|
VMM_Service (_RegQueryValueEx)
|
|
VMM_Service (_RegSetValueEx)
|
|
/*ENDMACROS*/
|
|
|
|
#ifndef REG_SZ // define only if not there already
|
|
|
|
#define REG_SZ 0x0001
|
|
#define REG_BINARY 0x0003
|
|
|
|
#endif
|
|
|
|
#ifndef HKEY_LOCAL_MACHINE // define only if not there already
|
|
|
|
#define HKEY_CLASSES_ROOT 0x80000000
|
|
#define HKEY_CURRENT_USER 0x80000001
|
|
#define HKEY_LOCAL_MACHINE 0x80000002
|
|
#define HKEY_USERS 0x80000003
|
|
#define HKEY_PERFORMANCE_DATA 0x80000004
|
|
#define HKEY_CURRENT_CONFIG 0x80000005
|
|
#define HKEY_DYN_DATA 0x80000006
|
|
|
|
#endif
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_CallRing3)
|
|
VMM_Service (Exec_PM_Int)
|
|
VMM_Service (_RegFlushKey)
|
|
VMM_Service (_PageCommitContig)
|
|
VMM_Service (_GetCurrentContext)
|
|
|
|
VMM_Service (_LocalizeSprintf)
|
|
VMM_Service (_LocalizeStackSprintf)
|
|
|
|
VMM_Service (Call_Restricted_Event)
|
|
VMM_Service (Cancel_Restricted_Event)
|
|
|
|
VMM_Service (Register_PEF_Provider, VMM_ICODE)
|
|
|
|
VMM_Service (_GetPhysPageInfo)
|
|
|
|
VMM_Service (_RegQueryInfoKey)
|
|
VMM_Service (MemArb_Reserve_Pages)
|
|
/*ENDMACROS*/
|
|
|
|
/*
|
|
* Return values for _GetPhysPageInfo
|
|
*/
|
|
#define PHYSINFO_NONE 0 /* no pages in the specified range exist */
|
|
#define PHYSINFO_SOME 1 /* some pages in the specified range exist */
|
|
#define PHYSINFO_ALL 2 /* all pages in the specified range exist */
|
|
|
|
// New timeslicer services
|
|
/*MACROS*/
|
|
VMM_Service (Time_Slice_Sys_VM_Idle)
|
|
VMM_Service (Time_Slice_Sleep)
|
|
VMM_Service (Boost_With_Decay)
|
|
VMM_Service (Set_Inversion_Pri)
|
|
VMM_Service (Reset_Inversion_Pri)
|
|
VMM_Service (Release_Inversion_Pri)
|
|
VMM_Service (Get_Thread_Win32_Pri)
|
|
VMM_Service (Set_Thread_Win32_Pri)
|
|
VMM_Service (Set_Thread_Static_Boost)
|
|
VMM_Service (Set_VM_Static_Boost)
|
|
VMM_Service (Release_Inversion_Pri_ID)
|
|
VMM_Service (Attach_Thread_To_Group)
|
|
VMM_Service (Detach_Thread_From_Group)
|
|
VMM_Service (Set_Group_Static_Boost)
|
|
|
|
VMM_Service (_GetRegistryPath, VMM_ICODE)
|
|
VMM_Service (_GetRegistryKey)
|
|
/*ENDMACROS*/
|
|
|
|
// TYPE definitions for _GetRegistryKey
|
|
|
|
#define REGTYPE_ENUM 0
|
|
#define REGTYPE_CLASS 1
|
|
#define REGTYPE_VXD 2
|
|
|
|
// Flag definitions for _GetRegistryKey
|
|
#define REGKEY_OPEN 0
|
|
#define REGKEY_CREATE_IFNOTEXIST 1
|
|
|
|
// Flag definitions for _Assert_Range
|
|
#define ASSERT_RANGE_NULL_BAD 0x00000000
|
|
#define ASSERT_RANGE_NULL_OK 0x00000001
|
|
#define ASSERT_RANGE_IS_ASCIIZ 0x00000002
|
|
#define ASSERT_RANGE_IS_NOT_ASCIIZ 0x00000000
|
|
#define ASSERT_RANGE_NO_DEBUG 0x80000000
|
|
#define ASSERT_RANGE_BITS 0x80000003
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Cleanup_Thread_State)
|
|
VMM_Service (_RegRemapPreDefKey)
|
|
VMM_Service (End_V86_Serialization)
|
|
VMM_Service (_Assert_Range)
|
|
VMM_Service (_Sprintf)
|
|
VMM_Service (_PageChangePager)
|
|
VMM_Service (_RegCreateDynKey)
|
|
VMM_Service (_RegQueryMultipleValues)
|
|
|
|
// Additional timeslicer services
|
|
VMM_Service (Boost_Thread_With_VM)
|
|
/*ENDMACROS*/
|
|
|
|
// Flag definitions for Get_Boot_Flags
|
|
|
|
#define BOOT_CLEAN 0x00000001
|
|
#define BOOT_DOSCLEAN 0x00000002
|
|
#define BOOT_NETCLEAN 0x00000004
|
|
#define BOOT_INTERACTIVE 0x00000008
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Get_Boot_Flags)
|
|
VMM_Service (Set_Boot_Flags)
|
|
|
|
// String and memory services
|
|
VMM_Service (_lstrcpyn)
|
|
VMM_Service (_lstrlen)
|
|
VMM_Service (_lmemcpy)
|
|
|
|
VMM_Service (_GetVxDName)
|
|
|
|
// For vwin32 use only
|
|
VMM_Service (Force_Mutexes_Free)
|
|
VMM_Service (Restore_Forced_Mutexes)
|
|
/*ENDMACROS*/
|
|
|
|
// Reclaimable low memory services
|
|
/*MACROS*/
|
|
VMM_Service (_AddReclaimableItem)
|
|
VMM_Service (_SetReclaimableItem)
|
|
VMM_Service (_EnumReclaimableItem)
|
|
/*ENDMACROS*/
|
|
|
|
// completely wake sys VM from idle state
|
|
/*MACROS*/
|
|
VMM_Service (Time_Slice_Wake_Sys_VM)
|
|
VMM_Service (VMM_Replace_Global_Environment)
|
|
VMM_Service (Begin_Non_Serial_Nest_V86_Exec)
|
|
VMM_Service (Get_Nest_Exec_Status)
|
|
/*ENDMACROS*/
|
|
|
|
// Bootlogging services
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Open_Boot_Log)
|
|
VMM_Service (Write_Boot_Log)
|
|
VMM_Service (Close_Boot_Log)
|
|
VMM_Service (EnableDisable_Boot_Log)
|
|
VMM_Service (_Call_On_My_Stack)
|
|
/*ENDMACROS*/
|
|
|
|
// Another instance data service
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Get_Inst_V86_Int_Vec_Base)
|
|
/*ENDMACROS*/
|
|
|
|
// Case insensitive functions -- SEE WARNINGS IN DOCS BEFORE USING!
|
|
/*MACROS*/
|
|
VMM_Service (_lstrcmpi)
|
|
VMM_Service (_strupr)
|
|
/*ENDMACROS*/
|
|
|
|
/*MACROS*/
|
|
VMM_Service (Log_Fault_Call_Out)
|
|
VMM_Service (_AtEventTime)
|
|
/*ENDMACROS*/
|
|
|
|
#endif /* WIN40SERVICES */
|
|
|
|
#ifdef WIN403SERVICES
|
|
//
|
|
// 4.03 Services
|
|
//
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_PageOutPages)
|
|
/*ENDMACROS*/
|
|
|
|
// Flag definitions for _PageOutPages
|
|
|
|
#define PAGEOUT_PRIVATE 0x00000001
|
|
#define PAGEOUT_SHARED 0x00000002
|
|
#define PAGEOUT_SYSTEM 0x00000004
|
|
#define PAGEOUT_REGION 0x00000008
|
|
#define PAGEOUT_ALL (PAGEOUT_PRIVATE | PAGEOUT_SHARED | PAGEOUT_SYSTEM)
|
|
|
|
/*MACROS*/
|
|
VMM_Service (_Call_On_My_Not_Flat_Stack)
|
|
VMM_Service (_LinRegionLock)
|
|
VMM_Service (_LinRegionUnLock)
|
|
VMM_Service (_AttemptingSomethingDangerous)
|
|
VMM_Service (_Vsprintf)
|
|
VMM_Service (_Vsprintfw)
|
|
VMM_Service (Load_FS_Service)
|
|
VMM_Service (Assert_FS_Service)
|
|
VMM_StdCall_Service (RtlUnwind, 4)
|
|
VMM_StdCall_Service (RtlRaiseException, 1)
|
|
VMM_StdCall_Service (RtlRaiseStatus, 1)
|
|
|
|
VMM_StdCall_Service (KeGetCurrentIrql, 0)
|
|
VMM_FastCall_Service (KfRaiseIrql, 1)
|
|
VMM_FastCall_Service (KfLowerIrql, 1)
|
|
|
|
VMM_Service (_Begin_Preemptable_Code)
|
|
VMM_Service (_End_Preemptable_Code)
|
|
VMM_FastCall_Service (Set_Preemptable_Count, 1)
|
|
|
|
VMM_StdCall_Service (KeInitializeDpc, 3)
|
|
VMM_StdCall_Service (KeInsertQueueDpc, 3)
|
|
VMM_StdCall_Service (KeRemoveQueueDpc, 1)
|
|
|
|
VMM_StdCall_Service (HeapAllocateEx, 4)
|
|
VMM_StdCall_Service (HeapReAllocateEx, 5)
|
|
VMM_StdCall_Service (HeapGetSizeEx, 2)
|
|
VMM_StdCall_Service (HeapFreeEx, 2)
|
|
//VMM_Service (_Get_CPUID_Flags)
|
|
|
|
/*ENDMACROS*/
|
|
|
|
#endif /* WIN403SERVICES */
|
|
|
|
/*MACROS*/
|
|
End_Service_Table(VMM, VMM)
|
|
/*ENDMACROS*/
|
|
|
|
/*XLATOFF*/
|
|
#pragma warning (default:4003) // turn on not enough params warning
|
|
|
|
#ifndef try
|
|
#define try __try
|
|
#define except __except
|
|
#define finally __finally
|
|
#define leave __leave
|
|
#ifndef exception_code
|
|
#define exception_code __exception_code
|
|
#endif // exception_code
|
|
#endif // try
|
|
|
|
#ifndef EXCEPTION_EXECUTE_HANDLER
|
|
#define EXCEPTION_EXECUTE_HANDLER 1
|
|
#define EXCEPTION_CONTINUE_SEARCH 0
|
|
#define EXCEPTION_CONTINUE_EXECUTION -1
|
|
#endif
|
|
/*XLATON*/
|
|
|
|
#define COMNFS_FLAT 0xFFFFFFFF
|
|
|
|
#define ASD_MAX_REF_DATA 64 // If bigger than this, a checksum is used
|
|
|
|
struct _vmmguid {
|
|
unsigned long Data1;
|
|
unsigned short Data2;
|
|
unsigned short Data3;
|
|
unsigned char Data4[8];
|
|
};
|
|
|
|
typedef struct _vmmguid VMMGUID;
|
|
typedef VMMGUID *VMMREFIID;
|
|
|
|
typedef DWORD ASD_RESULT;
|
|
|
|
#define ASD_ERROR_NONE 0x00000000
|
|
#define ASD_CHECK_FAIL 0x00000001 // The flag is set that this failed before
|
|
#define ASD_CHECK_SUCCESS 0x00000002 // The flag is set that this succeeded before
|
|
#define ASD_CHECK_UNKNOWN 0x00000003 // No flag is set
|
|
#define ASD_ERROR_BAD_TIME 0x00000004 // Under cli
|
|
#define ASD_REGISTRY_ERROR 0x00000005 // Unknown registry error
|
|
#define ASD_CLEAN_BOOT 0x00000006 // Clean booting fails everything
|
|
#define ASD_OUT_OF_MEMORY 0x00000007 // Ran out of memory (extremely rare)
|
|
#define ASD_FILE_ERROR 0x00000008 // Int 21 to flush the info file failed
|
|
#define ASD_ALREADY_SET 0x00000009 // ASD_CHECK* done twice on same vgOperation/pRefData
|
|
#define ASD_MISSING_CHECK 0x0000000A // ASD_DONE* on something not set
|
|
#define ASD_BAD_PARAMETER 0x0000000B // Invalid operation, refiid or ref pointer
|
|
|
|
#define ASD_OP_CHECK_AND_WRITE_FAIL_IF_UNKNOWN 0x00000000
|
|
#define ASD_OP_CHECK_AND_ALWAYS_WRITE_FAIL 0x00000001
|
|
#define ASD_OP_CHECK 0x00000002
|
|
#define ASD_OP_DONE_AND_SET_SUCCESS 0x00000003
|
|
#define ASD_OP_SET_FAIL 0x00000004
|
|
#define ASD_OP_SET_SUCCESS 0x00000005
|
|
#define ASD_OP_SET_UNKNOWN 0x00000006
|
|
#define ASD_OP_DONE 0x00000007
|
|
|
|
// Flag definitions for _Add/_Set/_EnumReclaimableItem
|
|
|
|
#define RS_RECLAIM 0x00000001
|
|
#define RS_RESTORE 0x00000002
|
|
#define RS_DOSARENA 0x00000004
|
|
|
|
// Structure definition for _EnumReclaimableItem
|
|
|
|
struct ReclaimStruc {
|
|
ULONG RS_Linear; // low (< 1meg) address of item
|
|
ULONG RS_Bytes; // size of item in bytes
|
|
ULONG RS_CallBack; // callback, if any (zero if none)
|
|
ULONG RS_RefData; // reference data for callback, if any
|
|
ULONG RS_HookTable; // real-mode hook table (zero if none)
|
|
ULONG RS_Flags; // 0 or more of the RS_* equates
|
|
};
|
|
|
|
typedef struct ReclaimStruc *PReclaimStruc;
|
|
|
|
//
|
|
// Structures for Force_Mutexes_Free/Restore_Forced_Mutexes
|
|
//
|
|
typedef struct frmtx {
|
|
struct frmtx *frmtx_pfrmtxNext;
|
|
DWORD frmtx_hmutex;
|
|
DWORD frmtx_cEnterCount;
|
|
DWORD frmtx_pthcbOwner;
|
|
DWORD frmtx_htimeout;
|
|
} FRMTX;
|
|
|
|
typedef struct vmmfrinfo {
|
|
struct frmtx vmmfrinfo_frmtxDOS;
|
|
struct frmtx vmmfrinfo_frmtxV86;
|
|
struct frmtx vmmfrinfo_frmtxOther;
|
|
} VMMFRINFO;
|
|
|
|
/*
|
|
* Data structure for _GetDemandPageInfo
|
|
*/
|
|
struct DemandInfoStruc {
|
|
ULONG DILin_Total_Count; /* # pages in linear address space */
|
|
ULONG DIPhys_Count; /* Count of phys pages */
|
|
ULONG DIFree_Count; /* Count of free phys pages */
|
|
ULONG DIUnlock_Count; /* Count of unlocked Phys Pages */
|
|
ULONG DILinear_Base_Addr; /* Base of pageable address space */
|
|
ULONG DILin_Total_Free; /* Total Count of free linear pages */
|
|
|
|
/*
|
|
* The following 5 fields are all running totals, kept from the time
|
|
* the system was started
|
|
*/
|
|
ULONG DIPage_Faults; /* total page faults */
|
|
ULONG DIPage_Ins; /* calls to pagers to page in a page */
|
|
ULONG DIPage_Outs; /* calls to pagers to page out a page*/
|
|
ULONG DIPage_Discards; /* pages discarded w/o calling pager */
|
|
ULONG DIInstance_Faults; /* instance page faults */
|
|
|
|
ULONG DIPagingFileMax; /* maximum # of pages that could be in paging file */
|
|
ULONG DIPagingFileInUse; /* # of pages of paging file currently in use */
|
|
|
|
ULONG DICommit_Count; /* Total committed memory, in pages */
|
|
|
|
ULONG DIReserved[2]; /* Reserved for expansion */
|
|
};
|
|
|
|
/*
|
|
* Data structure for _AddInstanceItem
|
|
*/
|
|
struct InstDataStruc {
|
|
ULONG InstLinkF; /* INIT <0> RESERVED */
|
|
ULONG InstLinkB; /* INIT <0> RESERVED */
|
|
ULONG InstLinAddr; /* Linear address of start of block */
|
|
ULONG InstSize; /* Size of block in bytes */
|
|
ULONG InstType; /* Type of block */
|
|
};
|
|
|
|
/*
|
|
* Values for InstType
|
|
*/
|
|
#define INDOS_FIELD 0x100 /* Bit indicating INDOS switch requirements */
|
|
#define ALWAYS_FIELD 0x200 /* Bit indicating ALWAYS switch requirements */
|
|
#define OPTIONAL_FIELD 0x400 /* Bit indicating optional instancing requirements */
|
|
|
|
/*
|
|
* Data structure for Hook_Invalid_Page_Fault handlers.
|
|
*
|
|
* This is the structure of the "invalid page fault information"
|
|
* which is pointed to by EDI when Invalid page fault hookers
|
|
* are called.
|
|
*
|
|
* Page faults can occur on a VM which is not current by touching the VM at
|
|
* its high linear address. In this case, IPF_FaultingVM may not be the
|
|
* current VM, it will be set to the VM whos high linear address was touched.
|
|
*/
|
|
|
|
struct IPF_Data {
|
|
ULONG IPF_LinAddr; /* CR2 address of fault */
|
|
ULONG IPF_MapPageNum; /* Possible converted page # of fault */
|
|
ULONG IPF_PTEEntry; /* Contents of PTE that faulted */
|
|
ULONG IPF_FaultingVM; /* May not = Current VM (IPF_V86PgH set) */
|
|
ULONG IPF_Flags; /* Flags */
|
|
};
|
|
|
|
/*
|
|
*
|
|
* Install_Exception_Handler data structure
|
|
*
|
|
*/
|
|
|
|
struct Exception_Handler_Struc {
|
|
ULONG EH_Reserved;
|
|
ULONG EH_Start_EIP;
|
|
ULONG EH_End_EIP;
|
|
ULONG EH_Handler;
|
|
};
|
|
|
|
/*
|
|
* Flags passed in new memory manager functions
|
|
*/
|
|
|
|
/* PageReserve arena values */
|
|
#define PR_PRIVATE 0x80000400 /* anywhere in private arena */
|
|
#define PR_SHARED 0x80060000 /* anywhere in shared arena */
|
|
#define PR_SYSTEM 0x80080000 /* anywhere in system arena */
|
|
|
|
/* PageReserve flags */
|
|
#define PR_FIXED 0x00000008 /* don't move during PageReAllocate */
|
|
#define PR_4MEG 0x00000001 /* allocate on 4mb boundary */
|
|
#define PR_STATIC 0x00000010 /* see PageReserve documentation */
|
|
|
|
/* PageCommit default pager handle values */
|
|
#define PD_ZEROINIT 0x00000001 /* swappable zero-initialized pages */
|
|
#define PD_NOINIT 0x00000002 /* swappable uninitialized pages */
|
|
#define PD_FIXEDZERO 0x00000003 /* fixed zero-initialized pages */
|
|
#define PD_FIXED 0x00000004 /* fixed uninitialized pages */
|
|
|
|
/* PageCommit flags */
|
|
#define PC_FIXED 0x00000008 /* pages are permanently locked */
|
|
#define PC_LOCKED 0x00000080 /* pages are made present and locked*/
|
|
#define PC_LOCKEDIFDP 0x00000100 /* pages are locked if swap via DOS */
|
|
#define PC_WRITEABLE 0x00020000 /* make the pages writeable */
|
|
#define PC_USER 0x00040000 /* make the pages ring 3 accessible */
|
|
#define PC_INCR 0x40000000 /* increment "pagerdata" each page */
|
|
#define PC_PRESENT 0x80000000 /* make pages initially present */
|
|
#define PC_STATIC 0x20000000 /* allow commit in PR_STATIC object */
|
|
#define PC_DIRTY 0x08000000 /* make pages initially dirty */
|
|
#define PC_CACHEDIS 0x00100000 /* Allocate uncached pages - new for WDM */
|
|
#define PC_CACHEWT 0x00080000 /* Allocate write through cache pages - new for WDM */
|
|
|
|
/* PageCommitContig additional flags */
|
|
#define PCC_ZEROINIT 0x00000001 /* zero-initialize new pages */
|
|
#define PCC_NOLIN 0x10000000 /* don't map to any linear address */
|
|
|
|
|
|
/*MTRR type flags */
|
|
#define MTRR_UC 0
|
|
#define MTRR_WC 1
|
|
#define MTRR_WT 4
|
|
#define MTRR_WP 5
|
|
#define MTRR_WB 6
|
|
|
|
/*
|
|
* Structure and flags for PageQuery
|
|
*/
|
|
#ifndef _WINNT_
|
|
typedef struct _MEMORY_BASIC_INFORMATION {
|
|
ULONG mbi_BaseAddress;
|
|
ULONG mbi_AllocationBase;
|
|
ULONG mbi_AllocationProtect;
|
|
ULONG mbi_RegionSize;
|
|
ULONG mbi_State;
|
|
ULONG mbi_Protect;
|
|
ULONG mbi_Type;
|
|
} MEMORY_BASIC_INFORMATION, *PMEMORY_BASIC_INFORMATION;
|
|
|
|
#define PAGE_NOACCESS 0x01
|
|
#define PAGE_READONLY 0x02
|
|
#define PAGE_READWRITE 0x04
|
|
#define MEM_COMMIT 0x1000
|
|
#define MEM_RESERVE 0x2000
|
|
#define MEM_FREE 0x10000
|
|
#define MEM_PRIVATE 0x20000
|
|
#endif
|
|
|
|
|
|
/***ET+ PD - Pager Descriptor
|
|
*
|
|
* A PD describes a set of routines to call to bring a page into
|
|
* the system or to get it out. Each committed page in the system
|
|
* has an associated PD, a handle to which is stored in the page's
|
|
* VP.
|
|
*
|
|
* For any field that is 0, the pager will not be notified
|
|
* when that action takes place.
|
|
*
|
|
* For the purpose of pagers, a page can be in one of the two states
|
|
* describing its current contents:
|
|
*
|
|
* clean - page has not been written to since its last page out
|
|
* dirty - page has been written to since its last page out
|
|
*
|
|
* A page also is in one of two persistent states:
|
|
*
|
|
* virgin - page has never been written to since it was committed
|
|
* tainted - page has been written to since it was committed
|
|
*
|
|
* Note that a tainted page may be either dirty or clean, but a
|
|
* virgin page is by definition clean.
|
|
*
|
|
* Examples of PDs:
|
|
*
|
|
* For 32-bit EXE code or read-only data:
|
|
*
|
|
* pd_virginin = routine to load page from an exe file
|
|
* pd_taintedin = 0
|
|
* pd_cleanout = 0
|
|
* pd_dirtyout = 0
|
|
* pd_virginfree = 0
|
|
* pd_taintedfree = 0
|
|
* pd_dirty = 0
|
|
* pd_type = PD_PAGERONLY
|
|
*
|
|
* For 32-bit EXE writeable data:
|
|
*
|
|
* pd_virginin = routine to load page from an exe file
|
|
* pd_taintedin = routine to load page from swap file
|
|
* pd_cleanout = 0
|
|
* pd_dirtyout = routine to write a page out to the swap file
|
|
* pd_virginfree = 0
|
|
* pd_taintedfree = routine to free page from the swap file
|
|
* pd_dirty = routine to free page from the swap file
|
|
* pd_type = PD_SWAPPER
|
|
*
|
|
* For zero-initialized swappable data:
|
|
*
|
|
* pd_virginin = routine to zero-fill a page
|
|
* pd_taintedin = routine to load page from swap file
|
|
* pd_cleanout = 0
|
|
* pd_dirtyout = routine to write a page out to the swap file
|
|
* pd_virginfree = 0
|
|
* pd_taintedfree = routine to free page from the swap file
|
|
* pd_dirty = routine to free page from the swap file
|
|
* pd_type = PD_SWAPPER
|
|
*/
|
|
/* typedefs for various pager functions */
|
|
|
|
typedef ULONG _cdecl FUNPAGE(PULONG ppagerdata, PVOID ppage, ULONG faultpage);
|
|
|
|
typedef FUNPAGE * PFUNPAGE;
|
|
|
|
struct pd_s {
|
|
/*
|
|
* The following four fields are entry points in the pager which
|
|
* we call to page in or page out a page. The following parameters
|
|
* are passed to the pager during these calls:
|
|
*
|
|
* ppagerdata - pointer to the pager-specific dword of data
|
|
* stored with the virtual page. The pager is
|
|
* free to modify the contents of this dword
|
|
* DURING the page in or out, but not afterwards.
|
|
*
|
|
* ppage - pointer to page going in or out (a ring 0 alias
|
|
* to the physical page). The pager should use this
|
|
* address to access the contents of the page.
|
|
*
|
|
* faultpage - faulting linear page number for page-ins, -1 for
|
|
* page-outs. This address should not be accessed
|
|
* by the pager. It is provided for information
|
|
* only. Note that a single page can be mapped at
|
|
* more than one linear address because of the
|
|
* MapIntoV86 and LinMapIntoV86 services.
|
|
*
|
|
* The pager should return non-0 if the page was successfully
|
|
* paged, or 0 if it failed.
|
|
*/
|
|
PFUNPAGE pd_virginin; /* in - while page has never been written to */
|
|
PFUNPAGE pd_taintedin; /* in - page written to at least once */
|
|
PFUNPAGE pd_cleanout; /* out - page not written to since last out */
|
|
PFUNPAGE pd_dirtyout; /* out - page was written to since last out */
|
|
|
|
/*
|
|
* The pd_*free routines are used to inform the pager when the last
|
|
* reference to a virtual page controlled by the pager is
|
|
* decommitted. A common use of this notification is to
|
|
* free space in a backing file, or write the page contents
|
|
* into the backing file.
|
|
*
|
|
* These calls take the same parameters as the page-out and -in
|
|
* functions, but no return value is recognized. The "ppage"
|
|
* and "faultpage" parameters will always be 0.
|
|
*/
|
|
PFUNPAGE pd_virginfree; /* decommit of never-written-to page */
|
|
PFUNPAGE pd_taintedfree; /* decommit of page written to at least once*/
|
|
|
|
/*
|
|
* The pd_dirty routine is used to inform the pager when the
|
|
* memory manager detects that a page has been written to. The memory
|
|
* manager does not detect the write at the instant it occurs, so
|
|
* the pager should not depend upon prompt notification. A common
|
|
* use of this notification might be to invalidate cached data.
|
|
* If the page was dirtied in more than one memory context,
|
|
* the pager's pd_dirty routine will be called once for each
|
|
* context.
|
|
*
|
|
* These calls take the same parameters as the page-out and -in
|
|
* functions except that the "ppage" parameter isn't valid and
|
|
* no return value is recognized.
|
|
*/
|
|
PFUNPAGE pd_dirty;
|
|
|
|
/*
|
|
* The pd_type field gives the sytem information about the
|
|
* overcommit characteristics of pages controlled by this pager.
|
|
* The following are allowable values for the field:
|
|
*
|
|
* PD_SWAPPER - under some conditions, pages of this type
|
|
* may be paged out into the swap file
|
|
* PD_PAGERONLY - pages controlled by this pager will never
|
|
* be paged out to the swap file
|
|
*
|
|
* In addition, the following value may be or'ed in to the pd_type field:
|
|
*
|
|
* PD_NESTEXEC - must be specified if either the pd_cleanout or pd_dirtyout
|
|
* functions perform nested excecution or block using the
|
|
* BLOCK_SVC_INTS flag. To be safe, this flag should always be
|
|
* specified if the pager does any sort of file i/o to anything
|
|
* other than the default paging file.
|
|
*/
|
|
ULONG pd_type;
|
|
};
|
|
typedef struct pd_s PD;
|
|
typedef PD * PPD;
|
|
|
|
/* values for pd_type */
|
|
#define PD_SWAPPER 0 /* pages need direct accounting in swap file */
|
|
#define PD_PAGERONLY 1 /* pages will never be swapped */
|
|
#define PD_NESTEXEC 2 /* page out funtion uses nested execution */
|
|
|
|
#endif // Not_VxD
|
|
|
|
/*
|
|
* The size of a page of memory
|
|
*/
|
|
#define PAGESHIFT 12
|
|
#define PAGESIZE (1 << PAGESHIFT)
|
|
#define PAGEMASK (PAGESIZE - 1)
|
|
|
|
#define PAGE(p) ((DWORD)(p) >> PAGESHIFT)
|
|
#define NPAGES(cb) (((DWORD)(cb) + PAGEMASK) >> PAGESHIFT)
|
|
|
|
/*
|
|
* Address space (arena) boundaries
|
|
*/
|
|
#define MAXSYSTEMLADDR ((ULONG) 0xffbfffff) /* 4 gig - 4meg */
|
|
#define MINSYSTEMLADDR ((ULONG) 0xc0000000) /* 3 gig */
|
|
#define MAXSHAREDLADDR ((ULONG) 0xbfffffff)
|
|
#define MINSHAREDLADDR ((ULONG) 0x80000000) /* 2 gig */
|
|
#define MAXPRIVATELADDR ((ULONG) 0x7fffffff)
|
|
#define MINPRIVATELADDR ((ULONG) 0x00400000) /* 4 meg */
|
|
#define MAXDOSLADDR ((ULONG) 0x003fffff)
|
|
#define MINDOSLADDR ((ULONG) 0x00000000)
|
|
|
|
#define MAXSYSTEMPAGE (MAXSYSTEMLADDR >> PAGESHIFT)
|
|
#define MINSYSTEMPAGE (MINSYSTEMLADDR >> PAGESHIFT)
|
|
#define MAXSHAREDPAGE (MAXSHAREDLADDR >> PAGESHIFT)
|
|
#define MINSHAREDPAGE (MINSHAREDLADDR >> PAGESHIFT)
|
|
#define MAXPRIVATEPAGE (MAXPRIVATELADDR >> PAGESHIFT)
|
|
#define MINPRIVATEPAGE (MINPRIVATELADDR >> PAGESHIFT)
|
|
#define MAXDOSPAGE (MAXDOSLADDR >> PAGESHIFT)
|
|
#define MINDOSPAGE (MINDOSLADDR >> PAGESHIFT)
|
|
|
|
#define CBPRIVATE (1 + MAXPRIVATELADDR - MINPRIVATELADDR)
|
|
#define CBSHARED (1 + MAXSHAREDLADDR - MINSHAREDLADDR)
|
|
#define CBSYSTEM (1 + MAXSYSTEMLADDR - MINSYSTEMLADDR)
|
|
#define CBDOS (1 + MAXDOSLADDR - MINDOSLADDR)
|
|
|
|
#define CPGPRIVATE (1 + MAXPRIVATEPAGE - MINPRIVATEPAGE)
|
|
#define CPGSHARED (1 + MAXSHAREDPAGE - MINSHAREDPAGE)
|
|
#define CPGSYSTEM (1 + MAXSYSTEMPAGE - MINSYSTEMPAGE)
|
|
#define CPGDOS (1 + MAXDOSPAGE - MINDOSPAGE)
|
|
|
|
/*XLATOFF*/
|
|
/*
|
|
* Largest object that could theoretically be allocated
|
|
*/
|
|
#define CBMAXALLOC (max(CBSHARED,max(CBPRIVATE, CBSYSTEM)))
|
|
#define CPGMAXALLOC (max(CPGSHARED,max(CPGPRIVATE, CPGSYSTEM)))
|
|
|
|
/*XLATON*/
|
|
|
|
/* ASM
|
|
IFDEF DEBUG
|
|
DebFar EQU NEAR PTR
|
|
ELSE
|
|
DebFar EQU SHORT
|
|
ENDIF
|
|
*/
|
|
|
|
#ifndef Not_VxD
|
|
|
|
/******************************************************************************
|
|
*
|
|
* EQUATES FOR SYSTEM_CONTROL CALLS
|
|
*
|
|
*****************************************************************************/
|
|
|
|
/*
|
|
* SYS_CRITICAL_INIT is a device init call. Devices that have a
|
|
* critical function that needs initializing before interrupts are
|
|
* enabled should do it at Sys_Critical_Init. Devices which REQUIRE a
|
|
* certain range of V86 pages to operate (such as the VDD video memory)
|
|
* should claim them at Sys_Critical_Init. SYS VM Simulate_Int,
|
|
* Exec_Int ACTIVITY IS NOT ALLOWED. Returning carry aborts device
|
|
* load only.
|
|
*/
|
|
#define SYS_CRITICAL_INIT 0x0000 /* Devices req'd for virt mode */
|
|
|
|
/*
|
|
* DEVICE_INIT is where most devices do the bulk of their initialization.
|
|
* SYS VM Simulate_Int, Exec_Int activity is allowed. Returning carry
|
|
* aborts device load only.
|
|
*/
|
|
#define DEVICE_INIT 0x0001 /* All other devices init */
|
|
|
|
/*
|
|
* INIT_COMPLETE is the final phase of device init called just before the
|
|
* WIN386 INIT pages are released and the Instance snapshot is taken.
|
|
* Devices which wish to search for a region of V86 pages >= A0h to use
|
|
* should do it at INIT_COMPLETE.
|
|
* SYS VM Simulate_Int, Exec_Int activity is allowed. Returning carry
|
|
* aborts device load only.
|
|
*/
|
|
#define INIT_COMPLETE 0x0002 /* All devices initialized */
|
|
|
|
/* --------------- INITIALIZATION CODE AND DATA DISCARDED ------------------ */
|
|
|
|
/*
|
|
* Same as VM_Init, except for SYS VM.
|
|
*/
|
|
#define SYS_VM_INIT 0x0003 /* Execute the system VM */
|
|
|
|
/*
|
|
* Same as VM_Terminate, except for SYS VM (Normal WIN386 exit ONLY, on a crash
|
|
* exit this call is not made). SYS VM Simulate_Int, Exec_Int activity is
|
|
* allowed. This and Sys_VM_Terminate2 are your last chances to access
|
|
* and/or lock pageable data.
|
|
*/
|
|
#define SYS_VM_TERMINATE 0x0004 /* System VM terminated */
|
|
|
|
/*
|
|
* System_Exit call is made when WIN386 is exiting either normally or via
|
|
* a crash. INTERRUPTS ARE ENABLED. Instance snapshot has been restored.
|
|
* SYS VM Simulate_Int, Exec_Int ACTIVITY IS NOT ALLOWED.
|
|
*/
|
|
#define SYSTEM_EXIT 0x0005 /* Devices prepare to exit */
|
|
|
|
/*
|
|
* SYS_CRITICAL_EXIT call is made when WIN386 is exiting either normally or via
|
|
* a crash. INTERRUPTS ARE DISABLED. SYS VM Simulate_Int, Exec_Int ACTIVITY
|
|
* IS NOT ALLOWED.
|
|
*/
|
|
#define SYS_CRITICAL_EXIT 0x0006 /* System critical devices reset */
|
|
|
|
|
|
/*
|
|
* Create_VM creates a new VM. EBX = VM handle of new VM. Returning
|
|
* Carry will fail the Create_VM.
|
|
*/
|
|
#define CREATE_VM 0x0007
|
|
|
|
/*
|
|
* Second phase of Create_VM. EBX = VM handle of new VM. Returning
|
|
* Carry will cause the VM to go Not_Executeable, then be destroyed.
|
|
* VM Simulate_Int, Exec_Int activity is NOT allowed.
|
|
*/
|
|
#define VM_CRITICAL_INIT 0x0008
|
|
|
|
/*
|
|
* Third phase of Create_VM. EBX = VM handle of new VM. Returning
|
|
* Carry will cause the VM to go Not_Executeable, then be destroyed.
|
|
* VM Simulate_Int, Exec_Int activity is allowed.
|
|
*/
|
|
#define VM_INIT 0x0009
|
|
|
|
/*
|
|
* NORMAL (First phase) of Destroy_VM. EBX = VM Hanlde. This occurs
|
|
* on normal termination of the VM. Call cannot be failed. VM
|
|
* Simulate_Int, Exec_Int activity is allowed.
|
|
*/
|
|
#define VM_TERMINATE 0x000A /* Still in VM -- About to die */
|
|
|
|
/*
|
|
* Second phase of Destroy_VM. EBX = VM Handle, EDX = Flags (see
|
|
* below). Note that in the case of destroying a running VM, this is
|
|
* the first call made (VM_Terminate call does not occur). Call cannot
|
|
* be failed. VM Simulate_Int, Exec_Int activity is NOT allowed.
|
|
*/
|
|
#define VM_NOT_EXECUTEABLE 0x000B /* Most devices die (except VDD) */
|
|
|
|
/*
|
|
* Final phase of Destroy_VM. EBX = VM Handle. Note that considerable
|
|
* time can elaps between the VM_Not_Executeable call and this call.
|
|
* Call cannot be failed. VM Simulate_Int, Exec_Int activity is NOT
|
|
* allowed.
|
|
*/
|
|
#define DESTROY_VM 0x000C /* VM's control block about to go */
|
|
|
|
|
|
/*
|
|
* Flags for VM_Not_Executeable control call (passed in EDX)
|
|
*/
|
|
#define VNE_CRASHED_BIT 0x00 /* VM was crashed */
|
|
#define VNE_CRASHED (1 << VNE_CRASHED_BIT)
|
|
#define VNE_NUKED_BIT 0x01 /* VM was destroyed while active */
|
|
#define VNE_NUKED (1 << VNE_NUKED_BIT)
|
|
#define VNE_CREATEFAIL_BIT 0x02 /* Some device failed Create_VM */
|
|
#define VNE_CREATEFAIL (1 << VNE_CREATEFAIL_BIT)
|
|
#define VNE_CRINITFAIL_BIT 0x03 /* Some device failed VM_Critical_Init */
|
|
#define VNE_CRINITFAIL (1 << VNE_CRINITFAIL_BIT)
|
|
#define VNE_INITFAIL_BIT 0x04 /* Some device failed VM_Init */
|
|
#define VNE_INITFAIL (1 << VNE_INITFAIL_BIT)
|
|
#define VNE_CLOSED_BIT 0x05
|
|
#define VNE_CLOSED (1 << VNE_CLOSED_BIT)
|
|
|
|
|
|
/*
|
|
* EBX = VM Handle. Call cannot be failed.
|
|
*/
|
|
#define VM_SUSPEND 0x000D /* VM not runnable until resume */
|
|
|
|
/*
|
|
* EBX = VM Handle. Returning carry fails and backs out the resume.
|
|
*/
|
|
#define VM_RESUME 0x000E /* VM is leaving suspended state */
|
|
|
|
|
|
/*
|
|
* EBX = VM Handle to set device focus to. EDX = Device ID if device
|
|
* specific setfocus, == 0 if device critical setfocus (all devices).
|
|
* THIS CALL CANNOT BE FAILED.
|
|
*
|
|
* NOTE: In case where EDX == 0, ESI is a FLAG word that indicates
|
|
* special functions. Currently Bit 0 being set indicates that this
|
|
* Device critical set focus is also "VM critical". It means that we
|
|
* do not want some other VM to take the focus from this app now. This
|
|
* is primarily used when doing a device critical set focus to Windows
|
|
* (the SYS VM) it is interpreted by the SHELL to mean "if an old app
|
|
* currently has the Windows activation, set the activation to the
|
|
* Windows Shell, not back to the old app". ALSO in the case where Bit
|
|
* 0 is set, EDI = The VM handle of the VM that is "having trouble".
|
|
* Set this to 0 if there is no specific VM associated with the
|
|
* problem.
|
|
*/
|
|
#define SET_DEVICE_FOCUS 0x000F
|
|
|
|
|
|
/*
|
|
* EBX = VM Handle going into message mode. THIS CALL CANNOT BE FAILED.
|
|
*/
|
|
#define BEGIN_MESSAGE_MODE 0x0010
|
|
|
|
/*
|
|
* EBX = VM Handle leaving message mode. THIS CALL CANNOT BE FAILED.
|
|
*/
|
|
#define END_MESSAGE_MODE 0x0011
|
|
|
|
|
|
/* ----------------------- SPECIAL CONTROL CALLS --------------------------- */
|
|
|
|
/*
|
|
* Request for reboot. Call cannot be failed.
|
|
*/
|
|
#define REBOOT_PROCESSOR 0x0012 /* Request a machine reboot */
|
|
|
|
/*
|
|
* Query_Destroy is an information call made by the SHELL device before
|
|
* an attempt is made to initiate a destroy VM sequence on a running VM
|
|
* which has not exited normally. EBX = VM Handle. Returning carry
|
|
* indicates that a device "has a problem" with allowing this. THE
|
|
* DESTROY SEQUENCE CANNOT BE ABORTED HOWEVER, this decision is up to
|
|
* the user. All this does is indicate that there is a "problem" with
|
|
* allowing the destroy. The device which returns carry should call
|
|
* the SHELL_Message service to post an informational dialog about the
|
|
* reason for the problem.
|
|
*/
|
|
#define QUERY_DESTROY 0x0013 /* OK to destroy running VM? */
|
|
|
|
|
|
/* ----------------------- DEBUGGING CONTROL CALL -------------------------- */
|
|
|
|
/*
|
|
* Special call for device specific DEBUG information display and activity.
|
|
*/
|
|
#define DEBUG_QUERY 0x0014
|
|
|
|
|
|
/* -------- CALLS FOR BEGIN/END OF PROTECTED MODE VM EXECUTION ------------- */
|
|
|
|
/*
|
|
* About to run a protected mode application.
|
|
* EBX = Current VM handle.
|
|
* EDX = Flags
|
|
* EDI -> Application Control Block
|
|
* Returning with carry set fails the call.
|
|
*/
|
|
#define BEGIN_PM_APP 0x0015
|
|
|
|
/*
|
|
* Flags for Begin_PM_App (passed in EDX)
|
|
*/
|
|
#define BPA_32_BIT 0x01
|
|
#define BPA_32_BIT_FLAG 1
|
|
|
|
/*
|
|
* Protected mode application is terminating.
|
|
* EBX = Current VM handle. THIS CALL CAN NOT FAIL.
|
|
* EDI -> Application Control Block
|
|
*/
|
|
#define END_PM_APP 0x0016
|
|
|
|
/*
|
|
* Called whenever system is about to be rebooted. Allows VxDs to clean
|
|
* up in preperation for reboot.
|
|
*/
|
|
#define DEVICE_REBOOT_NOTIFY 0x0017
|
|
#define CRIT_REBOOT_NOTIFY 0x0018
|
|
|
|
/*
|
|
* Called when VM is about to be terminated using the Close_VM service
|
|
* EBX = Current VM handle (Handle of VM to close)
|
|
* EDX = Flags
|
|
* CVNF_CRIT_CLOSE = 1 if VM is in critical section while closing
|
|
*/
|
|
#define CLOSE_VM_NOTIFY 0x0019
|
|
|
|
#define CVNF_CRIT_CLOSE_BIT 0
|
|
#define CVNF_CRIT_CLOSE (1 << CVNF_CRIT_CLOSE_BIT)
|
|
|
|
/*
|
|
* Power management event notification.
|
|
* EBX = 0
|
|
* ESI = event notification message
|
|
* EDI -> DWORD return value; VxD's modify the DWORD to return info, not EDI
|
|
* EDX is reserved
|
|
*/
|
|
#define POWER_EVENT 0x001A
|
|
|
|
#define SYS_DYNAMIC_DEVICE_INIT 0x001B
|
|
#define SYS_DYNAMIC_DEVICE_EXIT 0x001C
|
|
|
|
/*
|
|
* Create_THREAD creates a new thread. EDI = handle of new thread.
|
|
* Returning Carry will fail the Create_THREAD. Message is sent in the
|
|
* context of the creating thread.
|
|
*
|
|
*/
|
|
#define CREATE_THREAD 0x001D
|
|
|
|
/*
|
|
* Second phase of creating a thread. EDI = handle of new thread. Call cannot
|
|
* be failed. VM Simulate_Int, Exec_Int activity is not allowed (because
|
|
* never allowed in non-initial threads). Message is sent in the context
|
|
* of the newly created thread.
|
|
*
|
|
*/
|
|
#define THREAD_INIT 0x001E
|
|
|
|
/*
|
|
* Normal (first) phase of Destroy_THREAD. EDI = handle of thread.
|
|
* This occurs on normal termination of the thread. Call cannot be failed.
|
|
* Simulate_Int, Exec_Int activity is allowed.
|
|
*/
|
|
#define TERMINATE_THREAD 0x001F
|
|
|
|
/*
|
|
* Second phase of Destroy_THREAD. EDI = Handle of thread,
|
|
* EDX = flags (see below). Note that in the case of destroying a
|
|
* running thread, this is the first call made (THREAD_Terminate call
|
|
* does not occur). Call cannot be failed. VM Simulate_Int, Exec_Int
|
|
* activity is NOT allowed.
|
|
*
|
|
*/
|
|
#define THREAD_Not_Executeable 0x0020
|
|
|
|
/*
|
|
* Final phase of Destroy_THREAD. EDI = Thread Handle. Note that considerable
|
|
* time can elapse between the THREAD_Not_Executeable call and this call.
|
|
* Call cannot be failed. VM Simulate_Int, Exec_Int activity is NOT
|
|
* allowed.
|
|
*
|
|
*/
|
|
#define DESTROY_THREAD 0x0021
|
|
|
|
/* -------------------- CALLS FOR PLUG&PLAY ------------------------- */
|
|
|
|
/*
|
|
* Configuration manager or a devloader is telling a DLVxD that a new devnode
|
|
* has been created. EBX is the handle of the new devnode and EDX is the load
|
|
* type (one of the DLVxD_LOAD_* defined in CONFIGMG.H). This is a 'C'
|
|
* system control call. Contrarily to the other calls, carry flags must be
|
|
* set if any error code other than CR_SUCCESS is to be return.
|
|
*
|
|
*/
|
|
#define PNP_NEW_DEVNODE 0x0022
|
|
|
|
|
|
/* -------------------- CALLS FOR Win32 ------------------------- */
|
|
|
|
/* vWin32 communicates with Vxds on behalf of Win32 apps thru this mechanism.
|
|
* FEATURE: need more doc here, describing the interface
|
|
*/
|
|
|
|
#define W32_DEVICEIOCONTROL 0x0023
|
|
|
|
/* sub-functions */
|
|
#define DIOC_GETVERSION 0x0
|
|
#define DIOC_OPEN DIOC_GETVERSION
|
|
#define DIOC_CLOSEHANDLE -1
|
|
|
|
/* -------------------- MORE SYSTEM CALLS ------------------------- */
|
|
|
|
/*
|
|
* All these messages are sent immediately following the corresponding
|
|
* message of the same name, except that the "2" messages are sent
|
|
* in *reverse* init order.
|
|
*/
|
|
|
|
#define SYS_VM_TERMINATE2 0x0024
|
|
#define SYSTEM_EXIT2 0x0025
|
|
#define SYS_CRITICAL_EXIT2 0x0026
|
|
#define VM_TERMINATE2 0x0027
|
|
#define VM_NOT_EXECUTEABLE2 0x0028
|
|
#define DESTROY_VM2 0x0029
|
|
#define VM_SUSPEND2 0x002A
|
|
#define END_MESSAGE_MODE2 0x002B
|
|
#define END_PM_APP2 0x002C
|
|
#define DEVICE_REBOOT_NOTIFY2 0x002D
|
|
#define CRIT_REBOOT_NOTIFY2 0x002E
|
|
#define CLOSE_VM_NOTIFY2 0x002F
|
|
|
|
/*
|
|
* VCOMM gets Address of Contention handler from VxDs by sending this
|
|
* control message
|
|
*/
|
|
|
|
#define GET_CONTENTION_HANDLER 0x0030
|
|
|
|
#define KERNEL32_INITIALIZED 0x0031
|
|
|
|
#define KERNEL32_SHUTDOWN 0x0032
|
|
|
|
#define CREATE_PROCESS 0x0033
|
|
#define DESTROY_PROCESS 0x0034
|
|
|
|
#define MAX_SYSTEM_CONTROL 0x0034
|
|
|
|
/*
|
|
* Dynamic VxD's can communicate with each other using Directed_Sys_Control
|
|
* and a private control message in the following range:
|
|
*/
|
|
|
|
#define BEGIN_RESERVED_PRIVATE_SYSTEM_CONTROL 0x70000000
|
|
#define END_RESERVED_PRIVATE_SYSTEM_CONTROL 0x7FFFFFFF
|
|
|
|
#endif // Not_VxD
|
|
|
|
/*
|
|
* Values returned from VMM_GetSystemInitState in EAX.
|
|
*
|
|
* Comments represent operations performed by VMM; #define's indicate
|
|
* what VMM_GetSystemInitState will return if you call it between the
|
|
* previous operation and the next.
|
|
*
|
|
* Future versions of Windows may have additional init states between the
|
|
* ones defined here, so you should be careful to use range checks instead
|
|
* of test for equality.
|
|
*/
|
|
|
|
/* Protected mode is entered */
|
|
#define SYSSTATE_PRESYSCRITINIT 0x00000000
|
|
/* SYS_CRITICAL_INIT is broadcast */
|
|
#define SYSSTATE_PREDEVICEINIT 0x10000000
|
|
/* DEVICE_INIT is broadcast */
|
|
#define SYSSTATE_PREINITCOMPLETE 0x20000000
|
|
/* INIT_COMPLETE is broadcast */
|
|
/* VxD initialization complete */
|
|
#define SYSSTATE_VXDINITCOMPLETED 0x40000000
|
|
/* KERNEL32_INITIALIZED is broadcast */
|
|
#define SYSSTATE_KERNEL32INITED 0x50000000
|
|
/* All initialization completed */
|
|
/* System running normally */
|
|
/* System shutdown initiated */
|
|
/* KERNEL32_SHUTDOWN is broadcast */
|
|
#define SYSSTATE_KERNEL32TERMINATED 0xA0000000
|
|
/* System shutdown continues */
|
|
#define SYSSTATE_PRESYSVMTERMINATE 0xB0000000
|
|
/* SYS_VM_TERMINATE is broadcast */
|
|
#define SYSSTATE_PRESYSTEMEXIT 0xE0000000
|
|
/* SYSTEM_EXIT is broadcast */
|
|
#define SYSSTATE_PRESYSTEMEXIT2 0xE4000000
|
|
/* SYSTEM_EXIT2 is broadcast */
|
|
#define SYSSTATE_PRESYSCRITEXIT 0xF0000000
|
|
/* SYS_CRITICAL_EXIT is broadcast */
|
|
#define SYSSTATE_PRESYSCRITEXIT2 0xF4000000
|
|
/* SYS_CRITICAL_EXIT2 is broadcast */
|
|
#define SYSSTATE_POSTSYSCRITEXIT2 0xFFF00000
|
|
/* Return to real mode */
|
|
/* Alternate path: CAD reboot */
|
|
#define SYSSTATE_PREDEVICEREBOOT 0xFFFF0000
|
|
/* DEVICE_REBOOT_NOTIFY is broadcast */
|
|
#define SYSSTATE_PRECRITREBOOT 0xFFFFF000
|
|
/* CRIT_REBOOT_NOTIFY is broadcast */
|
|
#define SYSSTATE_PREREBOOTCPU 0xFFFFFF00
|
|
/* REBOOT_PROCESSOR is broadcast */
|
|
/* Return to real mode */
|
|
|
|
/* ASM
|
|
BeginDoc
|
|
;******************************************************************************
|
|
; BeginProc is a macro for defining entry points to routines in VMM and in the
|
|
; VxDs. It correctly defines the procedure name for VxD services, DWORD
|
|
; aligns the procedure, takes care of public declaration and does some
|
|
; calling verification for debug versions of the software. EndProc is a
|
|
; macro which defines the end of the procedure.
|
|
;
|
|
; Valid parameters to the BeginProc macro are:
|
|
; PUBLIC ; Used outside this module (default)
|
|
; LOCAL ; Local to this module
|
|
; HIGH_FREQ ; DWORD align procedure
|
|
; SERVICE ; Routine is called via VxDCall
|
|
; ASYNC_SERVICE ; Same as "SERVICE" plus routine can
|
|
; ; be called under interrupt.
|
|
; HOOK_PROC ; Proc is a handler installed with
|
|
; ; with a call to Hook_xxx_Fault
|
|
; ; or Hook_Device_Service. The
|
|
; ; following parameter must be
|
|
; ; the label of a DWORD location
|
|
; ; which will hold the ptr to next
|
|
; ; hook proc. e.g.
|
|
;
|
|
; ;BeginProc foo, SERVICE, HOOK_PROC, foo_next_ptr
|
|
;
|
|
; NO_LOG ; Disable Queue_Out call logging
|
|
; NO_PROFILE ; Disable DynaLink profile counts
|
|
; NO_TEST_CLD ; Disable direction flag check
|
|
;
|
|
; TEST_BLOCK ; Trap if in NOBLOCK state
|
|
; ; (default if in pageable code seg)
|
|
; TEST_REENTER ; Trap if Get_VMM_Reenter_Count != 0
|
|
; ; (default for non-async services)
|
|
; NEVER_REENTER ; Trap if VMM has been reentered
|
|
; NOT_SWAPPING ; Trap if this thread is swapping
|
|
;
|
|
; NO_PROLOG ; Disable all prolog tests
|
|
;
|
|
; ESP ; Use ESP instead of EBP for stack
|
|
; ; frame base
|
|
; PCALL ; pascal calling convention
|
|
; SCALL ; stdcall calling convention
|
|
; FASTCALL ; stdcall, but first 2 parameters are passed in ECX & EDX
|
|
; CCALL ; "C" calling convention
|
|
; ICALL ; default calling convention
|
|
; W32SVC ; Win32 service
|
|
;
|
|
; segment type ; Place function in specified segment
|
|
;
|
|
; The NO_PROFILE flag merely suppresses incrementing the profile count.
|
|
; The DWORD of profiling information will still be emitted to appease
|
|
; the debugger. If you want to increment the profile count manually,
|
|
; use the IncProfileCount macro.
|
|
;
|
|
; TEST_REENTER and NEVER_REENTER differ in that the VMM reentry count
|
|
; returned by Get_VMM_Reenter_Count is artifically forced to zero by
|
|
; Begin_Reentrant_Execution, whereas the counter checked by NEVER_REENTER
|
|
; reflects the genuine count of VMM reentry.
|
|
;
|
|
; A segment type (such as LOCKED, PAGEABLE, STATIC, INIT, DEBUG_ONLY) can be
|
|
; provided, in which case the BeginProc and EndProc macros will
|
|
; automatically place the appropriate segment directives around the
|
|
; definition of the function.
|
|
;
|
|
; segment type ; Place function in specified segment
|
|
;
|
|
; After the routine header in which the routine entry conditions, exit
|
|
; conditions, side affects and functionality are specified, the BeginProc
|
|
; macro should be used to define the routine's entry point. It has up to
|
|
; four parameters as specified below. For example:
|
|
;
|
|
;BeginProc <Function_Name>,PUBLIC, HIGH_FREQ, SERVICE, ASYNC_SERVICE, ESP
|
|
;
|
|
; <code>
|
|
;
|
|
;EndProc <Function_Name>
|
|
;==============================================================================
|
|
EndDoc
|
|
;
|
|
; BeginProc handling takes place in the following phases:
|
|
;
|
|
; Phase 1: Parsing the arguments.
|
|
; Phase 2: Setting default flags.
|
|
; Phase 3: Combining the flags.
|
|
; Phase 4: Code emitted before the label
|
|
; Phase 5: Munge the name as exported to C/Pascal/whoever
|
|
; Phase 6: _Debug_Flags_Service prolog
|
|
; Phase 7: Code emitted after the label
|
|
;
|
|
|
|
??_pf_Check equ 1 ;; Do Enter/LeaveProc checking?
|
|
??_pf_ArgsUsed equ 2 ;; ArgVars were used
|
|
??_pf_Entered equ 4 ;; EnterProc performed
|
|
??_pf_Left equ 8 ;; LeaveProc performed
|
|
??_pf_Returned equ 16 ;; Return performed
|
|
|
|
??_pushed = 0 ;; For WIN31COMPAT
|
|
??_align = 0 ;; For WIN31COMPAT
|
|
??_ends equ <> ;; BeginProc segment
|
|
|
|
BeginProc macro Name, P1, P2, P3, P4, P5, P6, P7, LastArg
|
|
local Profile_Data, prelabeldata, ??_hookvar
|
|
??_frame = 0 ;; local frame base
|
|
??_aframe = 0 ;; argument frame base
|
|
??_taframe = 0 ;; true argument frame base
|
|
??_initaframe = 0 ;; initial aframe value
|
|
??_numargs = 0 ;; number of argvars
|
|
??_numlocals = 0 ;; number of localvars
|
|
??_numlocalsymbols = 0 ;; number of local symbols
|
|
??_procflags = 0 ;; misc. Enter/LeaveProc flags
|
|
??_esp = 0 ;; if VMM_TRUE, use esp instead of ebp
|
|
??_pushed = 0 ;; number of bytes pushed
|
|
??_align = 0 ;; set if proc should be dword aligned
|
|
??_hook = 0 ;; set if proc is a Hook_Proc
|
|
??_hookarg = 0
|
|
??_service = 0
|
|
??_async_service = 0
|
|
IF DEBLEVEL GT DEBLEVELNORMAL
|
|
??_log = DFS_LOG ;; logging on by default
|
|
??_profile = DFS_PROFILE ;; service profiling on by default
|
|
??_test_cld = DFS_TEST_CLD ;; test that direction is clear
|
|
ELSE
|
|
??_log = 0 ;; logging off
|
|
IFDEF DEBUG
|
|
IFDEF profileall
|
|
IF ?_ICODE
|
|
??_profile = DFS_PROFILE ;; service profiling on by default
|
|
ELSE
|
|
??_profile = 0 ;; service profiling off
|
|
ENDIF
|
|
ELSE
|
|
??_profile = 0 ;; service profiling off
|
|
ENDIF
|
|
ELSE
|
|
??_profile = 0 ;; service profiling off
|
|
ENDIF
|
|
??_test_cld = 0 ;; test that direction is clear
|
|
ENDIF
|
|
??_might_block = 0 ;; entering fn might cause VM to block
|
|
??_test_reenter = 0 ;; don't test for VMM reentry
|
|
??_never_reenter = 0 ;; don't test for genuine VMM reentry
|
|
??_not_swapping = 0 ;; don't test that we're not swapping
|
|
??_prolog_disabled = 0 ;; use a prolog by default
|
|
??_public = 1 ;; everything's public by default
|
|
??_cleanoff = 0 ;; don't cleanoff parameters
|
|
??_ccall = 0
|
|
??_pcall = 0
|
|
??_scall = 0
|
|
??_fastcall = 0
|
|
??_w32svc = 0
|
|
??_fleave = FALSE
|
|
; ??_dfs = 0 ;; parm for _Debug_Flags_Service
|
|
??_name equ <Name>
|
|
|
|
.errnb ??_ends, <Cannot nest functions with named segments>
|
|
.errnb <LastArg>, <Too many arguments to BeginProc>
|
|
|
|
;; Phase 1: Parsing the arguments
|
|
irp arg, <P1, P2, P3, P4, P5, P6, P7>
|
|
if ??_hookarg
|
|
??_hookarg = 0
|
|
??_hookvar equ <arg>
|
|
elseifdef ?&&arg&&_BeginProc
|
|
?&&arg&&_BeginProc
|
|
elseifdef VxD_&&arg&&_CODE_SEG
|
|
??_ends textequ <VxD_&&arg&&_CODE_ENDS>
|
|
VxD_&&arg&&_CODE_SEG
|
|
else
|
|
.err <Bad param "&arg" to BeginProc>
|
|
endif
|
|
endm
|
|
|
|
;; Phase 2: Setting default flags
|
|
ifndef Not_VxD
|
|
ife ??_service
|
|
ifndef profileall
|
|
??_profile = 0 ;; only services can be profiled
|
|
endif
|
|
ifdef VMMSYS
|
|
??_prolog_disabled = 1
|
|
endif
|
|
else
|
|
??_test_cld = DFS_TEST_CLD
|
|
endif ; ife ??_service
|
|
|
|
ife ?_16ICODE
|
|
??_prolog_disabled = 1
|
|
else
|
|
ife ?_RCODE ;; if real-mode code segment
|
|
??_prolog_disabled = 1 ;; don't do anything bad
|
|
else ;; else protected mode code segment
|
|
ife ?_PCODE ;; if swappable code
|
|
??_might_block = DFS_TEST_BLOCK
|
|
endif
|
|
if ??_service
|
|
ife ??_async_service
|
|
??_test_reenter = DFS_TEST_REENTER
|
|
endif
|
|
endif
|
|
endif ; ife ?_RCODE
|
|
endif ; ife ?_16ICODE
|
|
endif ; Not_VxD
|
|
|
|
if ??_esp
|
|
;; just return address on stack
|
|
??_basereg equ <esp+??_pushed>
|
|
??_initaframe = 4
|
|
else
|
|
;; ret addr and EBP on stack
|
|
??_basereg equ <ebp>
|
|
??_initaframe = 8
|
|
endif
|
|
@Caller equ <dword ptr [??_basereg+??_initaframe-4]>
|
|
|
|
??_cleanoff = ??_pcall or ??_scall or ??_fastcall
|
|
|
|
;; Phase 3: Combining the flags
|
|
??_dfs = ??_never_reenter + ??_test_reenter + ??_not_swapping + \
|
|
??_log + ??_profile + ??_test_cld + ??_might_block
|
|
|
|
if ??_prolog_disabled
|
|
??_dfs = 0
|
|
endif
|
|
|
|
;; Phase 4: Pre-label code
|
|
|
|
ifndef Not_VxD
|
|
|
|
if ??_hook
|
|
if ??_align
|
|
Dword_Align
|
|
endif
|
|
prelabeldata:
|
|
ifndef ??_hookvar
|
|
.err <HOOK_PROC requires next arg to be name of dword location>
|
|
endif
|
|
jmp short Name
|
|
jmp [??_hookvar]
|
|
ifdef DEBUG
|
|
Profile_Data dd 0
|
|
endif
|
|
if ??_align
|
|
.errnz ($ - prelabeldata) mod 4
|
|
endif
|
|
endif
|
|
|
|
ifdef DEBUG
|
|
?prolog_&Name label near
|
|
if (??_service OR ??_profile) AND (??_hook EQ 0)
|
|
jmp short Name
|
|
if ??_align
|
|
Dword_Align ; This also aligns the proc
|
|
endif ; since Profile_Data is a dd
|
|
|
|
IF ?_ICODE
|
|
ifdef profileall
|
|
?ProfileHeader_BeginProc Profile_Data, %@filename
|
|
else
|
|
Profile_Data dd 0
|
|
endif
|
|
ELSE
|
|
Profile_Data dd 0
|
|
ENDIF
|
|
|
|
endif
|
|
endif
|
|
|
|
if ??_align
|
|
Dword_Align
|
|
endif
|
|
|
|
endif ; Not_VxD
|
|
|
|
Name proc near ;; The label
|
|
|
|
;; Phase 5: Munge the name as exported to C/Pascal/whoever
|
|
;; Warning! Phase 5 cannot emit code!
|
|
ife ??_pcall or ??_ccall or ??_scall or ??_fastcall ;; if no munging
|
|
if ??_public
|
|
public Name
|
|
else
|
|
ifdef DEBUG
|
|
% ?merge @FileName,$,Name,:
|
|
% ?merge public,,,,,@FileName,$,Name
|
|
endif
|
|
endif
|
|
endif
|
|
if ??_ccall
|
|
if ??_public
|
|
_&Name equ Name
|
|
ifdef Not_VxD
|
|
public C Name
|
|
else
|
|
public _&Name
|
|
endif
|
|
endif
|
|
endif
|
|
if ??_pcall
|
|
if ??_public
|
|
?toupper Name
|
|
?merge public,,,,%?upper
|
|
endif
|
|
endif
|
|
;; Phase 6: _Debug_Flags_Service prolog
|
|
;; DO NOT CHANGE UNTIL YOU UNDERSTAND _Debug_Flags_Service
|
|
|
|
ife ??_scall or ??_fastcall
|
|
?_BeginProc_Debug_Prologue
|
|
endif
|
|
|
|
;; Phase 7: Post-label code
|
|
;; <none>
|
|
endm
|
|
|
|
?_BeginProc_Debug_Prologue MACRO
|
|
ifndef Not_VxD
|
|
ifdef DEBUG
|
|
if ??_dfs EQ DFS_LOG
|
|
VMMCall Log_Proc_Call ;; no test, just log
|
|
else
|
|
if ??_dfs EQ DFS_TEST_REENTER
|
|
VMMCall Test_Reenter ;; no log, just reenter
|
|
else
|
|
if ??_dfs or ?_LOCKABLECODE eq 0
|
|
ifdef WIN31COMPAT
|
|
if ??_dfs AND DFS_LOG
|
|
VMMCall Log_Proc_Call
|
|
endif
|
|
if ??_dfs AND DFS_TEST_REENTER
|
|
VMMCall Test_Reenter
|
|
endif
|
|
else
|
|
ife ?_LOCKABLECODE
|
|
ifdef ??_debug_flags
|
|
push ??_debug_flags
|
|
if ??_dfs
|
|
pushfd
|
|
or dword ptr [esp+4],??_dfs
|
|
popfd
|
|
endif
|
|
VMMCall _Debug_Flags_Service
|
|
elseif ??_dfs
|
|
push ??_dfs
|
|
VMMCall _Debug_Flags_Service
|
|
endif
|
|
else
|
|
push ??_dfs
|
|
VMMCall _Debug_Flags_Service
|
|
endif
|
|
endif
|
|
else
|
|
ifdef profileall
|
|
IncProfileCount
|
|
endif
|
|
endif ;if ??_dfs
|
|
endif ; if ??_dfs EQ DFS_TEST_REENTER
|
|
endif ; if ??_dfs EQ DFS_LOG
|
|
endif ; DEBUG
|
|
endif ; Not_VxD
|
|
ENDM
|
|
|
|
;
|
|
; For each BeginProc keyword, there is a corresponding macro ?XX_BeginProc.
|
|
;
|
|
; The macro ?_BeginProc is so that the null keyword is not an error.
|
|
|
|
?_BeginProc macro
|
|
endm
|
|
|
|
?PUBLIC_BeginProc macro
|
|
??_public = 1
|
|
endm
|
|
|
|
?LOCAL_BeginProc macro
|
|
??_public = 0
|
|
endm
|
|
|
|
?HIGH_FREQ_BeginProc macro
|
|
??_align = 1
|
|
endm
|
|
|
|
?HOOK_PROC_BeginProc macro
|
|
??_hook = 1
|
|
??_hookarg = 1 ; next arg is dword storage location
|
|
endm
|
|
|
|
?SERVICE_BeginProc macro
|
|
??_service = 1
|
|
.erre ?_16ICODE, <SERVICEs must be in 32 bit code>
|
|
.erre ?_RCODE, <SERVICEs must be in 32 bit code>
|
|
endm
|
|
|
|
?ASYNC_SERVICE_BeginProc macro
|
|
??_service = 1
|
|
??_async_service = 1
|
|
.errnz ?_LCODE, <ASYNC_SERVICE's must be in LOCKED code>
|
|
endm
|
|
|
|
?NO_LOG_BeginProc macro
|
|
??_log = 0
|
|
endm
|
|
|
|
?NO_PROFILE_BeginProc macro
|
|
??_profile = 0
|
|
endm
|
|
|
|
?NO_TEST_CLD_BeginProc macro
|
|
??_test_cld = 0
|
|
endm
|
|
|
|
?TEST_BLOCK_BeginProc macro
|
|
??_might_block = DFS_TEST_BLOCK
|
|
endm
|
|
|
|
?TEST_REENTER_BeginProc macro
|
|
??_test_reenter = DFS_TEST_REENTER
|
|
endm
|
|
|
|
?NEVER_REENTER_BeginProc macro
|
|
??_never_reenter = DFS_NEVER_REENTER
|
|
endm
|
|
|
|
?NOT_SWAPPING_BeginProc macro
|
|
??_not_swapping = DFS_NOT_SWAPPING
|
|
endm
|
|
|
|
?NO_PROLOG_BeginProc macro
|
|
??_prolog_disabled = 1
|
|
endm
|
|
|
|
?ESP_BeginProc macro
|
|
??_esp = VMM_TRUE
|
|
ifndef Not_VxD
|
|
.erre ?_16ICODE, <Beginproc ESP attribute invalid in 16 bit seg.>
|
|
.erre ?_RCODE, <Beginproc ESP attribute invalid in real-mode seg.>
|
|
endif
|
|
endm
|
|
|
|
?CCALL_BeginProc macro
|
|
??_ccall = 1
|
|
endm
|
|
|
|
?PCALL_BeginProc macro
|
|
??_pcall = 1
|
|
endm
|
|
|
|
?SCALL_BeginProc macro
|
|
??_scall = 1
|
|
endm
|
|
|
|
?FASTCALL_BeginProc macro
|
|
??_fastcall = 1
|
|
endm
|
|
|
|
?ICALL_BeginProc macro
|
|
??_scall = 1 ;; internal calling convention is StdCall
|
|
endm
|
|
|
|
?W32SVC_BeginProc macro
|
|
??_scall = 1
|
|
??_w32svc = 1
|
|
endm
|
|
|
|
ifdef DEBUG
|
|
ifdef profileall
|
|
?ProfileHeader_BeginProc macro PL, filename
|
|
ifndef _&filename&__proc_list
|
|
_&filename&__proc_list = 0
|
|
PUBLIC _&filename&__proc_list
|
|
endif
|
|
dd OFFSET32 _&filename&__proc_list
|
|
PL dd 0
|
|
_&filename&__proc_list = PL
|
|
endm
|
|
endif
|
|
|
|
IncProfileCount macro
|
|
if ??_service OR ??_profile
|
|
inc dword ptr [??_name-4]
|
|
else
|
|
ifndef profileall
|
|
.err <IncProfileCount can be used only in services.>
|
|
endif
|
|
endif
|
|
endm
|
|
else
|
|
IncProfileCount macro
|
|
endm
|
|
endif
|
|
|
|
;*** ArgVar - declares stack arguments
|
|
;
|
|
; Usage:
|
|
;
|
|
; name = name of argument.
|
|
; length = a numeric expression denoting the size (in bytes)
|
|
; of the argument. The symbols BYTE, WORD, and DWORD
|
|
; are synonyms for 1, 2, and 4 respectively.
|
|
; NB! All arguments sizes are rounded up to the nearest
|
|
; multiple of 4.
|
|
; used = usually blank, but can be the symbol NOTUSED
|
|
; to indicate that the argument will not be used
|
|
; by the procedure.
|
|
;
|
|
|
|
ArgVar macro name,length,used
|
|
??_numargs = ??_numargs + 1
|
|
if ??_pcall
|
|
?mkarg <name>, <length>, <used>, %??_numargs
|
|
else
|
|
?arg <name>, <length>, <used>
|
|
endif
|
|
??_procflags = ??_procflags OR ??_pf_Check
|
|
endm
|
|
|
|
?mkarg macro name, length, used, num
|
|
.xcref ?MKA&num
|
|
?deflocal <name>
|
|
?MKA&num ¯o
|
|
?argvar <name>, <length>, <used>
|
|
&endm
|
|
??_aframe = ??_aframe + 4
|
|
endm
|
|
.xcref ?mkarg
|
|
|
|
?argvar macro name,length,used
|
|
local a
|
|
a = ??_taframe
|
|
??_aframe = ??_aframe + 4
|
|
??_taframe = ??_taframe + 4
|
|
ifidni <length>,<BYTE>
|
|
?setname <name>, <byte ptr [??_basereg+??_initaframe+a]>, <used>
|
|
elseifidni <length>,<WORD>
|
|
?setname <name>, <word ptr [??_basereg+??_initaframe+a]>, <used>
|
|
elseifidni <length>,<DWORD>
|
|
?setname <name>, <dword ptr [??_basereg+??_initaframe+a]>, <used>
|
|
?setname <name&l>,<word ptr [??_basereg+??_initaframe+a]>, <used>
|
|
?setname <name&ll>,<byte ptr [??_basereg+??_initaframe+a]>, <used>
|
|
?setname <name&lh>,<byte ptr [??_basereg+??_initaframe+a+1]>, <used>
|
|
?setname <name&h>,<word ptr [??_basereg+??_initaframe+a+2]>, <used>
|
|
?setname <name&hl>,<byte ptr [??_basereg+??_initaframe+a+2]>, <used>
|
|
?setname <name&hh>,<byte ptr [??_basereg+??_initaframe+a+3]>, <used>
|
|
else
|
|
??_aframe = ??_aframe - 4 + ((length + 3)/4)*4
|
|
??_taframe = ??_taframe - 4 + ((length + 3)/4)*4
|
|
?setname <name>, <[??_basereg+??_initaframe+a]>, <used>
|
|
endif
|
|
endm
|
|
|
|
?arg macro name,length,used
|
|
if ??_fastcall
|
|
if ??_numargs le 2
|
|
if length gt 4
|
|
.err <First 2 parameters are dwords (ecx,edx) for fastcall functions>
|
|
endif
|
|
??_aframe = ??_aframe + 4
|
|
if ??_numargs eq 1
|
|
?merge ecx_,name,,,equ,ecx
|
|
else
|
|
?merge edx_,name,,,equ,edx
|
|
endif
|
|
else
|
|
?argvar name, length, used
|
|
endif
|
|
else
|
|
?argvar name, length, used
|
|
endif
|
|
endm
|
|
|
|
;*** ?setname - optionally creates the name of an ArgVar
|
|
;
|
|
; If <used> is <NOTUSED>, then the name is defined to something
|
|
; bogus.
|
|
|
|
?setname macro name, value, used
|
|
?deflocal <name>
|
|
ifidni <used>, <NOTUSED>
|
|
name equ _inaccessible_NOTUSED_
|
|
else
|
|
name equ value
|
|
??_procflags = ??_procflags OR ??_pf_ArgsUsed OR ??_pf_Check
|
|
endif
|
|
endm
|
|
|
|
|
|
;*** LocalVar - declares local stack variables
|
|
;
|
|
; Usage:
|
|
;
|
|
; name = name of local variable
|
|
; length = a numeric expression denoting the size (in bytes)
|
|
; of the argument. The symbols BYTE, WORD, and DWORD
|
|
; are synonyms for 1, 2, and 4 respectively.
|
|
; NB! All arguments sizes are rounded up to the nearest
|
|
; multiple of 4 (unless PACK is indicated)
|
|
; flag = usually blank, but can be the symbol PACK
|
|
; to suppress the usual padding and aligning of variables
|
|
; PACK is typically used when declaring a bunch of
|
|
; byte or word variables. Make sure that the total
|
|
; size of PACKed variables is a multiple of 4.
|
|
;
|
|
|
|
LocalVar macro name,length,flag
|
|
local a
|
|
??_numlocals = ??_numlocals + 1
|
|
??_pad = 1
|
|
ifidni <flag>, <PACK>
|
|
??_pad = 0
|
|
endif
|
|
ifidni <length>,<BYTE>
|
|
??_frame = ??_frame + 1 + 3 * ??_pad
|
|
a = ??_frame
|
|
?deflocal <name>
|
|
name equ byte ptr [??_basereg-a]
|
|
elseifidni <length>,<WORD>
|
|
??_frame = ??_frame + 2 + 2 * ??_pad
|
|
a = ??_frame
|
|
?deflocal <name>
|
|
name equ word ptr [??_basereg-a]
|
|
elseifidni <length>,<DWORD>
|
|
??_frame = ??_frame + 4
|
|
a = ??_frame
|
|
?deflocal <name, name&l, name&ll, name&lh, name&h, name&hl, name&hh>
|
|
name equ dword ptr [??_basereg-a]
|
|
name&l equ word ptr [??_basereg-a]
|
|
name&ll equ byte ptr [??_basereg-a]
|
|
name&lh equ byte ptr [??_basereg-a+1]
|
|
name&h equ word ptr [??_basereg-a+2]
|
|
name&hl equ byte ptr [??_basereg-a+2]
|
|
name&hh equ byte ptr [??_basereg-a+3]
|
|
else
|
|
??_frame = ??_frame + ((length + 3)/4)*4
|
|
a = ??_frame
|
|
?deflocal <name>
|
|
name equ [??_basereg-a]
|
|
endif
|
|
??_procflags = ??_procflags OR ??_pf_Check
|
|
endm
|
|
|
|
?deflocal macro name
|
|
irp nm, <name>
|
|
??_numlocalsymbols = ??_numlocalsymbols + 1
|
|
?dodeflocal <nm>, %(??_numlocalsymbols)
|
|
endm
|
|
endm
|
|
.xcref ?deflocal
|
|
|
|
?dodeflocal macro name, num
|
|
.xcref ?LOC&num
|
|
?LOC&num ¯o
|
|
name equ <__inaccessible__NOTINSCOPE__>
|
|
&endm
|
|
endm
|
|
.xcref ?dodeflocal
|
|
|
|
;*** EnterProc - generates stack frame on entry
|
|
|
|
EnterProc macro
|
|
.errnz ??_frame and 3, <Total size of local variables not a multiple of 4.>
|
|
if ??_scall
|
|
if ??_public
|
|
ifdef Not_VxD
|
|
?merge %??_name,@,%(??_aframe),,label,near
|
|
?merge public,,,,C,%??_name,@,%(??_aframe)
|
|
else
|
|
?merge _,%??_name,@,%(??_aframe),label,near
|
|
?merge public,,,,,_,%??_name,@,%(??_aframe)
|
|
endif
|
|
endif
|
|
?_BeginProc_Debug_Prologue
|
|
endif
|
|
if ??_fastcall
|
|
if ??_public
|
|
ifdef Not_VxD
|
|
?merge %??_name,@,%(??_aframe),,label,near
|
|
?merge public,,,,C,%??_name,@,%(??_aframe)
|
|
else
|
|
?merge @,%??_name,@,%(??_aframe),label,near
|
|
?merge public,,,,,@,%??_name,@,%(??_aframe)
|
|
endif
|
|
endif
|
|
?_BeginProc_Debug_Prologue
|
|
endif
|
|
if ??_pcall
|
|
??_aframe = 0
|
|
?count = ??_numargs
|
|
rept ??_numargs
|
|
?invprg <?MKA>,%?count
|
|
?count = ?count - 1
|
|
endm
|
|
endif
|
|
??_fleave = FALSE
|
|
if ??_esp
|
|
if ??_frame
|
|
sub esp, ??_frame
|
|
??_pushed = ??_pushed + ??_frame
|
|
??_fleave = VMM_TRUE
|
|
endif
|
|
else
|
|
if ??_frame eq 0
|
|
if (??_taframe eq 0) OR ((??_procflags AND ??_pf_ArgsUsed) EQ 0)
|
|
ifdef DEBUG
|
|
push ebp
|
|
mov ebp,esp
|
|
??_fleave = VMM_TRUE
|
|
endif
|
|
else
|
|
push ebp
|
|
mov ebp,esp
|
|
??_fleave = VMM_TRUE
|
|
endif
|
|
else
|
|
enter ??_frame, 0
|
|
??_fleave = VMM_TRUE
|
|
endif
|
|
endif
|
|
??_procflags = ??_procflags OR ??_pf_Entered
|
|
endm
|
|
|
|
;*** LeaveProc - removes stack frame on exit
|
|
;
|
|
; NOTE: If there are localvar and ESP kind of stack frame
|
|
; LeaveProc will destroy flags unless the "PRESERVE_FLAGS"
|
|
; flag is given. PRESERVE_FLAGS generates bigger, slower
|
|
; code, so use it only when necessary.
|
|
;
|
|
; WARNING: For "ESP" type stack frames, this macro DOES NOT adjust
|
|
; the internal stack depth for the local frame. This is
|
|
; to allow jumping around the LeaveProc/Return to code
|
|
; after the LeaveProc/Return to use args/local variables,
|
|
; but code that uses the stack frame executed after the
|
|
; LeaveProc won't work.
|
|
|
|
LeaveProc macro flags
|
|
if ??_fleave
|
|
if ??_esp
|
|
ifidni <flags>,<PRESERVE_FLAGS>
|
|
lea esp,[esp + ??_frame]
|
|
else
|
|
add esp,??_frame
|
|
endif
|
|
else
|
|
leave
|
|
endif
|
|
endif
|
|
??_procflags = ??_procflags OR ??_pf_Left
|
|
endm
|
|
|
|
;*** Return - return appropriately from a procedure
|
|
;
|
|
; For "ccall" functions it's just a ret; for "pcall" and "scall"
|
|
; it cleans the parameters off.
|
|
;
|
|
|
|
Return macro
|
|
if ??_cleanoff OR ??_w32svc
|
|
if ??_w32svc AND (??_taframe LT 8)
|
|
ret 8
|
|
else
|
|
ret ??_taframe
|
|
endif
|
|
else
|
|
ret
|
|
endif
|
|
??_procflags = ??_procflags OR ??_pf_Returned
|
|
endm
|
|
|
|
;*** EndProc - end the procedure
|
|
;
|
|
|
|
EndProc macro Name, Flag
|
|
Name endp ;; Masm will provide error msg for us
|
|
if ??_w32svc
|
|
if ??_taframe lt 8
|
|
cparm&Name equ 0
|
|
else
|
|
cparm&Name equ (??_taframe/4 - 2)
|
|
endif
|
|
endif
|
|
if ??_procflags AND ??_pf_Left
|
|
if ??_fleave
|
|
if ??_esp
|
|
??_pushed = ??_pushed - ??_frame
|
|
endif
|
|
endif
|
|
endif
|
|
ifdifi <Flag>,<NOCHECK>
|
|
if ??_pushed ne 0
|
|
%out Warning: stack not balanced in Name
|
|
endif
|
|
if ??_procflags AND ??_pf_Check
|
|
ife ??_procflags AND ??_pf_Entered
|
|
%out Warning: ArgVar/LocalVar without EnterProc in Name
|
|
endif
|
|
ife ??_procflags AND ??_pf_Left
|
|
%out Warning: ArgVar/LocalVar without LeaveProc in Name
|
|
endif
|
|
ife ??_procflags AND ??_pf_Returned
|
|
%out Warning: ArgVar/LocalVar without Return in Name
|
|
endif
|
|
endif
|
|
endif
|
|
ifdifi <Flag>,<KEEPFRAMEVARS>
|
|
?count = 0
|
|
rept ??_numlocalsymbols
|
|
?count = ?count + 1
|
|
?invprg <?LOC>,%?count
|
|
endm
|
|
endif
|
|
??_ends
|
|
??_ends equ <>
|
|
endm
|
|
|
|
;*** cCall - "C" call
|
|
;
|
|
; Arguments pushed in "C" order, caller cleans stack
|
|
;
|
|
; USES: Flags.
|
|
|
|
cCall macro name, arglst, flags
|
|
ife .TYPE name
|
|
CondExtern name, near
|
|
endif
|
|
ifdef ??_nonstandardccall_&name
|
|
PushCParams <arglst>, <FAST>
|
|
else
|
|
PushCParams <arglst>, <flags>
|
|
endif
|
|
call name
|
|
ifdef ??_nonstandardccall_&name
|
|
ClearCParams PRESERVE_FLAGS
|
|
else
|
|
ClearCParams <flags>
|
|
endif
|
|
endm
|
|
.xcref cCall
|
|
|
|
;*** pCall - pascal call
|
|
;
|
|
; Arguments pushed in pascal order, callee cleans stack
|
|
;
|
|
|
|
pCall macro name, arglst
|
|
local ??saved
|
|
ife .TYPE name
|
|
?toupper name
|
|
else
|
|
?upper equ <name>
|
|
endif
|
|
CondExtern %?upper, near
|
|
??saved = ??_pushed
|
|
irp x,<arglst>
|
|
push x
|
|
??_pushed = ??_pushed + 4
|
|
endm
|
|
call ?upper
|
|
??_pushed = ??saved
|
|
endm
|
|
.xcref pCall
|
|
|
|
;*** sCall - standard call
|
|
;
|
|
; Arguments pushed in "C" order, callee cleans stack,
|
|
; @argc appended to name
|
|
;
|
|
|
|
sCall macro name, arglst
|
|
local ??saved
|
|
??saved = ??_pushed
|
|
PushCParams <arglst>
|
|
?scall _, name, %(??_argc * 4)
|
|
??_pushed = ??saved
|
|
endm
|
|
.xcref sCall
|
|
|
|
;*** fCall - fastcall call
|
|
;
|
|
; Arguments pushed in "C" order (except first two parms,
|
|
; which are passed in ECX and EDX), callee cleans stack, and
|
|
; @argc appended to name.
|
|
;
|
|
; The only useful value for flags is PRESERVE_FLAGS,
|
|
; which can also be achieved by simply declaring the function
|
|
; as non-standard, like so:
|
|
;
|
|
; DeclareNonstandardCcallService <functionname>
|
|
;
|
|
|
|
fCall macro name, arglst, flags
|
|
local ??saved
|
|
??saved = ??_pushed
|
|
ife .TYPE name
|
|
CondExtern name, near
|
|
endif
|
|
PushCParams <arglst>, <FASTCALL>
|
|
?scall @, name, %(??_argc * 4)
|
|
ifdef ??_nonstandardccall_&name
|
|
ClearCParams PRESERVE_FLAGS
|
|
else
|
|
ClearCParams <flags>
|
|
endif
|
|
??_pushed = ??saved
|
|
endm
|
|
.xcref fCall
|
|
|
|
;*** iCall - internal routine call
|
|
;
|
|
; Set to whatever type we want to use as a default.
|
|
|
|
iCall equ <sCall>
|
|
|
|
;*** PushCParams
|
|
;
|
|
; Processes argument list
|
|
;
|
|
; arglist = <arg1, arg2, arg3, ...>
|
|
; flags = the word SMALL if we should prefer size over speed
|
|
; the word FAST if we should prefer speed over size
|
|
;
|
|
; The default flag is SMALL, unless the current procedure
|
|
; is High_Freq, in which case we default to FAST.
|
|
;
|
|
; To disable this optimization, define the symbol NONSTANDARD_CCALL.
|
|
;
|
|
IFNDEF STANDARD_CCALL
|
|
NONSTANDARD_CCALL = 1 ;; disabled by default for now
|
|
ENDIF
|
|
|
|
PushCParams macro arglst, flags
|
|
LOCAL ??_pushedargs
|
|
|
|
??_argc = 0 ;; number of dwords on stack (global)
|
|
IFDEF NONSTANDARD_CCALL
|
|
??_popargs = 0 ;; establish default
|
|
ELSE
|
|
??_popargs = ??_align EQ 0 ;; establish default
|
|
ENDIF
|
|
ifidni <flags>, <SMALL>
|
|
??_popargs = 1 ;; size, not speed
|
|
elseifidni <flags>, <FAST>
|
|
??_popargs = 0 ;; speed, not size
|
|
elseifidni <flags>, <FASTCALL>
|
|
??_popargs = 0 ;; speed, not size
|
|
endif
|
|
|
|
irp x,<arglst>
|
|
??_argc = ??_argc + 1
|
|
ifidni <flags>, <FASTCALL>
|
|
if ??_argc eq 1
|
|
ifdifi <x>, <ecx>
|
|
.err <first parameter must be ECX for fastcall functions>
|
|
endif
|
|
elseif ??_argc eq 2
|
|
ifdifi <x>, <edx>
|
|
.err <first parameter must be EDX for fastcall functions>
|
|
endif
|
|
else
|
|
?marg <x>,%??_argc
|
|
endif
|
|
else
|
|
?marg <x>,%??_argc
|
|
endif
|
|
endm
|
|
?count = ??_argc
|
|
ifidni <flags>, <FASTCALL>
|
|
??_pushedargs = ??_argc-2
|
|
else
|
|
??_pushedargs = ??_argc
|
|
endif
|
|
if ??_pushedargs GT 0
|
|
rept ??_argc
|
|
?invprg <?AM>,%?count
|
|
?count = ?count - 1
|
|
endm
|
|
endif
|
|
endm
|
|
|
|
;*** ClearCParams
|
|
;
|
|
; Processes stack clean up
|
|
;
|
|
; This routine will trade size for speed (if requested)
|
|
; by using `pop ecx' to clean off one or two arguments.
|
|
; This relies on the convention that C-call routines do
|
|
; not return useful information in ECX.
|
|
;
|
|
; To disable this optimization, define the symbol NONSTANDARD_CCALL.
|
|
;
|
|
; If flags must be preserved, pass PRESERVE_FLAGS as an argument.
|
|
; This will generate bigger, slower code, so use it only when
|
|
; necessary.
|
|
|
|
ClearCParams macro fPreserveFlags
|
|
if ??_argc ne 0
|
|
if (??_popargs) AND (??_argc LE 2)
|
|
rept ??_argc
|
|
pop ecx
|
|
endm
|
|
elseifidni <fPreserveFlags>, <PRESERVE_FLAGS>
|
|
lea esp, [esp][??_argc * 4]
|
|
else
|
|
add esp,??_argc * 4
|
|
endif
|
|
endif
|
|
??_pushed = ??_pushed - (??_argc * 4)
|
|
endm
|
|
|
|
; Makes a macro that will push argment when invoke - used by cCall only
|
|
|
|
?marg macro name, num
|
|
.xcref
|
|
.xcref ?AM&num
|
|
.cref
|
|
?AM&num ¯o
|
|
push name
|
|
??_pushed = ??_pushed + 4
|
|
&endm
|
|
endm
|
|
.xcref ?marg
|
|
|
|
; Concatenates, invokes and purges a macro name - used by PushCParams
|
|
|
|
?invprg macro name1, name2
|
|
name1&name2
|
|
purge name1&name2
|
|
endm
|
|
.xcref ?invprg
|
|
|
|
; Calls a concatenated standard call name and makes it external
|
|
|
|
?scall macro prefix, name1, name2
|
|
CondExtern prefix&name1&@&name2, near
|
|
call prefix&name1&@&name2
|
|
endm
|
|
.xcref ?scall
|
|
|
|
; Equates name to a name
|
|
|
|
?merge macro l1, l2, l3, l4, op, r1, r2, r3, r4, r5, r6, r7, r8, r9
|
|
l1&l2&l3&l4 op r1&r2&r3&r4&r5&r6&r7&r8&r9
|
|
endm
|
|
|
|
; Converts string to upper-case, returned in ?upper
|
|
|
|
?toupper macro s
|
|
?upper equ <>
|
|
irpc x,<s>
|
|
if '&x' GE 'a'
|
|
if '&x' LE 'z'
|
|
?t1 substr <ABCDEFGHIJKLMNOPQRSTUVWXYZ>,'&x'-'a'+1,1
|
|
?upper catstr ?upper,?t1
|
|
else
|
|
?upper catstr ?upper,<&x>
|
|
endif
|
|
else
|
|
?upper catstr ?upper,<&x>
|
|
endif
|
|
endm
|
|
endm
|
|
.xcref
|
|
|
|
;*** CondExtern - Make name external if not already defined
|
|
;
|
|
; This operation is quite different between MASM 5.1 and 6.0.
|
|
;
|
|
|
|
CondExtern macro name,dist
|
|
ifdef MASM6
|
|
ifndef name
|
|
externdef name:dist
|
|
endif
|
|
else
|
|
if2
|
|
ifndef name
|
|
extrn name:dist
|
|
endif
|
|
endif
|
|
endif
|
|
endm
|
|
|
|
;*** SaveReg - Save register, "fd" pushes flags, "ad" pushes all
|
|
|
|
SaveReg macro reglist ;; push those registers
|
|
irp reg,<reglist>
|
|
ifidni <reg>, <fd>
|
|
pushfd
|
|
??_pushed = ??_pushed + 4
|
|
else
|
|
ifidni <reg>, <ad>
|
|
pushad
|
|
??_pushed = ??_pushed + SIZE Pushad_Struc
|
|
else
|
|
push reg
|
|
??_pushed = ??_pushed + 4
|
|
endif
|
|
endif
|
|
endm
|
|
endm
|
|
|
|
;*** RestoreReg - Restore register, "fd" pops flags, "ad" pops all
|
|
;
|
|
; Note that registers must be restored in reverse order that they
|
|
; were saved.
|
|
;
|
|
|
|
RestoreReg macro reglist ;; pop those registers
|
|
irp reg,<reglist>
|
|
ifidni <reg>, <fd>
|
|
popfd
|
|
??_pushed = ??_pushed - 4
|
|
else
|
|
ifidni <reg>, <ad>
|
|
popad
|
|
??_pushed = ??_pushed - SIZE Pushad_Struc
|
|
else
|
|
pop reg
|
|
??_pushed = ??_pushed - 4
|
|
endif
|
|
endif
|
|
endm
|
|
endm
|
|
*/
|
|
|
|
#ifdef DEBUG
|
|
/******************************************************************************
|
|
* The following macros are for enabling procedure call profile counting
|
|
* of VxD's written in assembler.
|
|
*
|
|
* Begin_Profile_List needs to be used in the file that declares the device
|
|
* immediately after the Declare_Virtual_Device line. Then one Profile_Link
|
|
* line is required for each individual source file. The list is ended with
|
|
* the End_Profile_List macro. Profiling only works for debug builds and
|
|
* the sources must all be built with "-Dprofileall" masm switch.
|
|
******************************************************************************/
|
|
|
|
/* ASM
|
|
Begin_Profile_List macro devname
|
|
ifdef profileall
|
|
VxD_DATA_SEG
|
|
db 'PROCLIST'
|
|
PUBLIC devname&_Proc_Profile_List
|
|
devname&_Proc_Profile_List label dword
|
|
endif
|
|
endm
|
|
|
|
Profile_Link macro modname
|
|
ifdef profileall
|
|
ifdifi <modname>,@filename
|
|
EXTRN _&modname&__proc_list:near
|
|
endif
|
|
dd OFFSET32 _&modname&__proc_list
|
|
endif
|
|
endm
|
|
|
|
End_Profile_List macro
|
|
ifdef profileall
|
|
dd 0
|
|
VxD_DATA_ENDS
|
|
endif
|
|
endm
|
|
|
|
*/
|
|
#endif
|
|
|
|
#ifndef Not_VxD
|
|
|
|
/******************************************************************************
|
|
* S C H E D U L E R B O O S T V A L U E S
|
|
*****************************************************************************/
|
|
|
|
#define RESERVED_LOW_BOOST 0x00000001
|
|
#define CUR_RUN_VM_BOOST 0x00000004
|
|
#define LOW_PRI_DEVICE_BOOST 0x00000010
|
|
#define HIGH_PRI_DEVICE_BOOST 0x00001000
|
|
#define CRITICAL_SECTION_BOOST 0x00100000
|
|
#define TIME_CRITICAL_BOOST 0x00400000
|
|
#define RESERVED_HIGH_BOOST 0x40000000
|
|
|
|
|
|
/******************************************************************************
|
|
* F L A G S F O R C A L L _ P R I O R I T Y _ V M _ E V E N T
|
|
*****************************************************************************/
|
|
|
|
#define PEF_WAIT_FOR_STI_BIT 0
|
|
#define PEF_WAIT_FOR_STI (1 << PEF_WAIT_FOR_STI_BIT)
|
|
#define PEF_WAIT_NOT_CRIT_BIT 1
|
|
#define PEF_WAIT_NOT_CRIT (1 << PEF_WAIT_NOT_CRIT_BIT)
|
|
|
|
#define PEF_DONT_UNBOOST_BIT 2
|
|
#define PEF_DONT_UNBOOST (1 << PEF_DONT_UNBOOST_BIT)
|
|
#define PEF_ALWAYS_SCHED_BIT 3
|
|
#define PEF_ALWAYS_SCHED (1 << PEF_ALWAYS_SCHED_BIT)
|
|
#define PEF_TIME_OUT_BIT 4
|
|
#define PEF_TIME_OUT (1 << PEF_TIME_OUT_BIT)
|
|
|
|
#define PEF_WAIT_NOT_HW_INT_BIT 5
|
|
#define PEF_WAIT_NOT_HW_INT (1 << PEF_WAIT_NOT_HW_INT_BIT)
|
|
#define PEF_WAIT_NOT_NESTED_EXEC_BIT 6
|
|
#define PEF_WAIT_NOT_NESTED_EXEC (1 << PEF_WAIT_NOT_NESTED_EXEC_BIT)
|
|
#define PEF_WAIT_IN_PM_BIT 7
|
|
#define PEF_WAIT_IN_PM (1 << PEF_WAIT_IN_PM_BIT)
|
|
|
|
#define PEF_THREAD_EVENT_BIT 8
|
|
#define PEF_THREAD_EVENT (1 << PEF_THREAD_EVENT_BIT)
|
|
|
|
#define PEF_WAIT_FOR_THREAD_STI_BIT 9
|
|
#define PEF_WAIT_FOR_THREAD_STI (1 << PEF_WAIT_FOR_THREAD_STI_BIT)
|
|
|
|
#define PEF_RING0_EVENT_BIT 10
|
|
#define PEF_RING0_EVENT (1 << PEF_RING0_EVENT_BIT)
|
|
|
|
#define PEF_WAIT_CRIT_BIT 11
|
|
#define PEF_WAIT_CRIT (1 << PEF_WAIT_CRIT_BIT)
|
|
|
|
#define PEF_WAIT_CRIT_VM_BIT 12
|
|
#define PEF_WAIT_CRIT_VM (1 << PEF_WAIT_CRIT_VM_BIT)
|
|
|
|
#define PEF_PROCESS_LAST_BIT 13
|
|
#define PEF_PROCESS_LAST (1 << PEF_PROCESS_LAST_BIT)
|
|
|
|
#define PEF_WAIT_PREEMPTABLE_BIT 14
|
|
#define PEF_WAIT_PREEMPTABLE (1 << PEF_WAIT_PREEMPTABLE_BIT)
|
|
|
|
// synonyms for event restrictions above
|
|
|
|
#define PEF_WAIT_NOT_TIME_CRIT_BIT PEF_WAIT_NOT_HW_INT_BIT
|
|
#define PEF_WAIT_NOT_TIME_CRIT PEF_WAIT_NOT_HW_INT
|
|
#define PEF_WAIT_NOT_PM_LOCKED_STACK_BIT PEF_WAIT_NOT_NESTED_EXEC_BIT
|
|
#define PEF_WAIT_NOT_PM_LOCKED_STACK PEF_WAIT_NOT_NESTED_EXEC
|
|
|
|
|
|
/******************************************************************************
|
|
* F L A G S F O R B E G I N _ C R I T I C A L _ S E C T I O N,
|
|
* E N T E R _ M U T E X
|
|
* A N D W A I T _ S E M A P H O R E
|
|
*****************************************************************************/
|
|
|
|
#define BLOCK_SVC_INTS_BIT 0
|
|
#define BLOCK_SVC_INTS (1 << BLOCK_SVC_INTS_BIT)
|
|
#define BLOCK_SVC_IF_INTS_LOCKED_BIT 1
|
|
#define BLOCK_SVC_IF_INTS_LOCKED (1 << BLOCK_SVC_IF_INTS_LOCKED_BIT)
|
|
#define BLOCK_ENABLE_INTS_BIT 2
|
|
#define BLOCK_ENABLE_INTS (1 << BLOCK_ENABLE_INTS_BIT)
|
|
#define BLOCK_POLL_BIT 3
|
|
#define BLOCK_POLL (1 << BLOCK_POLL_BIT)
|
|
#define BLOCK_THREAD_IDLE_BIT 4
|
|
#define BLOCK_THREAD_IDLE (1 << BLOCK_THREAD_IDLE_BIT)
|
|
#define BLOCK_FORCE_SVC_INTS_BIT 5
|
|
#define BLOCK_FORCE_SVC_INTS (1 << BLOCK_FORCE_SVC_INTS_BIT)
|
|
|
|
/******************************************************************************
|
|
* The following structures are pointed to by EBP when VxD routines are
|
|
* entered, both for VxD control calls and traps(I/O traps, software INT
|
|
* traps, etc.). The first structure as DWORD values, the second WORD
|
|
* values and the last has BYTE values.
|
|
*****************************************************************************/
|
|
|
|
struct Client_Reg_Struc {
|
|
ULONG Client_EDI; /* Client's EDI */
|
|
ULONG Client_ESI; /* Client's ESI */
|
|
ULONG Client_EBP; /* Client's EBP */
|
|
ULONG Client_res0; /* ESP at pushall */
|
|
ULONG Client_EBX; /* Client's EBX */
|
|
ULONG Client_EDX; /* Client's EDX */
|
|
ULONG Client_ECX; /* Client's ECX */
|
|
ULONG Client_EAX; /* Client's EAX */
|
|
ULONG Client_Error; /* Dword error code */
|
|
ULONG Client_EIP; /* EIP */
|
|
USHORT Client_CS; /* CS */
|
|
USHORT Client_res1; /* (padding) */
|
|
ULONG Client_EFlags; /* EFLAGS */
|
|
ULONG Client_ESP; /* ESP */
|
|
USHORT Client_SS; /* SS */
|
|
USHORT Client_res2; /* (padding) */
|
|
USHORT Client_ES; /* ES */
|
|
USHORT Client_res3; /* (padding) */
|
|
USHORT Client_DS; /* DS */
|
|
USHORT Client_res4; /* (padding) */
|
|
USHORT Client_FS; /* FS */
|
|
USHORT Client_res5; /* (padding) */
|
|
USHORT Client_GS; /* GS */
|
|
USHORT Client_res6; /* (padding) */
|
|
ULONG Client_Alt_EIP;
|
|
USHORT Client_Alt_CS;
|
|
USHORT Client_res7;
|
|
ULONG Client_Alt_EFlags;
|
|
ULONG Client_Alt_ESP;
|
|
USHORT Client_Alt_SS;
|
|
USHORT Client_res8;
|
|
USHORT Client_Alt_ES;
|
|
USHORT Client_res9;
|
|
USHORT Client_Alt_DS;
|
|
USHORT Client_res10;
|
|
USHORT Client_Alt_FS;
|
|
USHORT Client_res11;
|
|
USHORT Client_Alt_GS;
|
|
USHORT Client_res12;
|
|
};
|
|
|
|
|
|
struct Client_Word_Reg_Struc {
|
|
USHORT Client_DI; /* Client's DI */
|
|
USHORT Client_res13; /* (padding) */
|
|
USHORT Client_SI; /* Client's SI */
|
|
USHORT Client_res14; /* (padding) */
|
|
USHORT Client_BP; /* Client's BP */
|
|
USHORT Client_res15; /* (padding) */
|
|
ULONG Client_res16; /* ESP at pushall */
|
|
USHORT Client_BX; /* Client's BX */
|
|
USHORT Client_res17; /* (padding) */
|
|
USHORT Client_DX; /* Client's DX */
|
|
USHORT Client_res18; /* (padding) */
|
|
USHORT Client_CX; /* Client's CX */
|
|
USHORT Client_res19; /* (padding) */
|
|
USHORT Client_AX; /* Client's AX */
|
|
USHORT Client_res20; /* (padding) */
|
|
ULONG Client_res21; /* Dword error code */
|
|
USHORT Client_IP; /* Client's IP */
|
|
USHORT Client_res22; /* (padding) */
|
|
ULONG Client_res23; /* CS */
|
|
USHORT Client_Flags; /* Client's flags (low) */
|
|
USHORT Client_res24; /* (padding) */
|
|
USHORT Client_SP; /* SP */
|
|
USHORT Client_res25;
|
|
ULONG Client_res26[5];
|
|
USHORT Client_Alt_IP;
|
|
USHORT Client_res27;
|
|
ULONG Client_res28;
|
|
USHORT Client_Alt_Flags;
|
|
USHORT Client_res29;
|
|
USHORT Client_Alt_SP;
|
|
};
|
|
|
|
|
|
|
|
struct Client_Byte_Reg_Struc {
|
|
ULONG Client_res30[4]; /* EDI, ESI, EBP, ESP at pushall */
|
|
UCHAR Client_BL; /* Client's BL */
|
|
UCHAR Client_BH; /* Client's BH */
|
|
USHORT Client_res31;
|
|
UCHAR Client_DL; /* Client's DL */
|
|
UCHAR Client_DH; /* Client's DH */
|
|
USHORT Client_res32;
|
|
UCHAR Client_CL; /* Client's CL */
|
|
UCHAR Client_CH; /* Client's CH */
|
|
USHORT Client_res33;
|
|
UCHAR Client_AL; /* Client's AL */
|
|
UCHAR Client_AH; /* Client's AH */
|
|
};
|
|
|
|
|
|
typedef union tagCLIENT_STRUC { /* */
|
|
struct Client_Reg_Struc CRS;
|
|
struct Client_Word_Reg_Struc CWRS;
|
|
struct Client_Byte_Reg_Struc CBRS;
|
|
} CLIENT_STRUCT;
|
|
|
|
typedef struct Client_Reg_Struc CRS;
|
|
typedef CRS *PCRS;
|
|
|
|
#if 0 /* causes problems with MASM 6 */
|
|
/* ASM
|
|
.ERRNZ Client_SP - Client_ESP
|
|
.ERRNZ Client_AL - Client_EAX
|
|
*/
|
|
#endif
|
|
|
|
#define DYNA_LINK_INT 0x20
|
|
|
|
/* ASM
|
|
|
|
;*** DeclareNonstandardCcallService
|
|
;
|
|
; Declare services as conforming to the C calling convention
|
|
; for parameter-passing, but *not* conforming to the C calling
|
|
; convention for register usage.
|
|
;
|
|
; Services which do not use the C calling convention for
|
|
; parameter-passing need not be declared as nonstandard.
|
|
;
|
|
; arglst - list of services to declare as nonstandard
|
|
;
|
|
DeclareNonstandardCcallService macro arglst
|
|
irp x,<arglst>
|
|
??_nonstandardccall_&&x = 1
|
|
endm
|
|
endm
|
|
|
|
;
|
|
; The following VMM services are nonstandard:
|
|
; _BlockOnID and _LocalizeSprintf modify no registers except flags.
|
|
; _SetLastV86Page modifies no registers except EAX and flags.
|
|
;
|
|
DeclareNonstandardCcallService <_BlockOnID, _LocalizeSprintf>
|
|
DeclareNonstandardCcallService <_SetLastV86Page>
|
|
|
|
BeginDoc
|
|
;******************************************************************************
|
|
; The VMMCall and VxDCall macros provide a dynamic link to the VMM and VxD
|
|
; service routines. For example:
|
|
;
|
|
; VMMCall Enable_VM_Ints ; Equivalent to STI in VM code
|
|
;
|
|
; mov eax,[My_IRQ_Handle]
|
|
; VxDCall VPICD_Set_Int_Request ; Set IRQ for my device's interrupt
|
|
;
|
|
; Note that Enable_VM_Ints is defined in VMM.INC and VPICD_Set_Int_Request is
|
|
; defined in VPICD.INC
|
|
;
|
|
;==============================================================================
|
|
EndDoc
|
|
|
|
|
|
BeginDoc
|
|
;******************************************************************************
|
|
; VxDCall
|
|
;==============================================================================
|
|
;
|
|
; BlockOnID is always FAST because it doesn't
|
|
; conform to the C calling convention. (It preserves
|
|
; all registers.)
|
|
|
|
EndDoc
|
|
|
|
DefTable MACRO vt, vn
|
|
vt EQU <vn>
|
|
ENDM
|
|
|
|
GenDD2 MACRO vt, sn, jf
|
|
dd OFFSET32 vt[sn+jf]
|
|
ENDM
|
|
|
|
GenDD MACRO P, vid, snum, jflag
|
|
LOCAL vtable
|
|
IFDEF @@VxDName&vid
|
|
Deftable vtable, %@@VxDName&vid
|
|
EXTRN vtable:DWORD
|
|
GenDD2 %vtable, snum, jflag
|
|
ELSE
|
|
dd @@&P+jflag
|
|
ENDIF
|
|
|
|
ENDM
|
|
|
|
|
|
VxDCall MACRO P, Param, flags
|
|
??_vxdid = (@@&P SHR 16)
|
|
??_servicenum = (@@&P AND 0FFFFh)
|
|
|
|
ifdef ??_standardccall_&P
|
|
PushCParams <Param>, <FAST>
|
|
.errnz ??_argc ne ??_standardccall_&P, <wrong # of parameters passed to &P&>
|
|
else
|
|
ifdef ??_fastcall_&P
|
|
PushCParams <Param>, <FASTCALL>
|
|
.errnz ??_argc ne (??_fastcall_&P), <wrong # of parameters passed to fastcall function &P&>
|
|
else
|
|
ifdef ??_nonstandardccall_&P
|
|
PushCParams <Param>, <flags>
|
|
else
|
|
PushCParams <Param>, <FAST>
|
|
endif
|
|
endif
|
|
endif
|
|
int Dyna_Link_Int
|
|
GenDD P, %??_vxdid, %??_servicenum, 0
|
|
ifndef ??_standardccall_&P
|
|
ifndef ??_fastcall_&P
|
|
ifdef ??_nonstandardccall_&P
|
|
ClearCParams PRESERVE_FLAGS
|
|
else
|
|
ClearCParams
|
|
endif
|
|
else
|
|
if(??_argc gt 2)
|
|
??_pushed = ??_pushed - ((??_argc - 2) * 4)
|
|
endif
|
|
endif
|
|
else
|
|
??_pushed = ??_pushed - (??_argc * 4)
|
|
endif
|
|
ENDM
|
|
|
|
VxDJmp MACRO P, Param
|
|
??_vxdid = (@@&P SHR 16)
|
|
??_servicenum = (@@&P AND 0FFFFh)
|
|
ifdef ??_fastcall_&P
|
|
PushCParams <Param>, <FASTCALL>
|
|
.errnz ??_argc gt 2, <More than 2 parameters may not be passed to fastcall functions thru VxDJmp>
|
|
else
|
|
.errnb <Param>, <Parameters may not be passed to VxDJmp or VMMJmp>
|
|
endif
|
|
int Dyna_Link_Int
|
|
GenDD P, %??_vxdid, %??_servicenum, DL_Jmp_Mask
|
|
ENDM
|
|
|
|
DL_Jmp_Mask EQU 8000h
|
|
DL_Jmp_Bit EQU 0Fh
|
|
|
|
VMMCall MACRO P, Param
|
|
.ERRNZ (@@&P SHR 16) - VMM_DEVICE_ID
|
|
VxDCall <P>, <Param>
|
|
ENDM
|
|
|
|
VMMJmp MACRO P, Param
|
|
.ERRNZ (@@&P SHR 16) - VMM_DEVICE_ID
|
|
VxDJmp <P>, <Param>
|
|
ENDM
|
|
|
|
BeginDoc
|
|
;******************************************************************************
|
|
; Segment definition macros
|
|
;
|
|
; The segment definition macros are a convenience used in defining the
|
|
; segments used by the device driver. They are:
|
|
;VxD_INIT_CODE_SEG defines start of initialization code segment
|
|
;VxD_INIT_CODE_ENDS defines end of initialization code segment
|
|
;VxD_ICODE_SEG is an alias for VxD_INIT_CODE_SEG
|
|
;VxD_ICODE_ENDS is an alias for VxD_INIT_CODE_ENDS
|
|
;VxD_IDATA_SEG defines start of initialization data segment
|
|
;VxD_IDATA_ENDS defines end of initialization data segment
|
|
;VxD_CODE_SEG defines start of always present code segment
|
|
;VxD_CODE_ENDS defines end of always present code segment
|
|
;VxD_DATA_SEG defines start of always present data segment
|
|
;VxD_DATA_ENDS defines end of always present data segment
|
|
;VxD_LOCKED_CODE_SEG defines start of always present code segment
|
|
;VxD_LOCKED_CODE_ENDS defines end of always present code segment
|
|
;VxD_PAGEABLE_CODE_SEG defines start of swappable code segment
|
|
;VxD_PAGEABLE_CODE_ENDS defines end of swappable code segment
|
|
;VxD_DEBUG_ONLY_CODE_SEG defines code only loaded if debugger is present
|
|
;VxD_DEBUG_ONLY_CODE_ENDS
|
|
;VxD_DEBUG_ONLY_DATA_SEG defines data only loaded if debugger is present
|
|
;VxD_DEBUG_ONLY_DATA_ENDS
|
|
;==============================================================================
|
|
|
|
|
|
|
|
|
|
EndDoc
|
|
|
|
|
|
; Resident protected mode code
|
|
|
|
VxD_CODE_SEG EQU <VxD_LOCKED_CODE_SEG>
|
|
VxD_CODE_ENDS EQU <VxD_LOCKED_CODE_ENDS>
|
|
|
|
|
|
VxD_LOCKED_CODE_SEG MACRO
|
|
_LTEXT SEGMENT
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + ??_LCODE
|
|
ASSUME cs:FLAT, ds:FLAT, es:FLAT, ss:FLAT
|
|
|
|
ENDM
|
|
|
|
VxD_LOCKED_CODE_ENDS MACRO
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHR 4
|
|
_LTEXT ENDS
|
|
ENDM
|
|
|
|
|
|
; Pageable protected mode code
|
|
|
|
VxD_PAGEABLE_CODE_SEG MACRO
|
|
_PTEXT SEGMENT
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + ??_PCODE
|
|
ASSUME cs:FLAT, ds:FLAT, es:FLAT, ss:FLAT
|
|
|
|
ENDM
|
|
|
|
VxD_PAGEABLE_CODE_ENDS MACRO
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHR 4
|
|
_PTEXT ENDS
|
|
ENDM
|
|
|
|
|
|
; Debug only protected mode code
|
|
|
|
VxD_DEBUG_ONLY_CODE_SEG MACRO
|
|
_DBOCODE SEGMENT
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + ??_DBOCODE
|
|
ASSUME cs:FLAT, ds:FLAT, es:FLAT, ss:FLAT
|
|
ENDM
|
|
|
|
VxD_DEBUG_ONLY_CODE_ENDS MACRO
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHR 4
|
|
_DBOCODE ENDS
|
|
ENDM
|
|
|
|
|
|
; Protected mode initialization code
|
|
|
|
VxD_INIT_CODE_SEG MACRO
|
|
_ITEXT SEGMENT
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + ??_ICODE
|
|
ASSUME cs:FLAT, ds:FLAT, es:FLAT, ss:FLAT
|
|
ENDM
|
|
|
|
VxD_INIT_CODE_ENDS MACRO
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHR 4
|
|
_ITEXT ENDS
|
|
ENDM
|
|
|
|
VxD_ICODE_SEG equ VxD_INIT_CODE_SEG
|
|
VxD_ICODE_ENDS equ VxD_INIT_CODE_ENDS
|
|
|
|
|
|
; Resident protected mode data
|
|
|
|
VxD_DATA_SEG EQU <VxD_LOCKED_DATA_SEG>
|
|
VxD_DATA_ENDS EQU <VxD_LOCKED_DATA_ENDS>
|
|
|
|
VxD_LOCKED_DATA_SEG MACRO NO_ALIGN
|
|
_LDATA SEGMENT
|
|
IFB <NO_ALIGN>
|
|
ALIGN 4
|
|
ENDIF
|
|
ENDM
|
|
|
|
VxD_LOCKED_DATA_ENDS MACRO
|
|
_LDATA ENDS
|
|
ENDM
|
|
|
|
|
|
; Protected mode initialization data
|
|
|
|
VxD_IDATA_SEG MACRO
|
|
_IDATA SEGMENT
|
|
ENDM
|
|
VxD_IDATA_ENDS MACRO
|
|
_IDATA ENDS
|
|
ENDM
|
|
|
|
|
|
; Pageable protected mode data
|
|
|
|
VxD_PAGEABLE_DATA_SEG MACRO NO_ALIGN
|
|
_PDATA SEGMENT
|
|
IFB <NO_ALIGN>
|
|
ALIGN 4
|
|
ENDIF
|
|
ENDM
|
|
|
|
VxD_PAGEABLE_DATA_ENDS MACRO
|
|
_PDATA ENDS
|
|
ENDM
|
|
|
|
|
|
; Static code segment for DL-VxDs
|
|
|
|
VxD_STATIC_CODE_SEG MACRO
|
|
_STEXT SEGMENT
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + ??_SCODE
|
|
ASSUME cs:FLAT, ds:FLAT, es:FLAT, ss:FLAT
|
|
|
|
ENDM
|
|
|
|
VxD_STATIC_CODE_ENDS MACRO
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHR 4
|
|
_STEXT ENDS
|
|
ENDM
|
|
|
|
|
|
; Static data segment for DL-VxDs
|
|
|
|
VxD_STATIC_DATA_SEG MACRO NO_ALIGN
|
|
_SDATA SEGMENT
|
|
IFB <NO_ALIGN>
|
|
ALIGN 4
|
|
ENDIF
|
|
ENDM
|
|
|
|
VxD_STATIC_DATA_ENDS MACRO
|
|
_SDATA ENDS
|
|
ENDM
|
|
|
|
; Debug only protected mode data
|
|
|
|
VxD_DEBUG_ONLY_DATA_SEG MACRO NO_ALIGN
|
|
_DBODATA SEGMENT
|
|
IFB <NO_ALIGN>
|
|
ALIGN 4
|
|
ENDIF
|
|
ENDM
|
|
|
|
VxD_DEBUG_ONLY_DATA_ENDS MACRO
|
|
_DBODATA ENDS
|
|
ENDM
|
|
|
|
|
|
; 16 bit code/data put in the init group (IGROUP)
|
|
|
|
VxD_16BIT_INIT_SEG MACRO
|
|
_16ICODE SEGMENT
|
|
ASSUME CS:_16ICODE, DS:NOTHING, ES:NOTHING, SS:NOTHING
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + ??_16ICODE
|
|
ENDM
|
|
|
|
VxD_16BIT_INIT_ENDS MACRO
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHR 4
|
|
_16ICODE ENDS
|
|
ENDM
|
|
|
|
; Real mode segment (16 bit)
|
|
|
|
VxD_REAL_INIT_SEG MACRO
|
|
_RCODE SEGMENT
|
|
ASSUME CS:_RCODE, DS:_RCODE, ES:_RCODE, SS:_RCODE
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHL 4 + ??_RCODE
|
|
ENDM
|
|
|
|
VxD_REAL_INIT_ENDS MACRO
|
|
??_CUR_CODE_SEG = ??_CUR_CODE_SEG SHR 4
|
|
_RCODE ENDS
|
|
ENDM
|
|
*/
|
|
|
|
#endif // Not_VxD
|
|
|
|
#ifndef DDK_VERSION
|
|
|
|
#ifdef WIN31COMPAT
|
|
#define DDK_VERSION 0x30A /* 3.10 */
|
|
#else // WIN31COMPAT
|
|
|
|
#ifdef WIN40COMPAT
|
|
#define DDK_VERSION 0x400 /* 4.00 */
|
|
#else // WIN40COMPAT
|
|
|
|
#ifdef OPK3
|
|
#define DDK_VERSION 0x403 /* 4.03 */
|
|
#else // OPK3
|
|
#define DDK_VERSION 0x40A /*Memphis is 4.1 */
|
|
#endif // OPK3
|
|
|
|
#endif // WIN40COMPAT
|
|
|
|
#endif // WIN31COMPAT
|
|
|
|
#endif // DDK_VERSION
|
|
|
|
struct VxD_Desc_Block {
|
|
ULONG DDB_Next; /* VMM RESERVED FIELD */
|
|
USHORT DDB_SDK_Version; /* INIT <DDK_VERSION> RESERVED FIELD */
|
|
USHORT DDB_Req_Device_Number; /* INIT <UNDEFINED_DEVICE_ID> */
|
|
UCHAR DDB_Dev_Major_Version; /* INIT <0> Major device number */
|
|
UCHAR DDB_Dev_Minor_Version; /* INIT <0> Minor device number */
|
|
USHORT DDB_Flags; /* INIT <0> for init calls complete */
|
|
UCHAR DDB_Name[8]; /* AINIT <" "> Device name */
|
|
ULONG DDB_Init_Order; /* INIT <UNDEFINED_INIT_ORDER> */
|
|
ULONG DDB_Control_Proc; /* Offset of control procedure */
|
|
ULONG DDB_V86_API_Proc; /* INIT <0> Offset of API procedure */
|
|
ULONG DDB_PM_API_Proc; /* INIT <0> Offset of API procedure */
|
|
ULONG DDB_V86_API_CSIP; /* INIT <0> CS:IP of API entry point */
|
|
ULONG DDB_PM_API_CSIP; /* INIT <0> CS:IP of API entry point */
|
|
ULONG DDB_Reference_Data; /* Reference data from real mode */
|
|
ULONG DDB_Service_Table_Ptr; /* INIT <0> Pointer to service table */
|
|
ULONG DDB_Service_Table_Size; /* INIT <0> Number of services */
|
|
ULONG DDB_Win32_Service_Table; /* INIT <0> Pointer to Win32 services */
|
|
ULONG DDB_Prev; /* INIT <'Prev'> Ptr to prev 4.0 DDB */
|
|
ULONG DDB_Size; /* INIT <SIZE(VxD_Desc_Block)> Reserved */
|
|
ULONG DDB_Reserved1; /* INIT <'Rsv1'> Reserved */
|
|
ULONG DDB_Reserved2; /* INIT <'Rsv2'> Reserved */
|
|
ULONG DDB_Reserved3; /* INIT <'Rsv3'> Reserved */
|
|
};
|
|
|
|
typedef struct VxD_Desc_Block *PVMMDDB;
|
|
typedef PVMMDDB *PPVMMDDB;
|
|
|
|
#ifndef Not_VxD
|
|
|
|
/*
|
|
* Flag values for DDB_Flags
|
|
*/
|
|
|
|
#define DDB_SYS_CRIT_INIT_DONE_BIT 0
|
|
#define DDB_SYS_CRIT_INIT_DONE (1 << DDB_SYS_CRIT_INIT_DONE_BIT)
|
|
#define DDB_DEVICE_INIT_DONE_BIT 1
|
|
#define DDB_DEVICE_INIT_DONE (1 << DDB_DEVICE_INIT_DONE_BIT)
|
|
|
|
#define DDB_HAS_WIN32_SVCS_BIT 14
|
|
#define DDB_HAS_WIN32_SVCS (1 << DDB_HAS_WIN32_SVCS_BIT)
|
|
#define DDB_DYNAMIC_VXD_BIT 15
|
|
#define DDB_DYNAMIC_VXD (1 << DDB_DYNAMIC_VXD_BIT)
|
|
|
|
#define DDB_DEVICE_DYNALINKED_BIT 13
|
|
#define DDB_DEVICE_DYNALINKED (1 << DDB_DEVICE_DYNALINKED_BIT)
|
|
|
|
|
|
/* ASM
|
|
BeginDoc
|
|
;******************************************************************************
|
|
;
|
|
; Declare_Virtual_Device macro
|
|
;
|
|
; ???? Write something here ????
|
|
;
|
|
;==============================================================================
|
|
EndDoc
|
|
Declare_Virtual_Device MACRO Name, Major_Ver, Minor_Ver, Ctrl_Proc, Device_Num, Init_Order, V86_Proc, PM_Proc, Reference_Data
|
|
LOCAL V86_API_Offset, PM_API_Offset, Serv_Tab_Offset, Serv_Tab_Len, Ref_Data_Offset
|
|
|
|
dev_id_err MACRO
|
|
|
|
IFNDEF Name&_Name_Based
|
|
.err <Device ID required when providing services>
|
|
ENDIF
|
|
ENDM
|
|
|
|
IFB <V86_Proc>
|
|
V86_API_Offset EQU 0
|
|
ELSE
|
|
IFB <Device_Num>
|
|
dev_id_err
|
|
ENDIF
|
|
V86_API_Offset EQU <OFFSET32 V86_Proc>
|
|
ENDIF
|
|
|
|
IFB <PM_Proc>
|
|
PM_API_Offset EQU 0
|
|
ELSE
|
|
IFB <Device_Num>
|
|
dev_id_err
|
|
ENDIF
|
|
PM_API_Offset EQU <OFFSET32 PM_Proc>
|
|
ENDIF
|
|
|
|
IFDEF Name&_Service_Table
|
|
IFB <Device_Num>
|
|
dev_id_err
|
|
ELSE
|
|
IFE Device_Num - UNDEFINED_DEVICE_ID
|
|
dev_id_err
|
|
ENDIF
|
|
ENDIF
|
|
Serv_Tab_Offset EQU <OFFSET32 Name&_Service_Table>
|
|
Serv_Tab_Len EQU Num_&Name&_Services
|
|
ELSE
|
|
Serv_Tab_Offset EQU 0
|
|
Serv_Tab_Len EQU 0
|
|
ENDIF
|
|
|
|
IFNB <Device_Num>
|
|
.erre (Device_Num LT BASEID_FOR_NAMEBASEDVXD), <Device ID must be less than BASEID_FOR_NAMEBASEDVXD>
|
|
ENDIF
|
|
|
|
IFB <Reference_Data>
|
|
Ref_Data_Offset EQU 0
|
|
ELSE
|
|
Ref_Data_Offset EQU <OFFSET32 Reference_Data>
|
|
ENDIF
|
|
|
|
IFDEF DEBUG
|
|
VxD_IDATA_SEG
|
|
db 0dh, 0ah, 'D_E_B_U_G===>'
|
|
db "&Name", '<===', 0dh, 0ah
|
|
VxD_IDATA_ENDS
|
|
ENDIF
|
|
|
|
VxD_LOCKED_DATA_SEG
|
|
|
|
PUBLIC Name&_DDB
|
|
Name&_DDB VxD_Desc_Block <,,Device_Num,Major_Ver,Minor_Ver,,"&Name",Init_Order,\
|
|
OFFSET32 Ctrl_Proc, V86_API_Offset, PM_API_Offset, \
|
|
,,Ref_Data_Offset,Serv_Tab_Offset, Serv_Tab_Len>
|
|
|
|
VxD_LOCKED_DATA_ENDS
|
|
|
|
ENDM
|
|
|
|
;BeginDoc ; comment out to make masm work ???
|
|
;******************************************************************************
|
|
; The Begin_Control_Dispatch macro is used for building a table for dispatching
|
|
; messages passed to the VxD_Control procedure. It is used with
|
|
; Control_Dispatch and End_Control_Dispatch. The only parameter is used to
|
|
; contruct the procedure label by adding "_Control" to the end (normally the
|
|
; device name is used i.e. VKD results in creating the procedure VKD_Control,
|
|
; this created procedure label must be included in Declare_Virtual_Device)
|
|
;
|
|
; An example of building a complete dispatch table:
|
|
;
|
|
; Begin_Control_Dispatch MyDevice
|
|
; Control_Dispatch Device_Init, MyDeviceInitProcedure
|
|
; Control_Dispatch Sys_VM_Init, MyDeviceSysInitProcedure
|
|
; Control_Dispatch Create_VM, MyDeviceCreateVMProcedure
|
|
; End_Control_Dispatch MyDevice
|
|
;
|
|
; (NOTE: Control_Dispatch can be used without Begin_Control_Dispatch, but
|
|
; then it is the programmer's responsibility for declaring a procedure
|
|
; in locked code (VxD_LOCKED_CODE_SEG) and returning Carry clear for
|
|
; any messages not processed. The advantage in using
|
|
; Begin_Control_Dispatch is when a large # of messages are processed by
|
|
; a device, because a jump table is built which will usually require
|
|
; less code space then the compares and jumps that are done when
|
|
; Control_Dispatch is used alone.
|
|
;
|
|
;==============================================================================
|
|
;EndDoc
|
|
Begin_Control_Dispatch MACRO VxD_Name, p1, p2
|
|
??_cd_low = 0FFFFFFFFh
|
|
??_cd_high = 0
|
|
|
|
BeginProc VxD_Name&_Control, p1, p2, LOCKED
|
|
ENDM
|
|
|
|
End_Control_Dispatch MACRO VxD_Name
|
|
LOCAL ignore, table
|
|
|
|
procoff MACRO num
|
|
IFDEF ??_cd_&&num
|
|
dd OFFSET32 ??_cd_&&num
|
|
ELSE
|
|
dd OFFSET32 ignore
|
|
ENDIF
|
|
ENDM
|
|
|
|
IF ??_cd_low EQ ??_cd_high
|
|
cmp eax, ??_cd_low
|
|
?merge <jz>,,,,,<??_cd_>, %(??_cd_low)
|
|
clc
|
|
ret
|
|
ELSE
|
|
IF ??_cd_low GT 0
|
|
sub eax, ??_cd_low
|
|
ENDIF ; ??cd_low GT 0
|
|
cmp eax, ??_cd_high - ??_cd_low + 1
|
|
jae short ignore
|
|
jmp [eax*4+table]
|
|
ignore:
|
|
clc ;; this is not redundant
|
|
ret
|
|
|
|
table label dword
|
|
REPT ??_cd_high - ??_cd_low + 1
|
|
procoff %(??_cd_low)
|
|
??_cd_low = ??_cd_low + 1
|
|
ENDM
|
|
ENDIF
|
|
|
|
EndProc VxD_Name&_Control
|
|
|
|
PURGE procoff
|
|
PURGE Begin_Control_Dispatch
|
|
PURGE Control_Dispatch
|
|
PURGE End_Control_Dispatch
|
|
ENDM
|
|
|
|
BeginDoc
|
|
;******************************************************************************
|
|
; The Control_Dispatch macro is used for dispatching based on message
|
|
; passed to the VxD_Control procedure. E.G.:
|
|
;
|
|
; Control_Dispatch Device_Init, MyDeviceInitProcedure
|
|
;
|
|
; For "C" control functions:
|
|
;
|
|
; Control_Dispatch Device_Init, MyDeviceInitProcedure, sCall, <arglst>
|
|
;
|
|
; The "callc" can be sCall, cCall or pCall depending on the calling
|
|
; convention. "arglst" is the list of registers to pass as parameters
|
|
; to "C" control procedure. The "C" control procedure returns VXD_SUCCESS
|
|
; or VXD_FAILURE and the carry flag gets set appropriately.
|
|
;
|
|
; (NOTE: Control_Dispatch can be used with Begin_Control_Dispatch and
|
|
; End_Control_Dispatch to create a jump table for dispatching messages,
|
|
; when a large # of messages are processed.)
|
|
;
|
|
;==============================================================================
|
|
EndDoc
|
|
Control_Dispatch MACRO Service, Procedure, callc, arglst
|
|
LOCAL Skip_Interseg_Jump
|
|
|
|
.errnz ?_LCODE, <Control_Dispatch must be in VxD_LOCKED_CODE_SEG.>
|
|
|
|
IFB <callc>
|
|
|
|
IFDEF ??_cd_low
|
|
Equate_Service MACRO Serv
|
|
??_cd_&&Serv equ Procedure
|
|
ENDM
|
|
|
|
Equate_Service %(Service)
|
|
|
|
IF Service LT ??_cd_low
|
|
??_cd_low = Service
|
|
ENDIF
|
|
IF Service GT ??_cd_high
|
|
??_cd_high = Service
|
|
ENDIF
|
|
|
|
PURGE Equate_Service
|
|
|
|
ELSE
|
|
cmp eax, Service
|
|
jz Procedure
|
|
ENDIF
|
|
|
|
ELSE ; ifb callc
|
|
|
|
cmp eax, Service
|
|
jne SHORT Skip_Interseg_Jump
|
|
callc Procedure, <arglst>
|
|
IF Service EQ PNP_NEW_DEVNODE
|
|
stc
|
|
ELSE
|
|
cmp eax,1
|
|
ENDIF
|
|
ret
|
|
Skip_Interseg_Jump:
|
|
|
|
ENDIF ; ifb callc
|
|
|
|
ENDM
|
|
*/
|
|
|
|
|
|
/******************************************************************************
|
|
* The following are the definitions for the "type of I/O" parameter passed
|
|
* to a I/O trap routine.
|
|
*****************************************************************************/
|
|
|
|
#define BYTE_INPUT 0x000
|
|
#define BYTE_OUTPUT 0x004
|
|
#define WORD_INPUT 0x008
|
|
#define WORD_OUTPUT 0x00C
|
|
#define DWORD_INPUT 0x010
|
|
#define DWORD_OUTPUT 0x014
|
|
|
|
#define OUTPUT_BIT 2
|
|
#define OUTPUT (1 << OUTPUT_BIT)
|
|
#define WORD_IO_BIT 3
|
|
#define WORD_IO (1 << WORD_IO_BIT)
|
|
#define DWORD_IO_BIT 4
|
|
#define DWORD_IO (1 << DWORD_IO_BIT)
|
|
|
|
#define STRING_IO_BIT 5
|
|
#define STRING_IO (1 << STRING_IO_BIT)
|
|
#define REP_IO_BIT 6
|
|
#define REP_IO (1 << REP_IO_BIT)
|
|
#define ADDR_32_IO_BIT 7
|
|
#define ADDR_32_IO (1 << ADDR_32_IO_BIT)
|
|
#define REVERSE_IO_BIT 8
|
|
#define REVERSE_IO (1 << REVERSE_IO_BIT)
|
|
|
|
#define IO_SEG_MASK 0x0FFFF0000 /* Use this to get segment */
|
|
#define IO_SEG_SHIFT 0x10 /* Must shift right this many */
|
|
|
|
|
|
/* ASM
|
|
BeginDoc
|
|
;******************************************************************************
|
|
;
|
|
; Dispatch_Byte_IO macro
|
|
;
|
|
; Dispatch_Byte_IO Byte_In_Proc, Byte_Out_Proc
|
|
;==============================================================================
|
|
EndDoc
|
|
Dispatch_Byte_IO MACRO In_Proc, Out_Proc
|
|
LOCAL Byte_IO
|
|
cmp ecx, Byte_Output
|
|
jbe SHORT Byte_IO
|
|
VMMJmp Simulate_IO
|
|
Byte_IO:
|
|
IFIDNI <In_Proc>, <Fall_Through>
|
|
je Out_Proc
|
|
ELSE
|
|
IFIDNI <Out_Proc>, <Fall_Through>
|
|
jb In_Proc
|
|
ELSE
|
|
je Out_Proc
|
|
jmp In_Proc
|
|
ENDIF
|
|
ENDIF
|
|
ENDM
|
|
|
|
BeginDoc
|
|
;******************************************************************************
|
|
;
|
|
; Emulate_Non_Byte_IO
|
|
;
|
|
; Emulate_Non_Byte_IO
|
|
;
|
|
;==============================================================================
|
|
EndDoc
|
|
Emulate_Non_Byte_IO MACRO
|
|
LOCAL Byte_IO
|
|
cmp ecx, Byte_Output
|
|
jbe SHORT Byte_IO
|
|
VMMJmp Simulate_IO
|
|
Byte_IO:
|
|
ENDM
|
|
*/
|
|
|
|
|
|
/* ASM
|
|
BeginDoc
|
|
;******************************************************************************
|
|
;
|
|
; Begin_VxD_IO_Table
|
|
;
|
|
; Example:
|
|
; Begin_VxD_IO_Table MyTableName
|
|
;
|
|
;==============================================================================
|
|
EndDoc
|
|
*/
|
|
|
|
|
|
struct VxD_IOT_Hdr {
|
|
USHORT VxD_IO_Ports;
|
|
};
|
|
|
|
struct VxD_IO_Struc {
|
|
USHORT VxD_IO_Port;
|
|
ULONG VxD_IO_Proc;
|
|
};
|
|
|
|
|
|
/* ASM
|
|
.ERRNZ SIZE VxD_IOT_Hdr - 2 ; Begin_VxD_IO_Table creates a 1 word count hdr
|
|
Begin_VxD_IO_Table MACRO Table_Name
|
|
PUBLIC Table_Name
|
|
Table_Name LABEL WORD
|
|
|
|
ifndef MASM6
|
|
IF2
|
|
IFNDEF Table_Name&_Entries
|
|
.err <No End_VxD_IO_Table for &Table_Name>
|
|
ENDIF
|
|
dw Table_Name&_Entries
|
|
ELSE
|
|
dw ?
|
|
ENDIF
|
|
ELSE ; MASM6 - skip the warning message - we'll get it anyway
|
|
dw Table_Name&_Entries
|
|
ENDIF ; MASM6
|
|
|
|
ENDM
|
|
|
|
.ERRNZ SIZE VxD_IO_Struc - 6 ; VxD_IO creates 6 byte I/O port entries
|
|
VxD_IO MACRO Port, Proc_Name
|
|
dw Port
|
|
dd OFFSET32 Proc_Name
|
|
ENDM
|
|
|
|
End_VxD_IO_Table MACRO Table_Name
|
|
|
|
IFNDEF Table_Name
|
|
.err <No Begin_VxD_IO_Table for &Table_Name>
|
|
ELSE
|
|
Table_Name&_Entries EQU (($-Table_Name)-2) / (SIZE VxD_IO_Struc)
|
|
IF Table_Name&_Entries LE 0
|
|
.err <Invalid number of port traps in &Table_Name>
|
|
ENDIF
|
|
ENDIF
|
|
ENDM
|
|
|
|
|
|
;******************************************************************************
|
|
;
|
|
; Push_Client_State takes an optional argument which if equal to the symbol
|
|
; USES_EDI saves code size by suppressing the preservation of the EDI register.
|
|
;
|
|
; Similarly, Pop_Client_State takes an optional argument which if equal to
|
|
; the symbol USES_ESI saves code size by suppressing the preservation of
|
|
; the ESI register.
|
|
;
|
|
;******************************************************************************
|
|
|
|
Push_Client_State MACRO Can_Trash_EDI
|
|
sub esp, SIZE Client_Reg_Struc
|
|
??_pushed = ??_pushed + SIZE Client_Reg_Struc
|
|
ifidni <Can_Trash_EDI>, <USES_EDI>
|
|
mov edi, esp
|
|
VMMCall Save_Client_State
|
|
else
|
|
push edi
|
|
lea edi, [esp+4]
|
|
VMMCall Save_Client_State
|
|
pop edi
|
|
endif
|
|
ENDM
|
|
|
|
Pop_Client_State MACRO Can_Trash_ESI
|
|
ifdifi <Can_Trash_ESI>, <USES_ESI>
|
|
push esi
|
|
lea esi, [esp+4]
|
|
VMMCall Restore_Client_State
|
|
pop esi
|
|
else
|
|
mov esi, esp
|
|
VMMCall Restore_Client_State
|
|
endif
|
|
add esp, SIZE Client_Reg_Struc
|
|
??_pushed = ??_pushed - SIZE Client_Reg_Struc
|
|
ENDM
|
|
|
|
BeginDoc
|
|
;******************************************************************************
|
|
;
|
|
; CallRet -- Call procedure and return. For debugging purposes only.
|
|
; If compiled with debugging then this will generate a call
|
|
; followed by a return. If non-debugging version then the
|
|
; specified label will be jumped to.
|
|
;
|
|
; PARAMETERS:
|
|
; Label_Name = Procedure to be called
|
|
;
|
|
; EXIT:
|
|
; Return from current procedure
|
|
;
|
|
;------------------------------------------------------------------------------
|
|
EndDoc
|
|
|
|
CallRet MACRO P1, P2
|
|
IFDEF DEBUG
|
|
IFIDNI <P1>, <SHORT>
|
|
call P2
|
|
ELSE
|
|
call P1
|
|
ENDIF
|
|
ret
|
|
ELSE
|
|
jmp P1 P2
|
|
ENDIF
|
|
ENDM
|
|
|
|
BeginDoc
|
|
;******************************************************************************
|
|
;
|
|
; VxDCallRet
|
|
; VMMCallRet -- CallRet for VxDCall and VMMCall.
|
|
;
|
|
;------------------------------------------------------------------------------
|
|
EndDoc
|
|
|
|
IFDEF DEBUG
|
|
|
|
VxDCallRet macro p:req
|
|
VxDCall p
|
|
ret
|
|
endm
|
|
|
|
VMMCallRet macro p:req
|
|
VMMCall p
|
|
ret
|
|
endm
|
|
|
|
ELSE ; RETAIL
|
|
|
|
VxDCallRet equ <VxDJmp>
|
|
VMMCallRet equ <VMMJmp>
|
|
|
|
ENDIF
|
|
|
|
|
|
; ebp offsets to segments pushed by PMode_Fault in Fault_Dispatch
|
|
PClient_DS equ WORD PTR -4
|
|
PClient_ES equ WORD PTR -8
|
|
PClient_FS equ WORD PTR -12
|
|
PClient_GS equ WORD PTR -16
|
|
|
|
|
|
;******************************************************************************
|
|
;
|
|
; Client_Ptr_Flat takes an optional third argument which if equal to the
|
|
; symbol USES_EAX saves code size by supressing the preservation of the
|
|
; EAX register. The USES_EAX flag is ignored if the destination register
|
|
; is itself EAX.
|
|
;
|
|
;******************************************************************************
|
|
|
|
Client_Ptr_Flat MACRO Reg_32, Cli_Seg, Cli_Off, Can_Trash_EAX
|
|
|
|
IFDIFI <Reg_32>, <EAX>
|
|
IFDIFI <Can_Trash_EAX>, <USES_EAX>
|
|
xchg Reg_32, eax
|
|
ENDIF
|
|
ENDIF
|
|
IFB <Cli_Off>
|
|
mov ax, (Client_&Cli_Seg * 100h) + 0FFh
|
|
ELSE
|
|
mov ax, (Client_&Cli_Seg * 100h) + Client_&Cli_Off
|
|
ENDIF
|
|
VMMCall Map_Flat
|
|
|
|
IFDIFI <Reg_32>, <EAX>
|
|
xchg Reg_32, eax
|
|
ENDIF
|
|
|
|
ENDM
|
|
|
|
;------------------------------------------------------------------------------
|
|
|
|
VxDint MACRO Int_Number
|
|
if (OPATTR Int_Number) AND 4
|
|
push Int_Number
|
|
else
|
|
push DWORD PTR Int_Number
|
|
endif
|
|
VMMCall Exec_VxD_Int
|
|
ENDM
|
|
|
|
VxDintMustComplete MACRO Int_Number
|
|
if (OPATTR Int_Number) AND 4
|
|
push Int_Number
|
|
else
|
|
push DWORD PTR Int_Number
|
|
endif
|
|
VMMCall _ExecVxDIntMustComplete
|
|
ENDM
|
|
|
|
Load_FS macro
|
|
VMMCall Load_FS_Service
|
|
endm
|
|
*/
|
|
|
|
/*XLATOFF*/
|
|
#define Load_FS VMMCall(Load_FS_Service)
|
|
/*XLATON*/
|
|
|
|
#endif // Not_VxD
|
|
|
|
|
|
/******************************************************************************
|
|
*
|
|
* The following equates are for flags sent to the real mode
|
|
* initialization portion of a device driver:
|
|
*
|
|
*****************************************************************************/
|
|
#define DUPLICATE_DEVICE_ID_BIT 0 /* loaded */
|
|
#define DUPLICATE_DEVICE_ID (1 << DUPLICATE_DEVICE_ID_BIT)
|
|
#define DUPLICATE_FROM_INT2F_BIT 1 /* loaded from INT 2F list */
|
|
#define DUPLICATE_FROM_INT2F (1 << DUPLICATE_FROM_INT2F_BIT)
|
|
#define LOADING_FROM_INT2F_BIT 2 /* in the INT 2F device list */
|
|
#define LOADING_FROM_INT2F (1 << LOADING_FROM_INT2F_BIT)
|
|
|
|
|
|
/******************************************************************************
|
|
*
|
|
* The following equates are used to indicate the result of the real mode
|
|
* initialization portion of a device driver:
|
|
*
|
|
*****************************************************************************/
|
|
|
|
#define DEVICE_LOAD_OK 0 /* load protected mode portion */
|
|
#define ABORT_DEVICE_LOAD 1 /* don't load protected mode portion */
|
|
#define ABORT_WIN386_LOAD 2 /* fatal-error: abort load of Win386 */
|
|
#define DEVICE_NOT_NEEDED 3 /* don't load protected mode portion */
|
|
/* b/c the driver's presence is not needed */
|
|
|
|
|
|
|
|
#define NO_FAIL_MESSAGE_BIT 15 /* set bit to suppress error message */
|
|
#define NO_FAIL_MESSAGE (1 << NO_FAIL_MESSAGE_BIT)
|
|
|
|
|
|
/******************************************************************************
|
|
*
|
|
* The following equates define the loader services available to the real-mode
|
|
* initialization portion of a device driver:
|
|
*
|
|
*****************************************************************************/
|
|
|
|
#define LDRSRV_GET_PROFILE_STRING 0 /* search SYSTEM.INI for string */
|
|
#define LDRSRV_GET_NEXT_PROFILE_STRING 1 /* search for next string */
|
|
#define LDRSRV_RESERVED 2 /* RESERVED */
|
|
#define LDRSRV_GET_PROFILE_BOOLEAN 3 /* search SYSTEM.INI for boolean */
|
|
#define LDRSRV_GET_PROFILE_DECIMAL_INT 4 /* search SYSTEM.INI for integer */
|
|
#define LDRSRV_GET_PROFILE_HEX_INT 5 /* search SYSTEM.INI for hex int */
|
|
#define LDRSRV_COPY_EXTENDED_MEMORY 6 /* allocate/init extended memory */
|
|
#define LDRSRV_GET_MEMORY_INFO 7 /* get info about machine memory */
|
|
|
|
/* Add the new loader services contiguously here */
|
|
|
|
/****** Registry services for Real mode init time *************
|
|
* The parameters for these are as defined in Windows.h for the
|
|
* corresponding Win Reg API and should be on Stack. These are
|
|
* C Callable except that the function no has to be in AX
|
|
* ************************************************************
|
|
*/
|
|
|
|
#define LDRSRV_RegOpenKey 0x100
|
|
#define LDRSRV_RegCreateKey 0x101
|
|
#define LDRSRV_RegCloseKey 0x102
|
|
#define LDRSRV_RegDeleteKey 0x103
|
|
#define LDRSRV_RegSetValue 0x104
|
|
#define LDRSRV_RegQueryValue 0x105
|
|
#define LDRSRV_RegEnumKey 0x106
|
|
#define LDRSRV_RegDeleteValue 0x107
|
|
#define LDRSRV_RegEnumValue 0x108
|
|
#define LDRSRV_RegQueryValueEx 0x109
|
|
#define LDRSRV_RegSetValueEx 0x10A
|
|
#define LDRSRV_RegFlushKey 0x10B
|
|
|
|
|
|
/*
|
|
* For the Copy_Extended_Memory service, the following types of memory can be
|
|
* requested:
|
|
*/
|
|
|
|
#define LDRSRV_COPY_INIT 1 /* memory discarded after init */
|
|
#define LDRSRV_COPY_LOCKED 2 /* locked memory */
|
|
#define LDRSRV_COPY_PAGEABLE 3 /* pageable memory */
|
|
|
|
/****************************************************************************
|
|
*
|
|
* Object types supported by the vxd loader
|
|
*
|
|
* Notes : Low bit of all CODE type objects should be set (VXDLDR uses this)
|
|
* Also Init type objects should be added to the second part of the
|
|
* list (which starts with ICODE_OBJ).
|
|
*
|
|
*****************************************************************************/
|
|
|
|
#define RCODE_OBJ -1
|
|
|
|
#define LCODE_OBJ 0x01
|
|
#define LDATA_OBJ 0x02
|
|
#define PCODE_OBJ 0x03
|
|
#define PDATA_OBJ 0x04
|
|
#define SCODE_OBJ 0x05
|
|
#define SDATA_OBJ 0x06
|
|
#define CODE16_OBJ 0x07
|
|
#define LMSG_OBJ 0x08
|
|
#define PMSG_OBJ 0x09
|
|
|
|
#define DBOC_OBJ 0x0B
|
|
#define DBOD_OBJ 0x0C
|
|
|
|
#define PLCODE_OBJ 0x0D
|
|
#define PPCODE_OBJ 0x0F
|
|
|
|
#define ICODE_OBJ 0x11
|
|
#define IDATA_OBJ 0x12
|
|
#define ICODE16_OBJ 0x13
|
|
#define IMSG_OBJ 0x14
|
|
|
|
|
|
struct ObjectLocation {
|
|
ULONG OL_LinearAddr ;
|
|
ULONG OL_Size ;
|
|
UCHAR OL_ObjType ;
|
|
} ;
|
|
|
|
#define MAXOBJECTS 25
|
|
|
|
/*****************************************************************************
|
|
*
|
|
* Device_Location structure
|
|
*
|
|
*****************************************************************************/
|
|
|
|
struct Device_Location_List {
|
|
ULONG DLL_DDB ;
|
|
UCHAR DLL_NumObjects ;
|
|
struct ObjectLocation DLL_ObjLocation[1];
|
|
};
|
|
|
|
|
|
/* ========================================================================= */
|
|
|
|
/*
|
|
* CR0 bit assignments
|
|
*/
|
|
#define PE_BIT 0 /* 1 = Protected Mode */
|
|
#define PE_MASK (1 << PE_BIT)
|
|
#define MP_BIT 1 /* 1 = Monitor Coprocessor */
|
|
#define MP_MASK (1 << MP_BIT)
|
|
#define EM_BIT 2 /* 1 = Emulate Math Coprocessor */
|
|
#define EM_MASK (1 << EM_BIT)
|
|
#define TS_BIT 3 /* 1 = Task Switch occured */
|
|
#define TS_MASK (1 << TS_BIT)
|
|
#define ET_BIT 4 /* 1 = 387 present, 0 = 287 present */
|
|
#define ET_MASK (1 << ET_BIT)
|
|
#define PG_BIT 31 /* 1 = paging enabled, 0 = paging disabled */
|
|
#define PG_MASK (1 << PG_BIT)
|
|
|
|
|
|
/*
|
|
* EFLAGs bit assignments
|
|
*/
|
|
#define CF_BIT 0
|
|
#define CF_MASK (1 << CF_BIT)
|
|
#define PF_BIT 2
|
|
#define PF_MASK (1 << PF_BIT)
|
|
#define AF_BIT 4
|
|
#define AF_MASK (1 << AF_BIT)
|
|
#define ZF_BIT 6
|
|
#define ZF_MASK (1 << ZF_BIT)
|
|
#define SF_BIT 7
|
|
#define SF_MASK (1 << SF_BIT)
|
|
#define TF_BIT 8
|
|
#define TF_MASK (1 << TF_BIT)
|
|
#define IF_BIT 9
|
|
#define IF_MASK (1 << IF_BIT)
|
|
#define DF_BIT 10
|
|
#define DF_MASK (1 << DF_BIT)
|
|
#define OF_BIT 11 /* Overflow flag */
|
|
#define OF_MASK (1 << OF_BIT)
|
|
#define IOPL_MASK 0x3000 /* IOPL flags */
|
|
#define IOPL_BIT0 12
|
|
#define IOPL_BIT1 13
|
|
#define NT_BIT 14 /* Nested task flag */
|
|
#define NT_MASK (1 << NT_BIT)
|
|
#define RF_BIT 16 /* Resume flag */
|
|
#define RF_MASK (1 << RF_BIT)
|
|
#define VM_BIT 17 /* Virtual Mode flag */
|
|
#define VM_MASK (1 << VM_BIT)
|
|
#define AC_BIT 18 /* Alignment check */
|
|
#define AC_MASK (1 << AC_BIT)
|
|
#define VIF_BIT 19 /* Virtual Interrupt flag */
|
|
#define VIF_MASK (1 << VIF_BIT)
|
|
#define VIP_BIT 20 /* Virtual Interrupt pending */
|
|
#define VIP_MASK (1 << VIP_BIT)
|
|
|
|
|
|
|
|
/* ASM
|
|
;------------------------------------------------------------------------------
|
|
;
|
|
; Temporary MASM macros (to be removed when supported by MASM)
|
|
;
|
|
;------------------------------------------------------------------------------
|
|
|
|
IFDEF MASM6
|
|
loopde EQU <looped>
|
|
loopdne EQU <loopned>
|
|
loopdz EQU <loopzd>
|
|
loopdnz EQU <loopnzd>
|
|
ELSE
|
|
loopd EQU <loop>
|
|
loopde EQU <loope>
|
|
loopdne EQU <loopne>
|
|
loopdz EQU <loopz>
|
|
loopdnz EQU <loopnz>
|
|
ENDIF
|
|
*/
|
|
|
|
|
|
/******************************************************************************
|
|
* PAGE TABLE EQUATES
|
|
*****************************************************************************/
|
|
|
|
|
|
#define P_SIZE 0x1000 /* page size */
|
|
|
|
/******************************************************************************
|
|
*
|
|
* PAGE TABLE ENTRY BITS
|
|
*
|
|
*****************************************************************************/
|
|
|
|
#define P_PRESBIT 0
|
|
#define P_PRES (1 << P_PRESBIT)
|
|
#define P_WRITEBIT 1
|
|
#define P_WRITE (1 << P_WRITEBIT)
|
|
#define P_USERBIT 2
|
|
#define P_USER (1 << P_USERBIT)
|
|
#define P_ACCBIT 5
|
|
#define P_ACC (1 << P_ACCBIT)
|
|
#define P_DIRTYBIT 6
|
|
#define P_DIRTY (1 << P_DIRTYBIT)
|
|
|
|
#define P_AVAIL (P_PRES+P_WRITE+P_USER) /* avail to user & present */
|
|
|
|
/****************************************************
|
|
*
|
|
* Page types for page allocator calls
|
|
*
|
|
***************************************************/
|
|
|
|
#define PG_VM 0
|
|
#define PG_SYS 1
|
|
#define PG_RESERVED1 2
|
|
#define PG_PRIVATE 3
|
|
#define PG_RESERVED2 4
|
|
#define PG_RELOCK 5 /* PRIVATE to MMGR */
|
|
#define PG_INSTANCE 6
|
|
#define PG_HOOKED 7
|
|
#define PG_IGNORE 0xFFFFFFFF
|
|
|
|
/****************************************************
|
|
*
|
|
* Definitions for the access byte in a descriptor
|
|
*
|
|
***************************************************/
|
|
|
|
/*
|
|
* Following fields are common to segment and control descriptors
|
|
*/
|
|
#define D_PRES 0x080 /* present in memory */
|
|
#define D_NOTPRES 0 /* not present in memory */
|
|
|
|
#define D_DPL0 0 /* Ring 0 */
|
|
#define D_DPL1 0x020 /* Ring 1 */
|
|
#define D_DPL2 0x040 /* Ring 2 */
|
|
#define D_DPL3 0x060 /* Ring 3 */
|
|
|
|
#define D_SEG 0x010 /* Segment descriptor */
|
|
#define D_CTRL 0 /* Control descriptor */
|
|
|
|
#define D_GRAN_BYTE 0x000 /* Segment length is byte granular */
|
|
#define D_GRAN_PAGE 0x080 /* Segment length is page granular */
|
|
#define D_DEF16 0x000 /* Default operation size is 16 bits */
|
|
#define D_DEF32 0x040 /* Default operation size is 32 bits */
|
|
|
|
|
|
/*
|
|
* Following fields are specific to segment descriptors
|
|
*/
|
|
#define D_CODE 0x08 /* code */
|
|
#define D_DATA 0 /* data */
|
|
|
|
#define D_X 0 /* if code, exec only */
|
|
#define D_RX 0x02 /* if code, readable */
|
|
#define D_C 0x04 /* if code, conforming */
|
|
|
|
#define D_R 0 /* if data, read only */
|
|
#define D_W 0x02 /* if data, writable */
|
|
#define D_ED 0x04 /* if data, expand down */
|
|
|
|
#define D_ACCESSED 1 /* segment accessed bit */
|
|
|
|
|
|
/*
|
|
* Useful combination access rights bytes
|
|
*/
|
|
#define RW_DATA_TYPE (D_PRES+D_SEG+D_DATA+D_W)
|
|
#define R_DATA_TYPE (D_PRES+D_SEG+D_DATA+D_R)
|
|
#define CODE_TYPE (D_PRES+D_SEG+D_CODE+D_RX)
|
|
|
|
#define D_PAGE32 (D_GRAN_PAGE+D_DEF32) /* 32 bit Page granular */
|
|
|
|
/*
|
|
* Masks for selector fields
|
|
*/
|
|
#define SELECTOR_MASK 0xFFF8 /* selector index */
|
|
#define SEL_LOW_MASK 0xF8 /* mask for low byte of sel indx */
|
|
#define TABLE_MASK 0x04 /* table bit */
|
|
#define RPL_MASK 0x03 /* privilige bits */
|
|
#define RPL_CLR (~RPL_MASK) /* clear ring bits */
|
|
|
|
#define IVT_ROM_DATA_SIZE 0x500
|
|
|
|
/*XLATOFF*/
|
|
|
|
#ifndef Not_VxD
|
|
|
|
#define ENABLE_INTERRUPTS() {__asm sti}
|
|
#define DISABLE_INTERRUPTS() {__asm cli}
|
|
|
|
#define SAVE_FLAGS(flags) {\
|
|
{__asm pushfd}; \
|
|
{__asm pop flags}}
|
|
|
|
#define RESTORE_FLAGS(flags) {\
|
|
{__asm push flags}; \
|
|
{__asm popfd}}
|
|
|
|
#define IO_Delay() {\
|
|
{__asm _emit 0xeb}; \
|
|
{__asm _emit 0x00}}
|
|
|
|
#define Touch_Register(Register) {_asm xor Register, Register}
|
|
|
|
typedef DWORD HEVENT;
|
|
|
|
#define VMM_GET_DDB_NAMED 0
|
|
|
|
#pragma warning (disable:4209) // turn off redefine warning (with basedef.h)
|
|
|
|
typedef ULONG HTIMEOUT; // timeout handle
|
|
typedef ULONG CMS; // count of milliseconds
|
|
|
|
#pragma warning (default:4209) // turn on redefine warning (with basedef.h)
|
|
|
|
typedef DWORD VMM_SEMAPHORE;
|
|
|
|
|
|
#ifndef WANTVXDWRAPS
|
|
|
|
WORD VXDINLINE
|
|
Get_VMM_Version()
|
|
{
|
|
WORD w;
|
|
VMMCall(Get_VMM_Version);
|
|
_asm mov [w], ax
|
|
return(w);
|
|
}
|
|
|
|
PVOID VXDINLINE
|
|
_HeapAllocate(ULONG Bytes, ULONG Flags)
|
|
{
|
|
PVOID p;
|
|
Touch_Register(eax)
|
|
Touch_Register(ecx)
|
|
Touch_Register(edx)
|
|
_asm push [Flags]
|
|
_asm push [Bytes]
|
|
VMMCall(_HeapAllocate)
|
|
_asm add esp, 8
|
|
_asm mov [p], eax
|
|
return(p);
|
|
}
|
|
|
|
ULONG VXDINLINE
|
|
_HeapFree(PVOID Address, ULONG Flags)
|
|
{
|
|
ULONG ul;
|
|
Touch_Register(eax)
|
|
Touch_Register(ecx)
|
|
Touch_Register(edx)
|
|
_asm push [Flags]
|
|
_asm push [Address]
|
|
VMMCall(_HeapFree)
|
|
_asm add esp, 8
|
|
_asm mov [ul], eax
|
|
return(ul);
|
|
}
|
|
|
|
HEVENT VXDINLINE
|
|
Call_Global_Event(void (__cdecl *pfnEvent)(), ULONG ulRefData)
|
|
{
|
|
HEVENT hevent;
|
|
_asm mov edx, [ulRefData]
|
|
_asm mov esi, [pfnEvent]
|
|
VMMCall(Call_Global_Event)
|
|
_asm mov [hevent], esi
|
|
return(hevent);
|
|
}
|
|
|
|
HEVENT VXDINLINE
|
|
Schedule_Global_Event(void (__cdecl *pfnEvent)(), ULONG ulRefData)
|
|
{
|
|
HEVENT hevent;
|
|
_asm mov edx, [ulRefData]
|
|
_asm mov esi, [pfnEvent]
|
|
VMMCall(Schedule_Global_Event)
|
|
_asm mov [hevent], esi
|
|
return(hevent);
|
|
}
|
|
|
|
void VXDINLINE
|
|
Cancel_Global_Event( HEVENT hevent )
|
|
{
|
|
_asm mov esi, hevent
|
|
VMMCall( Cancel_Global_Event );
|
|
}
|
|
|
|
HVM VXDINLINE
|
|
Get_Sys_VM_Handle(VOID)
|
|
{
|
|
HVM hvm;
|
|
Touch_Register(ebx)
|
|
VxDCall(Get_Sys_VM_Handle);
|
|
_asm mov [hvm], ebx
|
|
return(hvm);
|
|
}
|
|
|
|
VOID VXDINLINE
|
|
Fatal_Error_Handler(PCHAR pszMessage, DWORD dwExitFlag)
|
|
{
|
|
_asm mov esi, [pszMessage]
|
|
_asm mov eax, [dwExitFlag]
|
|
VMMCall(Fatal_Error_Handler);
|
|
}
|
|
|
|
VMM_SEMAPHORE VXDINLINE
|
|
Create_Semaphore(LONG lTokenCount)
|
|
{
|
|
VMM_SEMAPHORE vmm_semaphore;
|
|
_asm mov ecx, [lTokenCount]
|
|
VMMCall(Create_Semaphore)
|
|
_asm cmc
|
|
_asm sbb ecx, ecx
|
|
_asm and eax, ecx
|
|
_asm mov [vmm_semaphore], eax
|
|
return(vmm_semaphore);
|
|
}
|
|
|
|
void VXDINLINE
|
|
Destroy_Semaphore(VMM_SEMAPHORE vsSemaphore)
|
|
{
|
|
_asm mov eax, [vsSemaphore]
|
|
VMMCall(Destroy_Semaphore)
|
|
}
|
|
|
|
void VXDINLINE
|
|
Signal_Semaphore(VMM_SEMAPHORE vsSemaphore)
|
|
{
|
|
_asm mov eax, [vsSemaphore]
|
|
VMMCall(Signal_Semaphore)
|
|
}
|
|
|
|
void VXDINLINE
|
|
Wait_Semaphore(VMM_SEMAPHORE vsSemaphore, DWORD dwFlags)
|
|
{
|
|
_asm mov eax, [vsSemaphore]
|
|
_asm mov ecx, [dwFlags]
|
|
VMMCall(Wait_Semaphore)
|
|
}
|
|
|
|
HVM VXDINLINE
|
|
Get_Execution_Focus(void)
|
|
{
|
|
HVM hvm;
|
|
Touch_Register(ebx)
|
|
VMMCall(Get_Execution_Focus)
|
|
_asm mov [hvm], ebx
|
|
return(hvm);
|
|
}
|
|
|
|
void VXDINLINE
|
|
Begin_Critical_Section(ULONG Flags)
|
|
{
|
|
_asm mov ecx, [Flags]
|
|
VMMCall(Begin_Critical_Section)
|
|
}
|
|
|
|
void VXDINLINE
|
|
End_Critical_Section(void)
|
|
{
|
|
VMMCall(End_Critical_Section)
|
|
}
|
|
|
|
void VXDINLINE
|
|
Fatal_Memory_Handler(void)
|
|
{
|
|
VMMCall(Fatal_Memory_Error);
|
|
}
|
|
|
|
void VXDINLINE
|
|
Begin_Nest_Exec(void)
|
|
{
|
|
VMMCall(Begin_Nest_Exec)
|
|
}
|
|
|
|
void VXDINLINE
|
|
End_Nest_Exec(void)
|
|
{
|
|
VMMCall(End_Nest_Exec)
|
|
}
|
|
|
|
void VXDINLINE
|
|
Resume_Exec(void)
|
|
{
|
|
VMMCall(Resume_Exec)
|
|
}
|
|
|
|
HTIMEOUT VXDINLINE
|
|
Set_VM_Time_Out(void (*pfnTimeout)(), CMS cms, ULONG ulRefData)
|
|
{
|
|
HTIMEOUT htimeout;
|
|
_asm mov eax, [cms]
|
|
_asm mov edx, [ulRefData]
|
|
_asm mov esi, [pfnTimeout]
|
|
VMMCall(Set_VM_Time_Out)
|
|
_asm mov [htimeout], esi
|
|
return(htimeout);
|
|
}
|
|
|
|
HTIMEOUT VXDINLINE
|
|
Set_Global_Time_Out(void (__cdecl *pfnTimeout)(), CMS cms, ULONG ulRefData)
|
|
{
|
|
HTIMEOUT htimeout;
|
|
_asm mov eax, [cms]
|
|
_asm mov edx, [ulRefData]
|
|
_asm mov esi, [pfnTimeout]
|
|
VMMCall(Set_Global_Time_Out)
|
|
_asm mov [htimeout], esi
|
|
return(htimeout);
|
|
}
|
|
|
|
void VXDINLINE
|
|
Cancel_Time_Out(HTIMEOUT htimeout)
|
|
{
|
|
_asm mov esi, htimeout
|
|
VMMCall(Cancel_Time_Out)
|
|
}
|
|
|
|
|
|
void VXDINLINE
|
|
Update_System_Clock(ULONG msElapsed)
|
|
{
|
|
__asm mov ecx,[msElapsed]
|
|
VMMCall(Update_System_Clock)
|
|
}
|
|
|
|
void VXDINLINE
|
|
Enable_Touch_1st_Meg(void)
|
|
{
|
|
VMMCall(Enable_Touch_1st_Meg)
|
|
}
|
|
|
|
void VXDINLINE
|
|
Disable_Touch_1st_Meg(void)
|
|
{
|
|
VMMCall(Disable_Touch_1st_Meg)
|
|
}
|
|
|
|
void VXDINLINE
|
|
Out_Debug_String(char *psz)
|
|
{
|
|
__asm pushad
|
|
__asm mov esi, [psz]
|
|
VMMCall(Out_Debug_String)
|
|
__asm popad
|
|
}
|
|
|
|
void VXDINLINE
|
|
Queue_Debug_String(char *psz, ULONG ulEAX, ULONG ulEBX)
|
|
{
|
|
_asm push esi
|
|
_asm push [ulEAX]
|
|
_asm push [ulEBX]
|
|
_asm mov esi, [psz]
|
|
VMMCall(Queue_Debug_String)
|
|
_asm pop esi
|
|
}
|
|
|
|
#ifdef WIN40SERVICES
|
|
|
|
HTIMEOUT VXDINLINE
|
|
Set_Async_Time_Out(void (*pfnTimeout)(), CMS cms, ULONG ulRefData)
|
|
{
|
|
HTIMEOUT htimeout;
|
|
_asm mov eax, [cms]
|
|
_asm mov edx, [ulRefData]
|
|
_asm mov esi, [pfnTimeout]
|
|
VMMCall(Set_Async_Time_Out)
|
|
_asm mov [htimeout], esi
|
|
return(htimeout);
|
|
}
|
|
|
|
VXDINLINE struct VxD_Desc_Block *
|
|
VMM_Get_DDB(WORD DeviceID, PCHAR Name)
|
|
{
|
|
struct VxD_Desc_Block *p;
|
|
_asm movzx eax, [DeviceID]
|
|
_asm mov edi, [Name]
|
|
VMMCall(Get_DDB);
|
|
_asm mov [p], ecx
|
|
return(p);
|
|
}
|
|
|
|
DWORD VXDINLINE
|
|
VMM_Directed_Sys_Control(struct VxD_Desc_Block *DDB, DWORD SysControl, DWORD rEBX, DWORD rEDX, DWORD rESI, DWORD rEDI)
|
|
{
|
|
DWORD dw;
|
|
_asm mov eax, [SysControl]
|
|
_asm mov ebx, [rEBX]
|
|
_asm mov ecx, [DDB]
|
|
_asm mov edx, [rEDX]
|
|
_asm mov esi, [rESI]
|
|
_asm mov edi, [rEDI]
|
|
VMMCall(Directed_Sys_Control);
|
|
_asm mov [dw], eax
|
|
return(dw);
|
|
}
|
|
|
|
void VXDINLINE
|
|
_Trace_Out_Service(char *psz)
|
|
{
|
|
__asm push psz
|
|
VMMCall(_Trace_Out_Service)
|
|
}
|
|
|
|
void VXDINLINE
|
|
_Debug_Out_Service(char *psz)
|
|
{
|
|
__asm push psz
|
|
VMMCall(_Debug_Out_Service)
|
|
}
|
|
|
|
void VXDINLINE
|
|
_Debug_Flags_Service(ULONG flags)
|
|
{
|
|
__asm push flags
|
|
VMMCall(_Debug_Flags_Service)
|
|
}
|
|
|
|
void VXDINLINE _cdecl
|
|
_Debug_Printf_Service(char *pszfmt, ...)
|
|
{
|
|
__asm lea eax,(pszfmt + 4)
|
|
__asm push eax
|
|
__asm push pszfmt
|
|
VMMCall(_Debug_Printf_Service)
|
|
__asm add esp, 2*4
|
|
}
|
|
|
|
#endif // WIN40SERVICES
|
|
|
|
#endif // WANTVXDWRAPS
|
|
|
|
#endif // Not_VxD
|
|
|
|
/*XLATON*/
|
|
|
|
#endif /* _VMM_ */
|