837页

writeln('TRACING Current Buffer===');

holdup;

bcbtrc(cvtbase^.curbfr);

writeln;

holdup ;

writeln('TRACING       Buffer Chain===');

holdup ;

bCDtrc(Cvtbase^bfrchn);

END.

CVT3.PAS是用Turbo Pascal语言写成的。从5.0版开始,该程序已在任何版本上

(包括版本7.0)得到了验证并编译成功。这份用于VCT3.PAS的列表清单虽然初看上去

令人生畏,但实质上,它也不过如此。这一列表是由几个相当简单的过程拼凑起来的。其

中,前几页只是简单地定义了所有的CVT结构,以便CVT3.PAS能够引用它们。接下来,

该列表定义了许多实用程序功能,以协助显示输出。在这些实用程序功能,出现在该列表

中的是一些跟踪功能,用于监视每一个主要结构。最后,将上述所有的内容拼接在一起,便

构成了主程序。下面不妨让我们来一点一点地详细探讨一下。

数据结构

CVT3程序的开始是几种特殊的数据类型,其中,一种用于每个CVT结构,另一种则

作为指向cVT结构的指针。因此,BCB把缓冲控制块定义成了记录(等效于C语言中的

结构),并且bcptr定义了一个指向这种记录的指针。

同样,mcb和mcbptr定义的是内存控制块结构;dpb和dpbptr定义着驱动器参数块

结构;chn和chptr定义着用于DCB和FCB链中的链路头;dcb和dcbptr定义着设备控制

块的格式(也可供系统FCB使用);ldt和ldtptr定义着逻辑驱动器表;并且cvt定义的是

CVT。

余下的数据类型被分别定义为dtstr(用于报告日期和时间的字符串大小),pstrg(在

报告指针值时使用的大小和memst(用于报告内存使用情况的大小)。这里所以定义了这

三种数据类型,是因为Turbo Pascal考虑到仅有两种不同string[8]定义过于笼统;而定义

了这些特殊的数据类型不仅允许编译程序的类型检查能正常地运转,并且还公开了如何

使用更短的字符串。

定义了这些特殊的数据类型,便可依次用它们来推断出该程序使用的其它全局工作

变量:cvtbase是将作为CVT的基址而被返回的指针;curbuf、curmcb、curdpb、curchn、cur-

dcb、curfcb和curldt是指向种种被跟踪的当前指针;余下的变量则可用来报告生成进程

的情况。

实用程序功能

定义了数据类型和变量之后,该程序接着定义的是许多实用程序功能。hexn用来传

送一个单字节,并把该字节的低4位转换成一个十六进制数字,以便在返回时可将它作为

一个ASCII字符。接着,hexb分两次使用hexn(在第一次调用时转换4个位),以便把一个

完整的字节转换成2个字符的字符串;hexw按照同样的方法来使用hexb,以把一个16

位的字转换成一个16进制数字的字符串,并且hexl再一次使用这种方法来转换一个32

838页

位值。当某种方法比较简明有效时,就应使用这样的方法。

当全局PawsFlag出错时,在跟踪进程期间每产生一个输出行后就调用的进程holdup

将不会起任何作用。这样,如果把报告重定向给打印机或文件,系统便允许继续进行输出。

如果PawsFlag是正确的,则holdup会增加行计数lctr,并且,如果lctr值大于23(看起来

较舒服的完整屏幕),holdup便会复位lctr,并等待在返回之前按下Enter键。

功能xp是一种较为特殊的功能,它通过使用hexw来进行2进制至16进制的指针

转换;从而把指针值转换成常规的ASCII形式,即SSSS:OOOO。

最后一个进程为dmp,dmp是一个通用的建成块,可将它用在任何地方以创建一个

类似于DEBUG的且可由任意16个连续字节(可采用十六进制格式和AsCII格式)组成

的列表。它接收一个远指针作为输入,并接着把指针值写给屏幕,在这里,指针值为采用十

六进制格式的16个字节和同样的16个被转换成了ASCII的字节(除掉了高位,并且把控

制字符转换成了“.”,以防混淆)。以上所介绍的进程大多数可以直接或间接地通过dmp

来使用。此外报告进程使用dmp来显示内存区域如缓冲区的内容。

跟踪DPBs

过程dpbtrc能实现两方面的功能,即跟踪驱动器参数块并报告驱动器参数块的内

容。它定义了一个局部过程dpbrpt来实现报告功能。不过,只有当其它例程不能使用该过

程时,才能从dpbtrc内调用dpbrpt;此外,能否使用这样一种“局部的”过程是区别Pascal

语言和C语言的重要标志之一。

dpbrpt过程是在正确地设置了curdpt指针的基础上通过把驱动器代码转换成字母

及其它内容,来把数从指针指定的DPB格式中转换成更适合于显示的描述格式。其中一

些显示采用十六进制格式,并且还有一些输出采用十进制,所有这些的目的是增强报告的

可读性。因此,也可以把这一过程看作一个裸骨式的“专家系统”,因为它把DPB格式的内

容转换成了人们容易理解的且十分浅显易懂的语言。

接下来的是dpbrpt过程,该过程是通过设置一个指向传入的指针值(从CVT中传

入)的全局指针currdpb而开始的。然后,它把局部变量ofsv设置为0;这样,便保存了每一

个记录内的下一个DPB指针的偏移值;当这个值变成了FFFFh时,跟踪循环便会终止。

设置了上述变量之后,该过程便打印出一个标题行,并进入由SV控制的跟踪循环。

该循环过程中的每一步骤上,ofsv的值都被改变,该过程会调用dpbrpt来报告当前DPB

中所保存的每一件东西,并把curdpb设置成在下一个DPB指针内找到的值。在ofsv未变

成FFFFh之前,该循环会一直进行下去,以便报告出所有有效的DPB。在该过程中,每当

屏幕显示出内容时,holdup便可以创造一次等待,以便于用户有时间读清屏幕上的内容。

在显示完了最后一个DPB报告之后,dpbtrc便会通过“write(# 12)”把一个换页字符

送给它的输出。这一行只有CRT上显示一个图形字符;但是,如果是把输出发送给打印

机,该行将强迫下一个跟踪功能的输出出现在新的页面上。在返回到主调用例程之前,大

多跟踪过程能迅速地实现上述过程。

跟踪BCBs

过程bcbtrc跟踪并报告缓冲控制块。与dpbtrc一样,该过程也包括一个局部报告功

839页

能,其专用名为bcbrpt。

与dpbrpt一样,bcbrpt也把来自于缓冲区的信息转换成更为通用的格式,并在屏幕

上显示出这一信息。接着,它在返回到bcbrc之前,使用dmp来显示出缓冲器的全部内容。

与dpbtrc一样,bcbtrc也是一种必不可少的外围程序,它首先初始化与之自身相关的

变量,接着通过调用bcbrpt来循环,直到它到达缓冲链的末端。当到达缓冲链的末端时,

该过程将输出一个换页,并返回到其调用者处。

但是,与dpbtrc不一样,bcbtrc不能使用全局变量来保存其用于bcbrpt的指针,因为

系统是为两个不同的链而调用bcbtrc的。确切地讲,该指针是作为一个独立参数而被传送

给上述的每一个过程。

DCBs

跟踪并报告设备控制块和系统FCBs(与设备控制块共享同一结构)的过程较为复杂

一些。除了定义它的局部报告例程-dcbrpt外,该过程还定义了一对功能,即f_tm(把

保存在DCB中的时间值转换成常规显示格式)和f_dt(对日期值进行同样的转换)。

f_tm和f_dt都可作为我们通常称之为“暴力”编程的直接例子。例如,在f_tm中,输

入时间参数(按照DOS指定的文件目录格式而被紧紧地压缩过)将会根据请求而被移位

并被屏蔽为每小时一个字节,并且,随后进行的DIV和MOD操作会把该字节转换成一个

两位十进制值(加在这里的值48是ASCII字符0的二进制值)。对于分和秒来说,处理过

程是相同的。

这样,当系统把被压缩的值扩展成为类型dtstr字符串之后,该字符串就会作为该功

能的值而被返回。f_dt功能对月、日和年值所采用的处理方式也是相同的。

由于DCB可以保存有关文件或设备的信息,并且根据保存的是文件还是设备,对信

息的描述是不相同的,因此,dcbrpt相对要复杂一些。它包括用来确定每个DCB引用的是

文件还是设备的代码以及用于相应地改变报告的判断结构。

另一个复杂因素是将DCB按次序组合成链路的方式,以及如何通过指针将这些链路

链接起来。为了便于跟踪,系统会传送给dcbrpt一个计数(n),告诉它在返回到dcbtrc之

前,应报告出多少个DCBs,dcbtrc例程维护,链与链之间的链接。由于dcbrpt能引用DCB

或FCB,因此,传送给它的合适字符串应为t—一种类型dtstr参数。

当最终到达dcbtrc(在这里,全局和局部变量都被初始化)的起始处时,dcbtrs几乎是

从天而降。与在前面介绍的跟踪例程中一样,它也是沿着链路往下循环,直到它到达链的

末端,即nexLnk指针中的偏移值FFFFh指定的地方。

跟踪MCB链

MCB跟踪程序过程-memtrc采用的是目前较为通用的方式,即一种局部报告过

程memrpt,并且该过程含有一种局部功能memu。但是,这些与我们曾见到的过程要短得

多。

memu功能可解释每个MCB控制的内存块的第一个字符,以准确地推猜出该内存块

的用途。由于所有程序都开始于PSP,并且PSP依次通过字节CDh和20h(用于Int 20h

的操作码)来开始,因此,如果memu发现CDh为第一个字节,它就会推测到程序是在这

840页

里。如果程序不在这里,并且如果第一个字节时ASCII中的一个大字字母,memu就会推

测所遇到的是一个环境块;否则,将见到的内容报告为一个保存数据。memu将描述字串

符之一作为它的值而返回。

指定给memrpt过程的参数块有三个,即5、0和9,它们分别为大小,拥有者和第一个

字节的地址。如果拥有者为0000,表明内存块是自由的内存块;否则,它就是为其保留着

块的进程的进程ID。

memtrc例程根据主程序传送给它的值来初始化cumcb全局指针,显示报告标题,并

接着进入循环。该循环为当前MCB调用memrpt,根据当前段大小计算出下一个MCB段

值,并把curmcb设置给这一地址,只要能在每个MCB的第一个字节中找到MCB标识符

字节M,循环就会继续下去。当循环失败时,在第一个字节Z标识的是最后一个有效的

MCB块;任何其它值都表示MCB链已经出错。所有这些检查都是memtrc通过有限的几

种状态来实现的。

LDT跟踪

最后一个跟踪例程是ldttrc,它可显示用来保存系统中每个逻辑驱动器的当前目录信

息的区域,并且,还可用此区域来记录被替代的(SUBSTed)和指定的驱动器信息。

在调用局部报告例程ldtrpt时,它可获取一个指向第一个LDT的指针,以及在范围0

至逻辑驱动器号(减去1,因为驱动器代码是以0为基数的)。它把驱动器代码转换成供报

告使用的字母值,并接着检查LDT中的状态字节,以确定事实上是不是已定义了驱动器。

如果已经定义了驱动器,那么系统就会把保存在LDT的当前目录信息与逻辑驱动器

字母一起打印出来,并转换和显示状态字节(以及尚未对其含义进行编码的LDT项)。

ldttrc进程首先设置curldt并显示一个标题。接着,它将传送给它的逻辑驱动器计数

减去1,以便校正为以0为基数的驱动器代码用法。至此,ldttrc进程便沿着每个可能逻辑

驱动器的LDT循环起来,为每个LDT调用ldtrpt,并把curldt设置为指向下一个LDT,

同时,还根据每一项的大小计算出LDT的地址。

主例程

已经检查完了每个跟踪模块,我们就把它们全部放在主调用序列中。主例程是通过使

用中断21h,功能30h(通过Turbo Pascal库进程MsDos来调用)来检查在DOS 3.x版本

下运行的程序而开始的。如果程序不是在DOS版本3.X下进行,该程序就会显示出一个

出错信息,并返回出错代码255。

如果DOS版本正确,操作会继续进行,此时,如果出现在DOS命令行上的程序名,伴

随有参数,那系统会把全局变量PawsFlag设置为正确,若出现在DOS命令行上只有一个

程序名,该全局变量就会被设置为出错。该功能允许通过配置或删除某个参数来控制屏幕

暂停功能;在此过程中,所提供的其它任何参数都无需用到。

该程序接着初始化在需要进行全屏幕暂停时,由holdup使用的lctr全局变量,接着,

它使用未公开的中断21h,功能52h来从DOS中获取CVT的地址,把返回的偏移值减去

8,并把结果保存在全局指针cvbase中。

cvt地址可用来获取静态系统数,并接着把它们显示在屏幕上,这在CVT用来保存

841页

它们的序列中是不需要的,甚至可以说,在一种可能更利传送信息的序列中,这个值是多

余的。

显示了这些项之后,该程序接着调用所有的跟踪/报告例程。holdup例程可保证用户

有充足的时间来读完该程序显示的每一个全屏消息。到这一步,整个CVT3程序便完成

了。

可以轻易地改编程序的任何部分以处理某一个区域。例如,通过改变memtrc例程来

搜索而不是显示数据的话,那么只使用该例程就可以创建一道能访问DOS主环境的程

序,而不管它所运行的外壳程序嵌套得有多么深。

D.3.3 CVT4.PAS把CVT3.PAS升级为版本4

在版本4问世以后,由于对未公开的DOS结构进行了一些改变,使得原有的CVT3.

PAS不能应用于新的版本。从这一点上看,正好说明了它是在使用未公开的功能时存在

的一个重大的缺陷——它们中的一切都能随着版本的改变而变化。至于这一点,我们在本

附录一开始就提醒过读者。

由于在CVT3.PAS中定义数据所采用的方式的独特性,它使得我们无法让该程序运

行于其它版本。这样,我们便不得不对该程序进行必要的改变,并推出一种新的程序,这就

是CVT4.PAS。在以下对CVT4.PAS的介绍中(见Listing 3.2),我们所介绍的只是改变

的部分;余下部分与CVT3.PAS相同。CVT4.PAS也可通过Turbo Pascal的版本5.0至

7.0来编译。

{cvt4.pas-  TurbO Pascal V5.0 version for DOS 4.x

*       copyright 1987, 1988, 1989 Jim Kyle -All Rights Reserved

*       The program, and the information it contains, may be freely

*       distributed and used so long as this entir  comment section

*       is not altered.

}

PROGRAM CVT4;

{$A-,B-,D+,E-,F-,I+,L+,N+,O-,R-,S+,V-}

{$M 8192, 0, 0}

USESDOS;

TYPE

bcb=RECORD                  {Buffer Control Block format,v4}

prcb     : word ;                {  +00 prev  one in chain}

nxcb     : word;                  {  +02 next one in chain}

ldrv   :byte ;                    { +04 logical drive code}

action : byte ;                   { +05 action code         }

lsect : longint;                 { +06 logical sector #  }

nf       : byte;                 { +0A nbr FATs or 01      }

secf     : word ;                { +0b sectors/FAT or 00}

pdrv   zpointer;                {  +0D phys drv tbl  tr   }

fill2:word ;                      { +11 unknown            }

fill3 : byte ;                    { +13 unknown            }

buf      : array[ 0 .. 511 ]  of byte ; { +14 the buffer itself}

END;

bcbptr=^bcb;

bcblnk = RECORD              { BCB link record , new in V4          }

bxval : word ;                {  +00 , unknown          }

pagebase : bcbptr;         { +02 , points to a BCB}

842页

usrs : byte;  { +06, count of usage }

fill : longint; { +07, unknown }

END ;

bcblkp = ^bcblnk ;

bufctl = RECORD { EMS control for buffers, new in V4 }

Inkptr : bcblkp; { +00, point to pagebase }

pgs : word; { +04, number of pages }

fill1 : longint; { +06, unknown }

fill2 : word; { +0A, unknown }

emsflg : byte; { +0C, FF if no EMS used }

emshdl : word; { +0D, EMS handle }

emsppg : byte; { +0F, EMS physical page }

END ;

bcptp = ^bufctl;

nam8 = array[1 . .8] of char;

mcb = RECORD { Mem Alloc Block header format }

flag : char; { +00 must be M or Z }

owner : word; { +01 PSP seg or 0000 }

siz : word; { +03 Number paragraphs }

junk : array[5..7] of byte; { +05-+07 not used }

name : nam8; { +08 Name of owner }

END ;

mcbptr = ^mcb ;

dpb = RECORD { Physical Drive Table forMat }

drvc : byte; { +00 drive code }

dunit : byte; { +01 unit number }

bps : integer; { +02 bytes per sector }

spc : byte; { +04 sec per cluster -1 }

pwr2 : byte; { +05 power of 2 }

nsrvs : integer; { +06 reserved sectors }

nfats : byte; { +08 number of FATs }

dirsiz : word; { +09 root dir size }

fus : word; { +0B first usable sectr }

tcc : word; { +0D total clstr ct +1 }

spf : word; { +0F sec per FAT **CHG**}

fds : word; { +11 first dir sector }

drvr : pointer; { +13 driver pointer }

mcode : byte; { +17 media code }

accflg : byte; { +18 access flag }

nxt : pointer; { +19 next table ptr }

lastused : word; { +1D last used cluster }

filler : word; { +1F usually FFFF }

END ;

dpbptr = ^dpb ;

chn = RECORD { Chain links for DCB, FCB chains }

nxtlnk : pointer; { +00 next link or FFFF }

nmbr : integer; { +04 number blocks here }

END ;

Chnptr = ^ chn ;

dcb = RECORD { Device Control Block format }

nusers : integer; { +00 users for this DCB }

mode : integer; { +02 0,1,2 per OPEN }

datrb : byte; { +04 disk attrib byte }

dvatr : byte; { +05 device attrib (hi) }

atrb2 : byte; { +06 2nd device attrib }

pdrvr : pointer; { +07 points to driver }

frstc : word; { +0B first cluster nbr }

modtm : word; { +0D file time word }

moddt: word; { +0F file date word }

843页

totsiz : longint; { +11 total file size }

curpos : longint; { +15 current byte pos }

clsctr : word; { +19 total cluster ctr }

curcls : word; { +1B current cluster }

dirsec : word; { +1D directory sector }

dirndx : byte; { +1F dir entry index 0. .}

name : array[0. .7] of char; { +20 dev/file name }

ext : array[0..3] of char; { +28 file extension }

fill2 : word; { +2B unknown }

fill3 : word; { +2D unknown }

fill4 : word; { +2F unknown }

owner : word; { +31 PSP of owner proc }

fill5 : word; { +33 unknown }

fill6 : word; { +35 unknown }

fill7 : word; { +37 unknown }

fill8 : word, { +39 unknown }

END :

dcbptr = ^dcb ;

ldt = RECORD { Logical Drive Table format }

name : array[0. .67] of char;{ +00 drive and path }

code : byte; { +44 drive in use code }

mydpb : dpbptr; { +45 DPB fOr this drive }

dirclu : word; { +49 directory cluster }

filler2: word; { +4B unknown }

filler3: word; { +4D unknown }

patlen : word; { +4F SUbST path length }

filler4: byte; { +51 unknown   }

filler5: word; { +52 unknown   }

filler6: word; { +54 unknown }

filler7: word; { +56 unknown }

END ;

ldtptr  = ^ ldt ;

cvt = RECORD { Configuration Variable Table }

curbfr : bcbptr; { current pos in chain }

memchn : mcbptr; { stact of MCB chain }

pdrvs : dpbptr; { Fn $52 points to here }

dcbchn : dcbptr; { set up by FILES= }

clkdev : pointer; { set by DEVICE= (clock$)}

condev : pointer; { set by DEVICE= (con) }

secsiz : integer; { maximum block size }

bfrchn : bcblkp; { set up by BUFFERS= }

ldrvs : ldtptr; { set up by LASTDRIVE= }

fcbchn : chnptr; { set up by FCBS= }

filler : integer; { number of FCBs to keep }

npdrvs : byte;  { set by driver list }

nldrvs : byte; {set by LASTDRIVE= }

END ;

cvtptr = ^ cvt ;

dtstr = string[8] ; { date -time Strings }

pstrg = string[9] ; { pointer-conversion fmt }

memst = string[12] ; { memory usage string }

VAR

cvtbase : cvtptr;

curbuf : bcbptr;

curmcb : mcbptr;

curdpb : dpbptr;

curchn : chnptr;

curdcb ,

curfcb : dcbptr;

curldt : ldtptr;

844页

bcbctr,

dcbctr : integer ;   { counters }

b : Registers;

PawsFlag: Boolean; { controls screen-full pausing }

lctr : integer; { counts lines displayed }

function hexn(b:byte) : char; { convents nibble to char }

begin

b := b and 15; { force to only 4 bits }

if b > 9 then inc(b,7); { adjust for hex digits }

hexn := chr(b+48) ; { convert to character }

end ;

function hexb(b:byte) : string;

begin

hexb := hexn(b shr 4) + hexn(b);  { assemble the nibbles }

end ;

function hexw(w:word) : string;

begin

hexw := hexb(w shr 8) + hexb(w); { assemble the bytes }

end ;

function hexl(l: longint) : string;

begin

hexl := hexw(l shr 16) + hexw(l); { assemble the words }

end ;

procedure outcstr( var s  : nam8); { output ASCIIZ string }

VAR

i : integer;

BEGIN

i : = 1 ;

while (s[i] <> #0) and (i < 9) do

begin

write ( s [ i] ) ;

inc ( i ) ;

end ;

END ;

PROCEDURE holdup; { count lines and wait if full }

BEGIN

IF PawsFlag THEN

SEGIN

inc ( lctr ) ;

if lctr > 23 then

BEGIN

lCtr := 0;

readln ;

END

END

END ;

FUNCTION xp( p : pointer ) : pstrg; { displays pointer P }

BEGIN

xp := hexw(seg(p^) ) + ' : ' + hexw(ofs(p^) ) ;

END ;

PROCEDURE dmp(f  : pointer); { display hex dump of data to CRT }

VAR

x : ^byte;

i : integer;

c : char;

BEGIN

X : = f ;

845页

write( xp(f) , '> ' ) ;

FOR i:=0 to 15 DO

BEGIN

write( hexb(x^) ;

if i=7 then write( ' - ' )

else write ( ' ' );

x := pointer(longint(x) + 1) ;

END ;

write( ' ' ) ;

X : = f ;

FOR i:=0 tO 15 DO

BEGIN

c := char($7f AND x^) ;

if c< ' ' then c : = ' . ' ;

write( c );

if i=7 then write( ' ' ) ;

x := pointer(longint(x) + 1);

END ;

writeln ;

holdup ;

END ;

PROCEDURE dpbtrc(a: pointer) ; { trace andreport DPB data }

VAR

ofsv : word;

PROCEDURE dpbrpt; { reports each DPB's content }

BEGIN

writeln ;

hOldup ;

write( 'Drive' , char (curdpb^ .drvc+ord( 'A' ) ) ) ;

write( ' : (unit , ' curdpb^.dunit, ' of driver at ' , xp(curdpb^.drvr) ) ;

writeln( ' ) media code = ' , hexb(curdpb^ .mcode) ) ;

holdup ;

write( ' ' , curdpb^ .bps:3, ' bytes per sector, ' ) ;

write( ' ' , curdpb^ .spc+1 :2, ' sectors per cluster, ' ) ;

writeln( ' ' , curdpb^ .tcc-1 :4, ' total cluster count ' ) ;

holdup ;

write( ' Res sectors: ' , curdpb^ . rsrvs, ' ' ) ;

write( ' ' , curdpb^ .nfats, ' FAT' 's, ' , curdpb^ . spf :2, ' sec ea ') ;

write( ' FAT entry: ' ) ;

IF (curdpb^.tcc) > $0FFC

THEN write( 16 )

ELSE Write( 12 );

writeln( ' bits Root: ' , curdpb^.dirsiz:3, ' entries' ) ;

holdup ;

write(' First root sector: ', hexwlcurdpb^.fds) );

write(' First data sector: ', hexw(curdpb^.fus) );

writeln( ' Last cluster used:  ',hexw(curdpb^ . lastused) ) ;

holdup ;

END; { of dpbrpt proc }

BEGIN

curdpb := a;

ofsv := 0;

writeln ;

holdUp ;

writeln( ' DRIVE PARAMETER BLOCK (DPB) DATA- - ' ) ;

holdup ;

WHILE (ofsv <> $FFFF) DO

BEGIN

ofsv := word(longint( curdpb^ .nxt ) ) ;

846页

dpbrpt ;

curdpb : = dpbptr (curdpb^ . nxt ) ;

END ;

write (#12 ) ;

END; { of dpbtrc proc }

PROCEDURE bcbtrc2(a : pointer) ;

VAR

ofsv : word;

ofst : word;

PROCEDURE bcbrpt(a : bcbptr);

VAR

i : integer;

x : pointer;

BEGIN

writeln ;

holdup ;

inc ( bcbctr ) ;

write( 'Buffer Control Block ' , bcbctr:2, ' at ' , xp(a) ) ;

write( ' Prev: ' , hexw(a^ .prcb) ) ;

writeln( ' Next : ' , hexw(a^ .nxcb) ) ;

holdup ;

write ( ' Logical ' ' ' , char ( ord ( ' A ' ) +a^ . ldrv ) ) ;

write ( ' : ' ' , Sector ' , hexl ( a^ . lsect ) ) ;

writeln ( ' Action code : ' , hexb(a^ . action ) ) ;

holdup ;

write( ' NFATS: ' , hexb( a^ .nf) ) ;

write( ' SPF: ' , hexw(a^ . secf) ) ;

writeln( ' DPB address : ' , xp(a^ . pdrv) ) ;

holdup ;

write ( ' FILL2 : ' , hexw( a^ . fill2 ) ) ;

writeln ( ' FILL3 : ' , hexb ( a^ . fill3) ) ;

holdup ;

x := addr(bcbptr(a) ^ .buf[0] ) ;

FOR i:=0 to 31 DO

dmp (pointer ( longint (x) + ( i SHL 4) ) ) ;

END; { of bcbrpt proc }

BEGIN { bcbtrc2 routine }

bcbctr := 0 ;

ofSv := 0;

ofst := ofs(a^) ;

WHILE (ofsv <> ofst) DO

BEGIN

ofsv : =bcbptr(a) ^ . nxcb ;

bcbrpt (a);

a := ptr(seg(a^ ) ,ofsv) ;

END ;

write ( #12 ) ;

END ; { bcbtrc2 routine}

PROCEDURE bcbtrc1 ( a : pointer) ;

PROCEDURE bcbtrl ( a : bcblkp ) ;

BEGIN

writeln ;

holdup ;

writeln( ' Page base is at ' , xp(a^ .pagebase) ) ;

holdup ;

write ( 'BX Val = ' , hexw(a^.bxval) );

write ( ' Users = ' , hexb(a^.usrs) );

writeln(  ' Fill = ' , hexl(a^ .fill) ) ;

holdup ;

bcbtrc2 ( a^ . pagebase ) ;

847页

END ;

BEGIN

writeln( ' Link table is at ' , xp(bcptr(a) ^ . lnkptr) ) ;

hOldup ;

write ( 'Page count = ' , bcptr(a)^.pgs:4 );

write ( Fill1 = ' , hexl(bcptr(a)^.fill1) );

writeln ( ' Fill2 = ' , hexw(bcptr(a) ^ .fill2) ) ;

holdup ;

write ( 'EMS Flag = ' , hexb(bcptr(a)^.emsflg) ) ;

write ( ' Handle = ' , hexw(bcptr(a)^.emshdl) ) ;

writeln ( ' Physpg = ' , hexb (bcptr(a) ^ . emsppg ) ) ;

holdup ;

bcbtrl( bcptr(a) ^ . lnkptr) ;

END ;

PROCEDURE dcbtrc(t : dtstr; a : pointer);

VAR

ofsv : word;

FUNCTION f_tm(n : word) : dtstr;

VAR

buf : dtstr;

b : byte;

BEGIN

b := ((n SHR 11) AND 31);

buf[1] :=char((b DIV 10) + 48);

buf[2] :=char((b MOD 10) + 48);

buf[3] :=': ' ;

b := ((n SHR 5) AND 63);

buf[4] :=char((b DIV 10) + 48);

buf[5] :=char((b MOD 10) + 48);

buf[6] :=': ' ;

b := ((n SHL 1) AND 63);

buf[7] :=char((b DIV 10) + 48);

buf[8] :=char((b MOD 10) + 48);

f_tm := buf;

END ;

funCTION f_dt(n : word) : dtstr;

VAR

buf : dtstr;

b : byte;

BEGIN

b:=((n shR 5) AND 15);

buf[1]:=char((b DIV 10) + 48);

buf[2]:=char((b MOD 10) + 48);

buf[3]:= '/';

b:=(n   AND 31);

buf[4]:=char((b DIV 10) + 48);

buf[5]:=char((b MOD 10) + 48);

buf[6]:='/';

b:=((n SHR 9) AND 15) + 80;

buf[7]:=char( (b DIV 10) + 48) ;

buf[8] :=char((b MOD 10) + 48);

f_dt   buf;

END ;

PROCEDURE dcbrpt( var t : dtstr; n : integer );

type

acctyp = array[0. .31 of string[7] ;

const

actyp : acctyp = ('READ ','WRITE', 'R/W' , unknown'} ;

Var

848页

isdvc : Boolean;

BEGIN

WHILE (n > 0) DO

BEGIN

INC (dcbctr) ;

writeln ;

holdup ;

write (t, ' ', dcbctr:2 );

IF (curdcb^ . name[0] = #0) THEN

BEGIN

writeln('at ' , xp(curdcb),'not used since bootup' );

holdup ;

END

else

BEGIN

isdvc := (curdcb^.dvatr and 128) <> 0;

write( ' for') ;

IF isdvc

THEN write('device ' )

ELSE write( 'file ' ) ;

write ( curdcb ^ . name [ 0 ] , curdcb^ . name [ 1 ] , curdcb^ . name [ 2 ] ) ;

write ( curdcb^ . name [ 3 ] , curdcb ^ . name [ 4 ] , curdcb^ . name [ 5 ] ) ;

write (curdcb^ . name[6] , curdcb^ . name[ 7] ) ;

IF (NOT isdvc) THEN { block driver}

write('. ' , curdcb^ . ext[0] , curdcb^ .ext[1 ] , curdcb^ . ext [2] ) ;

write(' at ' , xp( curdcb ) );

write('shows ' , curdcb^ .nusers) ;

writeln( ' OPENs ' ) ;

holdup ;

write( ' opened for ' , actyp[3 AND (curdcb^ .mode) ] ) ;

write('access' ) ;

IF ($FFFC AND (curdcb^ .mode) )<>0 THEN

write( ' ( ' , hexw(curdcb^ .mode) , ' ) ' ) ;

writeln( ' by process ' , hexw(curdcb^ .owner) ) ;

holdup ;

IF( isdvc ) THEN { Device }

BEGIN

write( ' Device driver at ' , xp(curdcb^ .pdrvr) );

write( ' is in ' ) ;

if ((curdcb^ .dvatr) AND 32)<>0 THEN write( 'Raw' )

ELSE write( ' Cooked ' ) ;

write( ' mode and is ' ) ;

if ((curdcb^ .dvatr) AND 64)=0 THEN write( ' not ' ) ;

writeln ( ' ready ' ) ;

holdup ;

END

else { File }

BEGIN

write( ' File is on drive ' , char(ord( 'A' )+( (curdcb^ .dvatr) AND 31 ) ) ) ;

write( ' : (driver at ' , xp(curdcb^ .pdrvr) ) ;

write( ' ) and has ' ) ;

if ( (curdcb^ .dvatr) AND 64)<>0 THEN write( ' not ' ) ;

writeln( ' been written to . ' ) ;

holdup ;

writeln( ' File ' ' s attribute byte = ' , hexb(curdcb^ .datrb) ) ;

holdup ;

END ;

write (' Mod Time/date: ' );

write (f_tm(curdcb^.modtm) , ' , ' ) ;

writeln ( f_dt ( curdcb^ . moddt ) ) ;

holdup ;

write ( ' Dir Sector: ' , hexw(curdcb^.dirsec) , ' ' ) ;

849页

writeln ( ' Dir Index : ' , curdcb^ . dirndx : 4 , ' ' ) ;

holdup ;

write ( ' First Cluster: ' , hexw(curdcb^.frstc) , ' ' ) ;

write ('Prev Clusters: ' , curdcb^.clsctr:4, ' ' ) ;

writeln( ' Current Cluster: ' , hexw(curdcb^ .curcls) , ' ' ) ;

holdup ;

write ( ' Directory size: ' , curdcb^ .totsiz:6, ' ' ) ;

writeln( ' Curr byte count: ' , curdcb^ .curpos:6 ) ;

holdup ;

write ( ' Fill2=' , hexw(Curdcb^ .fill2) , ' ' ) ;

write ( ' Fill3=' , hexw(curdcb^.fill3) , ' ' ) ;

write ( ' Fill4=' , hexw(curdcb^ .fill4) , ' ' ) ;

writeln ( ' Fill5= ' , hexw(curdcb^ . fill5) ) ;

holdup ;

write ( ' Fill6=' , hexw(curdcb^ .fill6) , ' ' ) ;

write ( ' Fill7=' , hexw(curdcb^ .fill7) , ' ' ) ;

writeln ( ' Fill8= ' , hexw( curdcb^ . fill8) ) ;

holdup ;

END ;

curdcb := pointer(longint(curdcb) + sizeof(dcb) - 1 ) ;

DEC( n );

END; { n >= 0 loop }

END ; { of dcbrpt }

BEGIN

curchn := chnptr(a) ;

dcbctr := 0;

ofsv := 0 ;

WHILE (ofsv <> $FFFF) DO

BEGIN

ofsv : = word ( longint (curchn^ . nxtlnk ) ) ;

curdcb : = dcbptr( longint (curchn )+sizeof (chn) ) ;

writeln ;

holdup ;

write ( 'Link at ' , xp(curchn) , ' contains ' ) ;

writeln( curchn^ .nmbr, ' ' , t, ' s -- ' ) ;

hOldup ;

dcbrpt (t, curchn^ .nmbr) ;

curchn : = chnptr(curchn^ . nxtlnk ) ;

END ;

write ( #12 ) ;

END ;

PROCEDURE memtrc(a : pointer) ;

VAR

z : longint;

PROCEDURE memrpt(s,o,a : word; var n : nam8);

FUNCTION memu(a : word) : memst; { determine memory use }

Var

x : char;

BEGIN

x := char( mem[a:0] );

CASE x OF

#$CD :

memu : = 'Program' ;

'A' . . 'Z' :

memu := ' Environment ' ;

else

memu : ='Data ' ;

End ;

END ;

850页

BEGIN

z := longint(s) SHL 4;

write( z:6, ' bytes ' );

IF (O<>0) THEN

begin

write ('USED by proc ' , hexw(o));

write ( ' , at ' , hexw( a ), ' :0000, for ' , memu(a) ) ;

if n[1] in [ 'A'. . 'Z'] then

begin

write( ' . Pgm name:');

outcstr( n );

writeln ;

end

else

writeln( ' . No program name' ) ;

end

else

writeln( 'FREE at ' , hexw( a ) , ' :0000' ) ;

holdup ;

END; { of memrpt }

BEGIN

curmcb := mcbptr(a) ;

writeln ;

holdup ;

writeln ( ' MEMORY ALLOCATION CHAIN ' ) ;

holdup;

WHILE (curmcb^ .flag = 'M' ) DO

BEGIN

menrpt ( curmcb^ . siz, curmcb^ . owner, SEG(curmcb^ )+1 , curmcb^ . name) ;

curmcb := ptr( seg(curmcb^) + (curmcb^.siz + 1), 0 );

END ;

IF (curmcb^ .flag <> 'Z' ) THEN

BEGIN

writeln(#13, #10, 'MEmoRY ALLOCATION ERROR at ' , xp(curmcb) ) ;

halt ( 255 ) ;

END ;

memrpt ( curmcb^ . siz , curmcb^ .owner, SEG(curmcb^ ) +1 , curmcb^ . name) ;

write (#12 ) ;

END; { of memtrc }

PROCEDURE ldttrc( a: ldtptr; n: byte );

PROCEDURE ldtrpt( 1: ldtptr; d: byte ) ;

VAR

ldrive : char;

i : integer;

BEGIN

ldrive := chr( $41 + d );

if (l^ . code AND byte($40) )=0 then

writeln( 'Logical Drive ' , ldrive, ' not yet defined' )

else

begin

write ( 'Logical Drive' , ldrive );

writeln( ' = Physical drive',  l^.name[0] ) ;

holdUp ;

write ('The current (full) pathspec is: ' );

i : = 0 ;

repeat

write ( 1^.name[i] );

inc(i);

851页

until (l^ .name[ i] ~~ #0) ;

writeln ;

end ;

holdup ;

if (l^ . code = $50) then

writeln( 'Code = 0*50 - - result of SUBST command' )

else if(l^ .code = $40) then

writeln( 'Code = 0*40 -- physical (or aliased) device' )

else

writeln( ' Code = 0x ' , hexb(l^ . code) , ' - - unknown ' ) ;

hOldUp ;

writeln( 'Directory Cluster x . , hexw( ~~ -dirclu ) ) `,

holdUp ;

writeln( 'Path Length to ignore = ' , hexw( l^.patlen ) ) ;

hOldup ;

write('Filler2 =', hexw(1^ .filler2) , ' ' );

write (Filler3 = ' , hexw(l^ .filler3) ,' ') ;

writeln( Filler4 =', hexb( 1^ . filler4) ) ;

holdup ;

write ( 'Filler5 =' ,  hexw(l^ .fillers) ,' ') ;

write ( 'Filler6 = ' , hexw(l^ .filler6) , ' ' ) ;

writeln( Filler7 = ', hexw( l^ . filler7) ) ;

holdup ;

writeln ;

holdup ;

END; { of ldtrpt }

VAR

o : byte;

BEGIN

Curldt := a;

writeln ;

holdup ;

writeln( ' LOGIcAL DRIVE TABLES t set by LASTDRIVE=, SUBST, etc . ) : ' )

;

holdup ;

dec( n ) ; { convert for zero-based reference }

for o := 0 to n do { loop through continuous tables }

begin

ldtrpt (curldt , o) ;

curldt : = ptr(seg(curldt^ ) , ( ofs(curldt^ ) + sizeof(ldt) ) ) ;

end ;

write ( #12 ) ;

END; { of ldttrc }

{ main }

BEGIN

b.ah := $30; { Check for correct DOS version }

MsDos ( b ) ;

.if b.al < 4 then { If not V4.x or higher, get Out }

begin

writeln('Wrong DOS version: ' , b.al, ' . ' , b.ah ) ;

halt (255) ; { return ErrorLevel=255}

end ;

PawsFlag := ParamCount = 0;. { else set up for paging output }

852页

lctr := 0;

writeln('Configuration Variables, DOS version', b.al, ' . ' , b.ah ) ;

holdup ;

b.ah :=  $52; { Get CVT pointer set up }

msdos (b) ; { (using undocumented function ) }

cvtbase := ptr(b.es, b.bx-8); { hold pointer to CVT }

writeln ;

holdup ;

writeln('CVT is located at ' , xp(cvtbase) ) ;

holdup ;

write ('No. of Phys Drives (al', xp(@cvtbase^.npdrvs));

writeln( ' ) : , cvtbase^ .npdrvs) ;

holdup ;

write ( 'No. of Log. Drives (at ' , xp(@cvtbase^.nldrvs)) ;

writeln( ' ) : ' , cvtbase^ . nldrvs) ;

holdup ;

write ( ' Clock Device (ptr at ' , xp(@cvtbase^.clkdev));

writeln ( ' ) : ' , xp(cvtbase^ . clkdev));

holdup ;

write ( ' CON Device (ptr at ' , xp(@cvtbase^.condev));

writeln( ' ) : ' , xp(cvtbase^ . condev) ) ;

holdup ;

write ('Sector Size(?) (at ' , xp(@cvtbase^.secsiz)) ;

writeln( ' ) : ' , hexw( cvtbase^ . secsiz ) ) ;

holdup ;

write ( ' FCBs to keep (at ' , xp(@cvtbase^.filler)) ;

writeln ( ' ) : ' , hexw(cvtbase^ . filler) ) ;

holdup ;

write ( '1. Memory Chain (ptr at ' , xp(@cvtbase^.memchn)) ;

writeln ( ') : ' , xp(cvtbase^ . memchn ) ) ;

holdup ;

write ( '2. DCB Chain (ptr at', xp(@cvtbase^.dcbchn) ) ;

writeln ( ' ) : ' , xp(cvtbase^ . dcbchn ) ) ;

holdup ;

write ( '3. DPB Chain (ptr at ' , xp(@cvtbase^.pdrvs) ) ;

writeln (') :',  xp(cvtbase^ . pdrvs ) ) ;

holdup ;

write ( '4. FCB Chain (ptr at ' , xp(@cvtbase^.fcbchn)) ;

writeln ( ' ) : ' , xp(cvtbase^ . fcbchn) ) ;

holdup ;

write ( '5. LDT Chain (ptr at ' , xp(@cvtbase^.ldrvs) );

writeln( ' ) : ' , xp(cvtbase^ . ldrvs) ) ;

holdup ;

write ('6. Current Buffer (ptr at ' , xp(@cvtbase^.curbfr));

writeln( ' ) : ' , xp(cvtbase^ . curbfr) ) ;

holdup ;

write ( '7. Buffer Chain (link at ' , xp(@cvtbase^.bfrchn) ) ;

writeln( ' ) : ' , xp(cvtbase^ . bfrchn) ) ;

holdup ;

writeln ;

holdup ;

writeln ( ' TRACING MCB Chain=== ' ) ;

holdup ;

memtrc ( cvtbase ^ . memchn ) ;

writeln ;

holdup ;

writeln ( ' TRACING DCB Chain=== ' ) ;

holdup ;

853页

dcbtrc(' DCB ' ,cvtbase^.dcbchn);

writeln;

holdup;

writeln('TRACING          DPB Chain===');

holdup;

dpbtrc ( cvtbase^ . pdrvs);

writeln ;

holdup;

writeln('  TRACING        FCB Chain===');

holdup;

dcbtrc('FCB',cvtbase^. fcbchn);

writeln;

holdup;

writeln(' TRAcING        LDT Chain===');

holdup;

ldttrc(cvtbase^.ldrvs,cvtbase^.nldrvs);

writeln;

holdup;

writeln('  TRACING Buffer Chain from current buffer===');

holdup;

DcbtrC2(cvtbase^.Curbfr);

writeln ;

holdup;

writeln( ' TRACING   Buffer Chain thru EMS link record===' );

holdup;

bcbtrc1 (cvtbase^ .bfrchn);

END.

缓冲区控制块和EMS链接

同CVT3相比,在CVT4中的一个主要改变是bcb记录的定义。这种昔日为一个指向

下一个缓冲器的4字节远指针(nxcb)变成了一对2字节的近指针,分别指向前一个缓冲

区(prcb)和下一个缓冲区(nxcb)。在控制块中,余下的功能大体上是相同的,不同之处有

lsect元素(逻辑扇区号)从16位值变成了32位值,以及secf元素(每FAT的扇区数)从一

个字节扩展成了一个16位的字。由于这些改变,BCB的大小也相应地由16个字节变成了

20个字节。

为了适应DOS随着版本4的出现而开始对扩展内存的使用,该程序中也加进了两个

新的结构。它们在CVT4中分别被定义为bcblink,一种保存着散列值(bxval)并指向相应

的BCB(而基)的链接记录;以及bufcbt—一种把EMM句柄与bcblnk记录连结在一起

的记录,并且还保存其它供EMS软件使用的数据。正如上述所介绍的一样,并非这两种

记录中的所有字段都得到了完全的编码。与这两种记录相应的指针类型分别被定义为

bcblkp(指向bcblnk记录)和bcptr(指向bufctl记录)。

由于在缓区处理上发生了许多变化,bcbtrc和bcbrpt过程也几乎是彻底地改头换面

了,只是从最初的模块特征上才可能看出昔日的面貌。bcbrpt的改变直接反映出了在版本

4 BCB所出现的新信息。在版本4中,两种新的过程(bcbtrc1)和(bcbtrc2)代替了原有的

bcbtrc例程,以便能处理访问缓冲方法上的改变并能适应新的EMS链接结构。至于另一

854页

个新的过程bcbtrl则是出现在bcbtrcl中。

称作bcbrpt的循环最初是bcbtrc中的,而现在,它被定位在bcbtrc2中。bcbtrl过程可

从bcblnk记录中报告出页基数据,并接着调用bcbtrc2;bcbtrc1在从bufctl记录中报告出

数据之后,便会把一个bcblnk指针传送到bcbtrl。这些改变表面上看起来似乎无惊人之

处,但却是必须的,只有这样才能处理输入数据时的重大改变。这也是足以说明了模块化

设计用在这些程序上的威力。

内在控制块

在版本4中,对MCB的改变是次要的;目前,用于程序装载的MCB包括程序的名字

(去掉扩展名,该名字就是从中装载该程序的文件的名字)。CVT4是通过定义一种新类型

-nam8(它是一个字符数组,而不是一个字符串,因为MCB既不含Pascal长度字节,也

没有采用C语言的字符串结尾字节)来处理这种改变的,并且把mcb记录的范围扩展到

包括nam8字段。

程序名的增加要求改变memrpt过程,以便在程序名存在时,系统能报告出nam8字段。不

过,除此之外,再也没有其它改变了。

驱动器参数块

在版本3和版本4之间,对DPB的唯一改变是每FAT的扇区数这一字段在大小上

从一个字节变成了一个字,这样,dPb记录的定义也相应地作了修改,以适应其变化。从概

念上看,此修改很细微,但却是CVT3不能追寻DOS V4系统的一个主要原因;dpb记录

中spf元素的大小的改变,改变了nxt指针的位置,而nxt指针则用于搜索DPB链,而程

序便无法再正常工作。

设备控制块

在DOS V4里,DCB的大小扩展了6个字节。在dcb记录的定义里增加了3个字(其

含义未知),用于反映这一变化。

dcbrpt过程也作了改变,来报告新增的字,但是,不必再进行其他的修改来跟踪或报

告DCB和FCB。

逻辑设备表

在DOS V4里,LDT的大小也扩展了7个字节。程序在ldt记录的定义中增加了1个

字节及3个字(意义均未知),来保持修改同步。

因为增加了数据,所以对ldtrpt也进行了细微的修改来报告新字段里的内容。

CVT定义的本身

在版本4中,对CVT记录只进行了唯一的一点改变,这个改变是如此之小,以致于容

易引起人们的忽视,这就是指针bfrchn的类型从bcbptr(版本3中)变成了bcblkp(版本4

中)。但,这一细微的改变却有其重大的意义。

实用程序例程

在CVT4中增加了一种实用程序例程,即outcstr,以便为新的nam8数据类型(供改

855页

变后的MCB使用)提供输出功能。把nam8变量的地址传送给该例程,它就能不断地输出

字符,直到它处理完了全部8个变量或者遇到了00h字节。它只能由memrpt例程使用。

主例程

在主调用序列中的第一个变化是,CVT4通过DOS版本4而不是版本3来检查操

作,并且当不是版本4时,CVT4便会终止。

在总结静态报告的措词方面的改变是很轻微的,仅仅只反映出CVT记录中的curbfr

字段在含义上的改变。类似地还有,用于缓冲器跟踪的标题也一定程度地有所改变,因为

整个链接页(或在没有使用EMs时为所有缓冲器)是在CVT3仅仅用来报告当前缓冲器

的同一点上被报告出来的。

但是,程序之间的改变与DOS版本之间的实际变化相比,往往总是要简单得多。

D.4小        结

本附录介绍了一种可使用未公开的DOS功能的方法,以探索DOS内部的工作,并

列举了一个读者可以修改的例子,以便在DOS运行时,利用可以得到的信息。

但是,必须记住:无论何时,只要DOS的主版本号出现了变化,使用这些功能都会令

人十分头痛。因此,在程序中应尽量少用未公开的功能,这样便能在商业领域中或在大众

化的范围内使开发的产品具有更为广阔的天地,也能为每个用户拓宽应用前景以及减少

不必要的麻烦。

856页

附录E 支持资源清单

E.1 硬  件

Hogan, Thom. The Programmer's PC Sourcebook. Microsoft Press, Redmond, Washington,

1991.

Intel Corporation. 80286 and 80287 Programmer's Reference Manual. Intel Corporation,

Santa Clara, California, 1987.

Intel Corporation. 80386 Programmer's Reference Manual. Intel Corporation, Santa Clara,

California. 1986.

Intel Corporation. 80387 Programmer's Reference Manual. Intel Corporation, Santa Clara,

Califormia, 1987.

Intel Corporation. i486 Microprocessor Programmer's Reference Manual. Intel Corpora-

tion, Santa Clara, California, 1989.

Intel Corporation. iAPX 86/88, 186/188 User's Manual. Intel Corporation, Santa Clara,

California. 1987.

International Business Machines Corporation. Mouse Technical Reference. IBM, Boca Raton,

Florida. 1987.

International Business Machines Corporation. Personal Computer Technical Reference.

IBM, Boca Raton, Florida, 1984.

International Business Machines Corporation . Personal System/2 and Personal Computer

BIOS Technical Reference. IBM, Boca Raton, Florida, 1987.

Morse, Stephen P. The 8086/8088 Primer, 2nd Edition. Hayden Book Company, Rochelle

Park, New Jersey, 1982.

Woram, John. The PC Configuration Handbook. Bantam Books, New York, New York, 1987.

E.2 MS-DOS 和BIOS 编程

Brown, Raif, and Kyle, Jim. PC Interrupts. Addison-Wesley, Reading, Massachusetts, 1991 .

Campbell, Joe. C Programmer's Guide to Serial Communications. Howard W. Sains and

Company, Indianapolis, Indiana, 1987.

857页

Chesley, Harry R. and Mitchell Waite. Supercharging C with Assembly Language. Addison-

Wesley, Reading, Massachusetts, 1987.

Duncan, Ray. Advanced MSDOS Programming, Second Edition. Microsoft Press, Redmond,

Washington, 1988.

Hyman, Michael. Memery Resident Utilities, Interrupts, and Disk Management with MS

and PC DOS. MIS Press, Portland, Oregon, 1986.

Jamsa, Kris. DOS Programming: The Complete Reference. Osborne/McGraw-Hill, Berkeley,

California. 1991 .

Jourdain, Robert, and Norton, Peter. Programmer's Problem Solver, 2nd Edition. Brady

Books, New York, New York, 1986.

Kyle, Jim. DOS Developer's Guide, New Edition. Sams Publishing, Carmel, Indiana, 1993.

Lai, Robert S. Writing MS-DOS Device Drivers. Addison-Wesley, Reading, Massachusetts,

1987.

Phoenix Technologies Ltd. System BIOS for IBM PCs, Compatibles, and EISA Computers,

2nd Edition. Addison-Wesley, Reading, Massachusetts, 1991 .

Porter, Kent. Stretching Turbo Pascal. Brady Books, New York, New York, 1987.

Schulman, Andrew. Undocumted DOS Addison-Wesley, Reading, Massachusetts, 1990.

Wadlow, Thomas A. Memory Resident Programming on the IBM PC. Addison-Wesley,

Reading, Massachusetts, 1987.

Williams, AL DOS 5: A Developer's Guide. M&T Books, Redwood City, California, 1991 .

Wilton, Richard. Programmer's Guide to PC and PS/2 Video Systems. Microsoft Press,

Redmond, Washington, 1987.

E.3 编程语言

Abel, Peter. Assembler for the IBMPC and PC-XT Reston Publishing, Reston, Virginia,1984.

Feldman, Phil and Tom Rugg. Using QuickBASIC 4 . Que Corporation, Carmel, indiana, 1988.

Harbison, Samuel P. and Guy L.Steele, Jr. C:A Referme Manual. Prentice Hall, Engewood

Cliffs, New Jersey, 1987.

Holzner, Steve, and Peter Norton Computing, Inc. Advanced Assembly Language. Brady

Books, New York, New York, 1991 .

Johnson, Marcus. Assembly Language for Reat Programmers ONLY! Sams Publishing,

Carmel, Indiana, 1993.

Lafore, Robert. Microsoft C Programming for the PC, 2nd Edition. Howard W. Sams and

Company, Indianapolis, Indiana, 1990.

Scanlon, Leo. IBM PC and XT Assembly Language.Brady Books, Bowie, Maryland,1983.

Socha, John, and Norton, Peter. Peter Norton's Assembly Language Book for the IBM PC

Brady Books, New York, New York, 1986.

858页

Wyatt, Allen. Advanced Assembly Language. Que Corporation, Carmel, Indiana, 1992.

Wyatt, Allen. Using Assembly Language, 3rd Edition. Que Copooration, Carinel, Indiana,

1992.

Yester, Michael. Using Turbo Pascal 6, 2nd Edition. Que Corporation, Carmel, Indiana, 1991 .

E.4 一般编程技术

Birrell, N.D. and M.A. Ould. A Practical Handbook for Software Development. Cambridge

University Press, Cambridge, England, 1986.

Ledgard, Henry. Software Engineering Concepts. Addison-Wesley, Reading, Massachusetts,

1987.

Liffick, Blaise W. The Software Developer's Handbook. Addison-Wesley, Reading, Massachu-

setts, 1985.

Yourdon, Edward. Techniques of program Structure and Design. Prentice Hall, Englewood

Cliffs, New Jersey, 1975

其实句柄、API之类的概念在DOS中就有很大发展了,所以耐心看完这本书绝对大有好处!


---------------------------------------------------------------完------------------------------------------------------------------------

DOS程序员手册(十五)的更多相关文章

  1. DOS程序员手册(五)

    第8章磁           盘       学习编程语言,常常是从基本的输入和输出入手的(正如第5.6和第7章曾介绍的一 样).到目前为止,我们不仅学习了怎样输入和输出数据,还学习了如何进行数据操作 ...

  2. DOS程序员手册(一)

    当今MS-Windows横扫大江南北,让我们这就来研究一下它的祖宗——MS-DOS! 这本书很难得,希望读者好好学习! DOS程序员手册(一) DOS教程 (以下内容全部为原作者的阐述,照样保留) 这 ...

  3. DOS程序员手册(十四)

    附录A ASCII字符集 十进制        十六进制      二进制              AscII         控制        按键 X10         X16        ...

  4. DOS程序员手册(九)

    第14章参考手册概述     本书余下的章节将向读者们介绍BIOS.DOS各种各样API函数和服务,作为一名程 序员,了解和掌握这些知识是很有好处的.在所介绍的参考手册中,每部手册都汇集了大 量的资源 ...

  5. DOS程序员手册(十)

    终于到(十)了~~~ 503页 ES:DI       指向未更新且未打开的FCB的指针 注释:该功能最初用来从命令行中析取文件,并以正确的格式来保存此文件 以便打开FCB.为了实现这个目的,可首先将 ...

  6. DOS程序员手册(十二)

    DOS可安全使用 610页 在DOS控制台I/O操作进行轮询循环时,有规律地调用中断,以便允许终止 并驻留(TSR)程序(如适用于DOS的实用程序PRINT.COM),知道它可安全 地使用文件操作和其 ...

  7. DOS程序员手册(八)

    备,就可以从程序中访问驱动程序.可以用句柄功能调用来     打开设备(见列表12.9)         列表12.9           /*example.C               List ...

  8. DOS程序员手册(四)

    5.4打印机功能 打印机是能够直接控制的输出设备之外的唯一的重要输出设备.它们的功能比屏幕 107页 功能要简单得多,因为它们只涉及字符输出,并最小程度地与打印机的输入有关. 输出给打印机的最简单的方 ...

  9. DOS程序员手册(二)

    2.6存储设备     随着DOS的升级,磁盘存储容量也有了很大扩充.表2.4介绍了软盘容量的增加以     及所支持驱动器型号的数量.                                 ...

  10. DOS程序员手册(十三)

    744页 在DPMI 1.0下,系统会修改并重新装载所有含选择符的段寄存器,并且将所有 含有要释放的选择符的寄存器清空为0. 客户程序绝不能修改或释放该功能分配的任何描述符.Int 31h.功能010 ...

随机推荐

  1. 详细讲解:通过composer安装TP5.1(Thinkphp5.1)

    现在TP5越来越火了,TP5也更新到了5.1版本,但是5.1以上版本只能通过composer来进行安装,那么这里贴出详细的步骤 前提:PHP版本必须要5.6以上 参考网址:http://www.thi ...

  2. 【js基础修炼之路】- 微任务,宏任务和Event-Loop

    一段代码让你了解Event-Loop console.log(1); setTimeout(() => { console.log(2); }, 0); new Promise((resolve ...

  3. C语言头文件怎么写?(转载)

    ---恢复内容开始--- c语言头文件怎么写?我一直有这样的疑问,但是也一直没去问问到底咋回事:所以今天一定要把它弄明白! 其实学会写头文件之后可以为我们省去不少事情,可以避免书写大量的重复代码,还在 ...

  4. 【转载】#402 - Value Equality vs. Reference Equality

    When we normally think of "equality",we're thinking of value equality - the idea that the ...

  5. 关于使用Encoding转码的问题,以及StreamWriter的小应用

    StreamWriter write = new StreamWriter("../../test2.txt"); write.WriteLine("中国123巴西red ...

  6. 1.4 配置备份策略(Policy)

    1.1 配置备份策略(Policy) 一个备份策略由四部分组成. Attributes(属性) Policy是否Active Policy类型 由此Policy产生的任务的优先级 使用的Storage ...

  7. mysql关键字了解

    unsigned  无符号 就是没有负数 列-1  -2 auto_increment 自增 comment 注释 primary key 主键 foreign key ()   references ...

  8. 第44章 MPU6050传感器—姿态检测—零死角玩转STM32-F429系列

    第44章     MPU6050传感器—姿态检测 全套200集视频教程和1000页PDF教程请到秉火论坛下载:www.firebbs.cn 野火视频教程优酷观看网址:http://i.youku.co ...

  9. C#接口定义

    C#接口定义 C#不支持多重继承,但是客观世界出现多重继承的情况又比较多.为了避免传统的多重继承给程序带来的复杂性等问题,C# 提出了接口的概念.通过接口可以实现多重继承的功能.  继承该接口的类或结 ...

  10. python解析ini文件

    python解析ini文件 使用configparser - Configuration file parser sections() add_section(section) has_section ...