Myth number one: A procedural variable must always point to actual code; it can't have Nil as its value.
Myth number two: Since you can't assign object methods to procedural variables, you can't call methods indirectly (that is, through a pointer).
Myth number three: You can't write object-oriented DLL's in TPW, v1.
If you believe any of these myths, you should read on because, as you'll see, each is only `sort of' true: Each elevates a restriction or a difficulty to the status of a full-fledged impossibility.
Since Pascal treats parameter binding much like variable assignment, the same restrictions apply to passing procedural types to other procedures. If a procedure expects a ProcType argument, we can pass it ProcType(NilVar) but we can't pass it Nil or ProcType(Nil).
Since it's incredibly unlikely that the interrupt table at 0:0 will contain a reasonable procedure, we don't want to ever actually call a procedural variable whose value is Nil: the Nil value signals that the program shouldn't do anything here, or that it should take some sort of default action. Thus, if we know that at some point a procedural variable may be set to Nil, we should precede its use with a if (@ ProcVar) = Nil then {take the default action} else {use the ProcVar};.
The implicit parameters and the special handling of constructor results are the only differences between method calls and normal calls: there's no magic involved. If we simply define a procedural type, ProcType, that explicitly declares the method's implicit parameters after any normal parameters, we can then use ProcType to cast any pointer variable to a procedural variable. Once it's cast, the pointer acts just like a normal procedural variable; we can assign it to another procedural variable or use it to call a procedure. Just as with a normal procedural type, the only difference between a direct and indirect call lies in the way we make the call: The parameters are pushed and popped in the same way; the called code operates just the same; and indirectly called methods have the same full access to their object's fields (through the Self pointer) as directly called methods do.
Thus, if we have a method with no arguments and no results, we would simply make the declaration type Niladic = procedure (var Self);. To use it, we remember that we can only cast pointer variables, not pointer expressions, and so do something like PtrVar := @ ObjectType.Method; Niladic(PtrVar)(Self); Now, while there is something strange looking about a cast (in parentheses) followed by an argument list (in parentheses), indirect method calls are typically rare and concentrated in a few key routines, even in programs that rely heavily on them. (My typical uses for indirect method calls involve things like executing a list of object/method pairs on every timer tick, or calling a window object's message handler after DMT lookup reveals that it does have a handler.) What's more, the strange look of an indirect method call does not translate into strange object code: Using a cast to a procedural type generates the exact same code as using an normal procedural type, and that's both a little smaller and only slightly slower than a normal, direct procedure or function call.
Methods that require parameters or that return results are only slightly different than our Niladic example above. We simply have to remember to put any explicit parameters before the implicit parameter(s). Thus, we might use type SimplePredicate = function (var Self): boolean; for a method that takes no arguments and returns a boolean, and type UntypedDyadic = procedure (var A, B; var Self); for a method that requires two untyped memory references.
Just as with a normal procedure call, the compiler will not let us make an indirect method call with the wrong number or type of arguments. This is obviously desirable behavior, but it's tempered with a bit of a caveat: When we make a cast, we are effectively telling the compiler that we know exactly what we are doing. If we accidentally use a pointer to a UntypedDyadic method as a Niladic, the compiler will neither require nor accept the two var parameters to the UntypedDyadic method but the procedure will probably use them and the result will not be pretty! Similarly, the compiler will not complain if you cast a data pointer or the address of a near routine into a procedural type: it will blithely generate code that will (at best!) crash your computer.
There's no getting around this ban on virtual method calls, but it will probably go away in some future release of TPW and, in the meanwhile, it doesn't prevent us from making static method calls across EXE/DLL and DLL/DLL boundaries.
Now, any DLL entry point has to be declared export so that it can have the special prologue/epilogue code that saves and changes DS on entry and restores it on exit. A little experimentation will convince you that you can't directly export objects' methods: you have to export a `flat' (not object-oriented) shell routine that in turn calls the method. The bindings unit on the DLL user's end simply repeats the objects' declarations and declares that the DLL shell routines are their external implementations.
Figure 1 is an extremely simplistic example of an object-oriented DLL's export library. It consists entirely of a series of shell routines and the associated exports statements. (Figure 1 exports objects that are defined in the SimplObj unit, though there's no reason that exported objects can't be defined in the export library.) Just like procedural types that are used to make indirect method calls, the exported shell routines have to explicitly declare both the methods' explicit arguments (if any) and their implicit (VMT and Self) arguments. All the shell routines have to do (besides invisibly set and restore DS) is to pass all their arguments on to the actual method. The simplest and most general way to do this seems to be to use a procedural type to make an indirect call through a typed constant pointer to the actual method.
Figure 2 is the corresponding bindings unit for the DLL user. Windows is designed to make it easy for you to call DLL entry points under any name you wish, so there isn't any problem binding a shell routine named Simple_Setup to the using code's Simple.Setup.
Figure 2 also illustrates another use for renaming, or aliasing, DLL entry points. Simple.Method and Simple.Alias both refer to the same procedure - external DLL index 3 - but where Method expects a pointer argument, Alias expects segment and offset words. When you construct an alias, you have to be very careful that the alias pushes the same number of words, in the same order, as the routine it's aliasing. You should also be thoroughly familiar with the parameter passing conventions detailed in the Inside Turbo Pascal section of the manual, as in many cases only a pointer is passed and the called code is responsible for copying the argument into a local variable.
Virtual methods aside, DLL-resident objects act just like EXE-resident objects: The using code can even call constructors and destructors through the extended syntax to New() and Dispose(). While the shell routines do slow execution down a bit, so does normal DLL linkage. For most reasonably sized argument lists, the shell routine's repushing the arguments and calling the method will cost about as much as the standard export routine prologue/epilogue code. In other words, not much: Only the shortest and fastest methods will be significantly slowed by placing them in a DLL.
 
Admittedly, constructing the export library and the bindings unit is a 
tedious, error-prone chore, but it's also quite susceptible to 
automation:  My shareware 
While a ban on virtual methods is certainly not insignificant, it's not as if static objects are worthless! Not only do they provide us with handy tools to implement data abstraction and modularity, they also offer all the power of explicit inheritance. In other words, static objects are better than no objects.
As we head into spring here in California, I've dug a new garden bed along my North fence line and debunked three Pascal myths. The one thing all three parts of this article have in common is the basic unity of procedural and pointer types. With any luck, this `seed notion' will bear as fruitfully for you as the raspberries and blueberries I'll probably plant along the fence. One area you may wish to explore is using procedural types and indirect calls to implement procedure aliases without using DLLs.
I hope this article has born more fruit than that garden bed has!
Copyright © 1992, Jon Shemitz - jon@midnightbeach.com - html markup 9-3-94..10-16-94