MORE INFORMATION
When doing interlanguage calling with Microsoft QuickBASIC, the other
language can make calls to the BASIC run time to get the address of
passed arguments. This is accomplished with the GetNextLibArg
statement.
GetNextLibArg returns a pointer to a variant record containing another
pointer to each possible type of QuickBASIC variable (single
precision, double precision, integer, etc.). Usually a double
indirection of pointers is used to get the actual passed data.
Using double indirection of pointers with Apple MPW (Macintosh
Programmer's Workshop) Pascal, however, does not return the correct
value. This is a QuickBASIC header file problem. The routine below
demonstrates how an inline assembly-language routine can be used in
place of the double indirection to work around the problem.
More information on interlanguage calling with Microsoft QuickBASIC
can be found in the "Microsoft QuickBASIC for Apple Macintosh:
Language Reference," starting on page 444.
Compile and link PrintDbl.p from Apple MPW Pascal version 3.00 as
follows:
pascal -p PrintDbl.p
link -p -rt MBPC=999 PrintDbl.p.o BasicLib.a.o -o PrintDbl
Note: The file BasicLib.a.o comes with Microsoft QuickBASIC for the
Apple Macintosh on the Examples disk in the User Libraries:MBPC
Rsrcs:MPWP PCR Folder.
Once PrintDbl is compiled and linked, ResEdit should be used to change
parameters in PrintDbl as follows:
- In ResEdit, find the newly created file PrintDbl and click once on
the name so that it is highlighted.
- While holding down the COMMAND key, press the "I" key. This is the
same as choosing Get Info from the File menu. Change the following
in the dialog box that appears:
- Change the File field to "PrintDbl".
- Change the Type field to "MBPC".
- Change the Creator field to "MSBB".
- Close the dialog box and click Yes when asked if you want to save.
- Double-click PrintDbl to open it. There should be one resource
of MBPC displayed; double-click this also. Single-click the
displayed MBPC and press COMMAND+I. In the Get Info dialog box that
is brought up, change the Name field to PrintDbl.
PrintDbl is now a pure code resource that can be used with the RetDbl
QuickBASIC program.
It should be noted that PrintDbl uses the record structure BigRec to
pass the double precision number to the assembly language subroutine.
This is because MPW Pascal automatically converts single and double
precision numbers to an extended 10 byte format when passing them.
Using the record structure stops MPW Pascal from performing this
conversion.
Code Example
The following QuickBASIC program is RetDbl, which invokes an Apple MPW
Pascal routine to return series of double precision numbers:
LIBRARY "PrintDbl"
A# = 0 : B# = 0 : C# = 0
CALL PrintDouble(A#, B#, C#)
PRINT A#; B#; C#
WHILE INKEY$ = "" : WEND
The following Apple MPW Pascal routine is PrintDbl.p, which accepts a
series of passed double precision numbers from a Microsoft QuickBASIC
program and returns a constant value in them:
{**************************************************************}
{* PrintDbl.p (c) 1989 Microsoft Corporation *}
{**************************************************************}
{* Description: Example to show how to pass double precision *}
{* number from QuickBASIC to MPW Pascal and back.*}
{**************************************************************}
{$R-} { Turn off range checking}
UNIT PrintVerRec;
{*======================== INCLUDE FILE=======================*}
USES
{$U MemTypes.p } MemTypes,
{$U BasicLibMPWP.p } BasicLib;
TYPE
BigRec = RECORD
singnum: DOUBLE;
END;
procedure AssignDbl (ptr: LIBARGPTR; val: BigRec);
inline
$225f, { movea.l (a7)+,a1 }
$2e19, { move.l (a1)+,d7 ;get Long value }
$2c19, { move.l (a1)+,d6 ; get second part }
$205f, { movea.l (a7)+,a0 ;get ptr to Int variable }
$20C7, { move.l d7, (a0)+ ;assign the Int value }
$20C6; { move.l d6, (a0)+ ;assign the Int value }
PROCEDURE Main;
{$S Main}
{*------------------------------------------------------------*
* A routine to pass back to BASIC double precision constants *
* Called from BASIC as: *
* CALL PrintDbl (< double precision argument list >) *
*------------------------------------------------------------*}
PROCEDURE MAIN;
VAR
tempflag,argtype: INT16;
valptr: LIBARGPTR;
tempdbl: DOUBLE;
BEGIN
argtype := GetNextLibArg(valptr,tempflag);
WHILE (argtype <> _ARGSEND) DO
BEGIN
if (argtype = _DBLEARG) THEN
BEGIN
tempdbl := 20.25;
AssignDbl(valptr, BigRec(tempdbl));
END;
argtype := GetNextLibArg(valptr,tempflag);
END;
END; {* of MAIN *}
Running RetDbl produces the following output:
20.25 20.25 20.25