Windows以它一致的图形用户界面、完善的内存管理等特点,已被广大用户所接受,但在Windows下,计算机已运行在保护模式,虽然Windows已提供了一整套的内存管理函数,用户还是不可访问指定地址的内存,致使许多中、小型企业开发的计算机插件无法在Windows下使用,因为编写驱动程序是很费事的。他们仅希望能直接读、写电路板所占的内存即可。本文介绍的是在Windows3.1下,用BorlandC++3.1编程环境,实现用户对内存直接读写的方法,读者不难将它移植到其它编程语言。
1.利用Windows提供的各个段选择符标号,在C源程序中将选择符标号说明为外部变量。其对照表如下:
表1
2.利用宏定义MAKELONG(段内偏移量,&段选择符标号)即可得到一个长指针。
3.经上述方法得到的指针与C语言中定义的指针是一样的。
4.如果使用的段是D000H或E000H或其它上位内存时,需改动Windows的系统配置文件SYSTEM.INI,利用字符编辑器,在[386Enh]小节中加入EmmExclude=xxxxyyyy一行,禁止Windows使用这段存储空间。值xxxx和yyyy是16位内存范围。如开发的插件占用D000H段的64K,则应加入这样一行:EmmExclude=d000-dfff。
利用这一方法,已使我们自己开发的电路板(占用D000H段),在Windows控制下成功地运行了。考虑到读者没有相应的硬件,这里以读取中断向量、计算机ROM的制造时间和对DOS用户通讯区的读写为例,详见以下程序。对于DOS用户通讯区的内容,读者可用DOS的DEBUG程序检查(D命令)和修改(E命令)。DOS的用户通讯区在0040∶00F0H处开始,共计16个字节。在修改时请注意:从0040∶00F0H开始存放可显示ASCII码字符串,并以0结尾。
;MEMORY.DEF模块定义文件
NAME
Memory
DESCRIPTION’demonstrateanintergratedmenu’
EXETYPEWINDOWS
STUB’WINSTUB.EXE’
CODEPRELOADMOVEABLEDISCARDABLE
DATAPRELOADMOVEABLEMULTIPLE
HEAPSIZE1024
STACKSIZE8192
EXPORTSWndProc
/*----------
MEMORY.RC
直接读写内存的资源定义文件
----------*/
#include"memory.h"
MemoryMenuMENU
BEGIN
POPUP"取中断向量[&I]"
BEGIN
MENUITEM"int0&3h",IDM-INT03H
MENUITEM"int&10h",IDM-INT10H
MENUITEM"int&21h",IDM-INT21H
END
POPUP"ROM区域[&R]"
BEGIN
MENUITEM"制造时间[&T]",
IDM-TIME
MENUITEM"取用户通讯区[&U]",IDM-READ
MENUITEM"存用户通讯区[&S]",IDM-WRITE
END
MENUITEM"“a退出[&X]",
IDM-EXIT
END
/*----------
MEMORY.H
直接读写内存的头文件
----------*/
#defineIDM-INT03H101
#defineIDM-INT10H102
#defineIDM-INT21H103
#defineIDM-TIME201
#defineIDM-READ202
#defineIDM-WRITE203
#defineIDM-EXIT300
/*----------
MEMORY.C
用户对固定内存直接读写的表演程序
----------*/
#includewindows.h
#include"memory.h"
#includedos.h
intPASCALWinMain(HANDLE,HANDLE,LPSTR,int);
longFARPASCALWndProc(HWND,UINT,UINT,LONG);
/*----WinMain()----*/
intPASCALWinMain(HANDLEhInstance,HANDLEhPrevInstance
,LPSTRlpszCmdLine,intnCmdShow)
{
MSG
msg;
HWNDhWnd;
WNDCLASSwndclass;
if(!hPrevInstance)
{
wndclass.style=CS-HREDRAW|CS-VREDRAW;
wndclass.lpfnWndProc=WndProc;
wndclass.cbClsExtra=0;
wndclass.cbWndExtra=0;
wndclass.hInstance=hInstance;
wndclass.hIcon=LoadIcon(NULL,IDI-APPLICATION);
wndclass.hCursor=LoadCursor(NULL,IDC-ARROW);
wndclass.hbrBackground=GetStockObject(WHITE-BRUSH);
wndclass.lpszMenuName="MemoryMenu";
wndclass.lpszClassName="直接读写存储器";
if(!RegisterClass(&wndclass))
returnFALSE;
}
hWnd=CreateWindow(
"直接读写存储器",
"直接读写存储器",
WS-OVERLAPPEDWINDOW,
CW-USEDEFAULT,
CW-USEDEFAULT,
CW-USEDEFAULT,
CW-USEDEFAULT,
NULL,
NULL,
hInstance,
NULL);
if(!hWnd)
returnFALSE;
ShowWindow(hWnd,nCmdShow);
UpdateWindow(hWnd);
while(GetMessage(&msg,N
ULL,0,0))