Anchored Variables
Volume Number: | | 1
|
Issue Number: | | 8
|
Column Tag: | | Programmer's Forum
|
"Modula-2 and Anchored Variables"
By Tom Taylor, Software Engineer, Modula-2 Corp., Provo, Ut.
"Modula-2 and Anchored Variables"
The programming language Modula-2 supports a little known but extremely powerful feature called "anchored variables." Anchored variables allow one to specify in the variable's declaration the absolute address of the variable and override the compiler's allocation method. This feature was intended to allow access to device registers on computers with memory mapped I/O (like on a PDP-11 where the original Modula-2 compiler was developed). It also allows the Modula-2 programmer on the Macintosh to access the Mac's operating system global variables very conveniently and cleanly.
In this article, we will look at two examples of anchored variable usage. In the first, we will try a small example to show that anchored variables really do work. In the second example, we'll actually develop a useful module based on anchored variables. These examples have been programed using MacModula-2 by Modula Corporation. However, with minor changes to the programs (probably the import statements), these programs should run using any Modula-2 system.
Anchored variables are "anchored" to a specific address by specifying that address in brackets immediately following the variable name in a variable declaration and before specifying the variable's type. For example:
VAR
FinderName [2E0h]
: ARRAY [0..16] OF CHAR;
This statement tells the compiler to allocate the variable FinderName starting at hex location 2E0h. Location 2E0h just happens to contain a Mac style string with the name of the current finder. The following program, will print out the name of the currently installed finder:
MODULE PrintFinder;
FROM Terminal IMPORT
ClearScreen, Write,
Read, WriteLn, WriteString;
VAR
FinderName [2e0h] :
ARRAY [0..16] OF CHAR;
i : CARDINAL;
ch : CHAR;
BEGIN
ClearScreen;
WriteString("Current finder is: ");
FOR i := 1 TO ORD(FinderName[0]) DO
Write(FinderName[i]);
END;
WriteLn;
Read(ch);
END PrintFinder.
Volume Control Block Queue
This next example is slightly more useful. It demonstrates the use of anchored variables in traversing the Volume Control Block queue and returning information about any disk (or volume) visible on the desktop (inserted in a drive or ejected) at the time your program was launched. This technique is used, for example, in the MacModula-2 compiler and linker when they search for imported files. This feature allows users of single drive Macs to build programs that are spred out over a number of disks by having those disks visible on the desktop at the time the compiler or linker is launched. The compiler or linker will search each disk found on the desktop by traversing the VCB queue until the desired import file is found or the end of the queue is reached.
The following example actually consists of three modules:
1) a Definition Module that defines the records, types, global variables, and procedures that are available for use by any program.
2) An Implementation Module that contains the code for the procedures defined in the Definition Module.
3) A simple Program Module that uses the procedures we have written and demonstrates some of the information available from the Volume Control Blocks.
DEFINITION MODULE VolumeTracer;
FROM MacSystemTypes IMPORT LongCard, Ptr;
EXPORT QUALIFIED VCB, VolumesOnLine,
GetVolumeInfo;
TYPE
QElemPtr = POINTER TO VCB;
VCB = RECORD
qLink: QElemPtr;
(* next queue entry *)
qType: INTEGER;
(* not used *)
vcbFlags: INTEGER;
(* bit 15=1 if dirty *)
vcbSigWord: INTEGER;
(* always $D2D7 *)
vcbCrDate: LongCard;
(* date volume initialized *)
vcbLsBkUp: LongCard;
(* date of last backup *)
vcbAtrb: INTEGER;
(* volume attributes *)
vcbNmFls: INTEGER;
(* # of files in directory *)
vcbDirSt: INTEGER;
(* directory's first block *)
vcbBlLn: INTEGER;
(* length of file directory *)
vcbNmBlks: INTEGER;
(* # of allocation blocks *)
vcbAlBlkSiz: LongCard;
(* size of allocation blocks *)
vcbClpSiz: LongCard;
(* # of bytes to allocate *)
vcbAlBlSt: INTEGER;
(* first block in block map *)
vcbNxtFNum: LongCard;
(* next unused file number *)
vcbFreeBks: INTEGER;
(* number of unused blocks *)
vcbVN: ARRAY [0..27] OF CHAR;
(* vol name Str255 format *)
vcbDrvNum: INTEGER;
(* drive number *)
vcbDRefNum: INTEGER;
(* driver reference number *)
vcbFSID: INTEGER;
(* file system identifier *)
vcbVRefNum: INTEGER;
(* volume reference number *)
vcbMAdr: Ptr;
(* location of block map *)
vcbBufAdr: Ptr;
(* location of volume buffer *)
vcbMLen: INTEGER;
(* # of bytes in block map *)
vcbDirIndex: INTEGER;
(* used internally *)
vcbDirBlk: INTEGER;
(* used internally *)
END;
PROCEDURE VolumesOnLine(): CARDINAL;
(* Returns the maximum number of
volumes currently recognized by the
Mac operating system. *)
PROCEDURE GetVolumeInfo(VAR volume :
VCB; whichVol : CARDINAL);
(* Returns the current VCB block for
volume "whichVol" The variable
"whichVol" must be between 1 and
the result of the procedure of
"VolumesOnLine()". Otherwise,
the "volume" info is undefined. *)
END VolumeTracer.
IMPLEMENTATION MODULE VolumeTracer;
TYPE
QHdrPtr = POINTER TO QHdr;
QHdr = RECORD
qFlags : INTEGER;
(* queue flags *)
qHead : QElemPtr;
(* first queue entry *)
qTail : QElemPtr;
(* last queue entry *)
END;
VAR
VCBQHdr [0356h] : QHdr;
(* VCB queue header *)
PROCEDURE VolumesOnLine(): CARDINAL;
VAR
ptr : QElemPtr;
count : CARDINAL;
BEGIN
ptr := VCBQHdr.qHead;
count := 0;
WHILE ptr # NIL DO
INC(count);
ptr := ptr^.qLink;
END;
RETURN count;
END VolumesOnLine;
PROCEDURE GetVolumeInfo(VAR volume :
VCB; whichVol : CARDINAL);
VAR
ptr : QElemPtr;
count : CARDINAL;
BEGIN
ptr := VCBQHdr.qHead;
count := 0;
WHILE (ptr # NIL) AND
(count # whichVol) DO
INC(count);
IF count = whichVol THEN
volume := ptr^;
END;
ptr := ptr^.qLink;
END;
END GetVolumeInfo;
END VolumeTracer.
MODULE ListVolumes;
FROM VolumeTracer IMPORT
VCB, VolumesOnLine, GetVolumeInfo;
FROM InOut IMPORT WriteString, ClearScreen, WriteLn, WriteCard, WriteInt,
Write, Read;
VAR
i, maxVols : CARDINAL;
vcb : VCB;
ch : CHAR;
PROCEDURE PrintVolName;
VAR
i : CARDINAL;
BEGIN
FOR i := 1 TO ORD(vcb.vcbVN[0]) DO
Write(vcb.vcbVN[i]);
END;
WriteLn;
END PrintVolName;
BEGIN
ClearScreen;
WriteString
("Number of volumes on-line: ");
maxVols := VolumesOnLine();
WriteCard(maxVols,0);
WriteLn; WriteLn;
FOR i := 1 TO maxVols DO
GetVolumeInfo(vcb,i);
WriteString('Volume Name: ');
PrintVolName;
WriteString
('Number of files in volume: ');
WriteInt(vcb.vcbNmFls,0);
WriteLn;
WriteString('Drive Number: ');
WriteInt(vcb.vcbDrvNum,0);
WriteLn;
WriteLn;
END;
Read(ch);
END ListVolumes.
Figure 1. Typical VCB Queue
This is only one simple example of a typical use of anchored variables. Many times, Inside Macintosh will mention some variable that can be accessed from assembly language. The Window Manager, for example, mentions that putting a WindowPtr in the variable GhostWindow, will cause that window to never be the front window. In order to find the addresses of such variables, such as GhostWindow, one need simply paw through ToolEqu.TXT or SysEqu.TXT. Both of these source files are included with the MDS system by Apple. Not only have I used GhostWindow as an anchored variable in my applications, I have also used anchored variables to access the information set by the Finder when a program is launched.
By taking advantage of the power of anchored variables, you will be able to create very readable programs that use some of the Mac's low-level features.