[Back to COMM SWAG index] [Back to Main SWAG index] [Original]
UNIT PKTDRVR;
{
ÉÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÑÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍ»
º Filename : PKTDRVR.PAS ³ Program / Unit : [U] º
º Description : Turbo Pascal ÀÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ�
º Object to interface with Crynrware packet drivers. º
º º
ÇÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ�
º Compiler : Turbo Pascal 7.0 º
º OS-Version : MS-DOS 6.0 º
º Last edit : 08-Oct-93 º
º Version : 1.0 º
ÇÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄÄ�
º Author : Oliver Rehmann º
º Copyright : (C) 1993 Oliver Rehmann º
º º
º Released to public domain. º
º The author can not be held responsible for any damages resulting º
º from the use of this software. º
ÈÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍÍͼ
}
INTERFACE
USES
DOS,OBJECTS;
CONST
{ Packet driver interface classes }
CL_NONE = 0;
CL_ETHERNET = 1;
CL_PRONET_10 = 2;
CL_IEEE8025 = 3;
CL_OMNINET = 4;
CL_APPLETALK = 5;
CL_SERIAL_LINE = 6;
CL_STARLAN = 7;
CL_ARCNET = 8;
CL_AX25 = 9;
CL_KISS = 10;
CL_IEEE8023 = 11;
CL_FDDI = 12;
CL_INTERNET_X25 = 13;
CL_LANSTAR = 14;
CL_SLFP = 15;
CL_NETROM = 16;
NCLASS = 17;
{ Packet driver interface types (not a complete list) }
TC500 = 1;
PC2000 = 10;
WD8003 = 14;
PC8250 = 15;
ANYTYPE = $ffff;
{ Packet driver function call numbers. From Appendix B. }
DRIVER_INFO = 1;
ACCESS_TYPE = 2;
RELEASE_TYPE = 3;
SEND_PKT = 4;
TERMINATE = 5;
GET_ADDRESS = 6;
RESET_INTERFACE = 7;
GET_PARAMETERS = 10;
AS_SEND_PKT = 11;
SET_RCV_MODE = 20;
GET_RCV_MODE = 21;
SET_MULTICAST_LIST = 22;
GET_MULTICAST_LIST = 23;
GET_STATISTICS = 24;
SET_ADDRESS = 25;
{ Packet driver error return codes. From Appendix C. }
NO_ERROR = 0;
BAD_HANDLE = 1; { invalid handle number }
NO_CLASS = 2; { no interfaces of specified class found }
NO_TYPE = 3; { no interfaces of specified type found }
NO_NUMBER = 4; { no interfaces of specified number found }
BAD_TYPE = 5; { bad packet type specified }
NO_MULTICAST = 6; { this interface does not support multicast }
CANT_TERMINATE = 7; { this packet driver cannot terminate }
BAD_MODE = 8; { an invalid receiver mode was specified }
NO_SPACE = 9; { operation failed because of insufficient space }
TYPE_INUSE = 10; { the type had previously been accessed, and not
released } BAD_COMMAND = 11; { the command was out of range, or not
implemented } CANT_SEND = 12; { the packet couldn't be sent (usually
hardware error) } CANT_SET = 13; { hardware address couldn't be changed
(> 1 handle open) } BAD_ADDRESS = 14; { hardware address has bad
length or format } CANT_RESET = 15; { couldn't reset interface (> 1
handle open) }
CARRY_FLAG = 1;
CONST
Pkt_Sig : String[08] = 'PKT DRVR';
ParamLen : Byte = 14;
TYPE
TPKTSTATUS = (NO_PKTDRVR,INITIALIZED,NOT_INITIALIZED);
TACCESSTYPE = RECORD
if_class : Byte; { Interface class }
if_type : Word; { Interface Type }
if_number : Byte; { Interface number }
type_ : Pointer;
typelen : Word; { length of type_, set to 0 if
you want to receive all pkts }
receiver : Pointer; { receive handler }
END;
TPKTPARAMS = RECORD
major_rev : Byte; { Major revision ID of packet specs }
minor_rev : Byte; { Minor revision ID of packet specs }
length : Byte; { Length of structure in Bytes }
addr_len : Byte; { Length of a MAC address }
mtu : Word; { MTU, including MAC headers }
multicast_aval: Word; { buffer size for multicast addr. }
rcv_bufs : Word; { (# of back-to-back MTU rcvs) - 1 }
xmt_bufs : Word; { (# of successive xmits) - 1 }
int_num : Word; { Interrupt # to hook for post-EOI
processing, 0 == none }
END;
TDRVRINFO = RECORD
Version : Word; { Packet driver version }
Class : Byte; { Driver class }
Type_ : Word; { Driver type }
Number : Byte; { Driver number }
pName : Pointer;
Functionality : Byte; { How good is this driver }
END;
TSTATISTICS = RECORD
packets_in : LongInt;
packets_out : LongInt;
bytes_in : LongInt;
bytes_out : LongInt;
errors_in : LongInt;
errors_out : LongInt;
packets_lost : LongInt;
END;
TPKTDRVR = OBJECT(TOBJECT)
private
pktInt : Integer;
pktHandle : Integer;
pktRecvHandler : Pointer;
pktStatus : TPKTSTATUS;
pktError : Byte;
pktRegs : Registers;
pktAccessInfo : TACCESSTYPE;
PROCEDURE TestForPktDriver;
public
CONSTRUCTOR Init(IntNo : Integer);
DESTRUCTOR Done; VIRTUAL;
PROCEDURE ScanForPktDriver;
FUNCTION GetStatus : TPKTSTATUS;
FUNCTION GetError : Byte;
FUNCTION GetHandle : Word;
PROCEDURE GetAccessType (VAR pktAccessType : TACCESSTYPE);
PROCEDURE DriverInfo (VAR pktInfo : TDRVRINFO );
PROCEDURE AccessType (VAR pktAccessType : TACCESSTYPE);
PROCEDURE ReleaseType;
PROCEDURE TerminateDriver;
PROCEDURE GetAddress (Buffer : Pointer;BufLen : Word; VAR
BufCopied : Word); PROCEDURE ResetInterface; PROCEDURE GetParameters (VAR
pktParams : TPKTPARAMS);
PROCEDURE SendPkt (Buffer : Pointer;BufLen : Word );
PROCEDURE As_SendPkt (Buffer : Pointer;BufLen : Word;Upcall :
Pointer );
PROCEDURE SetRCVmode (Mode : Word);
FUNCTION GetRCVmode : Word;
PROCEDURE SetMulticastList(VAR mcList : Pointer; VAR mcLen : Word);
PROCEDURE GetMulticastList(VAR mcList : Pointer; VAR mcLen : Word);
PROCEDURE GetStatistics (VAR pktStatistics : TSTATISTICS );
PROCEDURE SetAddress (Address : Pointer; VAR AddrLen : Word);
END;
IMPLEMENTATION
CONSTRUCTOR TPKTDRVR.Init(IntNo : Integer);
BEGIN
Inherited Init;
pktInt := IntNo;
pktStatus := NOT_INITIALIZED;
FillChar(pktAccessInfo,SizeOf(pktAccessInfo),#00);
TestForPktDriver;
END;
DESTRUCTOR TPKTDRVR.Done;
BEGIN
{ Release allocated handle }
IF (pktStatus = INITIALIZED) THEN
BEGIN
ReleaseType;
END;
Inherited Done;
END;
FUNCTION TPKTDRVR.GetStatus : TPKTSTATUS;
BEGIN
GetStatus := pktStatus;
END;
PROCEDURE TPKTDRVR.GetAccessType(VAR pktAccessType : TACCESSTYPE);
BEGIN
pktAccessType := pktAccessInfo;
END;
PROCEDURE TPKTDRVR.TestForPktDriver;
(* Tests if the assigned interrupt points to a valid packet driver. *)
VAR tPointer : Pointer;
Signature : String[08];
I : Integer;
BEGIN
Signature := '';
GetIntVec(pktInt,tPointer);
FOR I := 3 TO 10 DO
BEGIN
Signature := Signature + Chr(Mem[Seg(tPointer^):Ofs(tPointer^)+I]);
END;
IF (POS(Pkt_Sig,Signature) = 0) THEN
pktStatus := NO_PKTDRVR
ELSE
pktStatus := INITIALIZED;
END;
PROCEDURE TPKTDRVR.ScanForPktDriver;
(* Scans interrupts ($60-$7F) for a packet driver. *)
(* Stops if it has found a valid driver. *)
VAR I : Integer;
BEGIN
I := $60; { Lower range of possible pktdrvr interrupt }
REPEAT
pktInt := I;
TestForPktDriver;
Inc(I);
UNTIL (I = $80) OR (pktStatus = INITIALIZED);
END;
PROCEDURE TPKTDRVR.DriverInfo(VAR pktInfo : TDRVRINFO);
BEGIN
WITH pktRegs DO
BEGIN
AH := DRIVER_INFO;
AL := $FF;
BX := pktHandle;
Intr(pktInt,pktRegs); { Call Packet Driver }
IF (pktRegs.Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
BEGIN
pktError := 0;
IF (pktError = NO_ERROR) THEN
BEGIN
pktInfo.Version := BX;
pktInfo.Class := CH;
pktInfo.Type_ := DX;
pktInfo.Number := CL;
pktInfo.pName := Ptr(DS,SI);
pktInfo.Functionality := AL;
END;
END;
END;
END;
PROCEDURE TPKTDRVR.AccessType(VAR pktAccessType : TACCESSTYPE);
(* Accesses the packet driver. *)
BEGIN
WITH pktRegs DO
BEGIN
AH := ACCESS_TYPE;
AL := pktAccessType.if_class;
BX := pktAccessType.if_type;
CX := pktAccessType.typelen;
DL := pktAccessType.if_number;
DS := Seg(pktAccessType.type_^);
SI := Ofs(pktAccessType.type_^);
ES := Seg(pktAccessType.receiver^);
DI := Ofs(pktAccessType.receiver^);
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
BEGIN
pktError := 0;
pktHandle := AX;
pktAccessInfo := pktAccessType;
END;
END;
END;
PROCEDURE TPKTDRVR.ReleaseType;
(* Releases a specific type handle *)
BEGIN
WITH pktRegs DO
BEGIN
AH := RELEASE_TYPE;
BX := pktHandle;
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
pktError := 0;
END;
END;
PROCEDURE TPKTDRVR.SendPkt(Buffer : Pointer;BufLen : Word);
BEGIN
WITH pktRegs DO
BEGIN
AH := SEND_PKT;
CX := BufLen;
DS := Seg(Buffer^);
ES := DS;
SI := Ofs(Buffer^);
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
pktError := 0;
END;
END;
PROCEDURE TPKTDRVR.TerminateDriver;
(* Terminates the Driver associated with pktHandle *)
BEGIN
WITH pktRegs DO
BEGIN
AH := TERMINATE;
BX := pktHandle;
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
pktError := 0;
END;
END;
PROCEDURE TPKTDRVR.GetAddress (Buffer : Pointer;BufLen : Word; VAR BufCopied :
Word);
BEGIN
WITH pktRegs DO
BEGIN
AH := GET_ADDRESS;
BX := pktHandle;
CX := BufLen;
ES := Seg(Buffer^);
DI := Ofs(Buffer^);
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
BEGIN
pktError := 0;
BufCopied := CX;
END;
END;
END;
PROCEDURE TPKTDRVR.ResetInterface;
BEGIN
WITH pktRegs DO
BEGIN
AH := RESET_INTERFACE;
BX := pktHandle;
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
pktError := 0;
END;
END;
PROCEDURE TPKTDRVR.GetParameters(VAR pktParams : TPKTPARAMS);
(* Description : ³ Gets specific parameters from the driver. *)
(* Not all drivers support this function. *)
VAR b : Byte;
BEGIN
WITH pktRegs DO
BEGIN
AH := GET_PARAMETERS;
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
BEGIN
pktError := 0;
FOR b := 0 TO ParamLen-1 DO { Copy contents of structure }
Mem[Seg(pktParams):Ofs(PktParams)+b] := Mem[ES:DI+b];
END;
END;
END;
PROCEDURE TPKTDRVR.As_SendPkt(Buffer : Pointer;BufLen : Word;Upcall :
Pointer);
(* Sends a data packet by accessing the packet driver. *)
(* Upcall is called when order was placed. *)
BEGIN
WITH pktRegs DO
BEGIN
AH := AS_SEND_PKT;
CX := BufLen;
DS := Seg(Buffer);
SI := Ofs(Buffer);
ES := Seg(Upcall^);
DI := Ofs(Upcall^);
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
pktError := 0;
END;
END;
PROCEDURE TPKTDRVR.SetRCVmode(Mode : Word);
BEGIN
WITH pktRegs DO
BEGIN
AH := SET_RCV_MODE;
BX := pktHandle;
CX := Mode;
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
pktError := 0;
END;
END;
FUNCTION TPKTDRVR.GetRCVmode : Word;
BEGIN
WITH pktRegs DO
BEGIN
AH := GET_RCV_MODE;
BX := pktHandle;
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
BEGIN
pktError := 0;
GetRCVmode := AX;
END;
END;
END;
PROCEDURE TPKTDRVR.SetMulticastList(VAR mcList : Pointer; VAR mcLen : Word);
BEGIN
WITH pktRegs DO
BEGIN
AH := SET_MULTICAST_LIST;
CX := mcLen;
ES := Seg(mcList^);
DI := Ofs(mcList^);
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
pktError := 0;
END;
END;
PROCEDURE TPKTDRVR.GetMulticastList(VAR mcList : Pointer; VAR mcLen : Word);
BEGIN
WITH pktRegs DO
BEGIN
AH := GET_MULTICAST_LIST;
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
BEGIN
pktError := 0;
mcList := Ptr(ES,DI);
mcLen := CX;
END;
END;
END;
PROCEDURE TPKTDRVR.GetStatistics(VAR pktStatistics : TSTATISTICS);
VAR b : Byte;
BEGIN
WITH pktRegs DO
BEGIN
AH := GET_STATISTICS;
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
BEGIN
pktError := 0;
FOR b := 0 TO SizeOf(TSTATISTICS)-1 DO { Copy contents of structure }
Mem[Seg(pktStatistics):Ofs(pktStatistics)+b] := Mem[DS:SI+b];
END;
END;
END;
PROCEDURE TPKTDRVR.SetAddress(Address : Pointer; VAR AddrLen : Word);
BEGIN
WITH pktRegs DO
BEGIN
AH := SET_ADDRESS;
CX := AddrLen;
ES := Seg(Address^);
DI := Ofs(Address^);
Intr(pktInt,pktRegs);
IF (Flags AND Carry_Flag) = Carry_Flag THEN
pktError := DH
ELSE
BEGIN
pktError := 0;
AddrLen := CX;
END;
END;
END;
FUNCTION TPKTDRVR.GetError : Byte;
BEGIN
GetError := pktError;
END;
FUNCTION TPKTDRVR.GetHandle : Word;
BEGIN
GetHandle := pktHandle;
END;
BEGIN
END.
{end}
[Back to COMM SWAG index] [Back to Main SWAG index] [Original]