/*++ Copyright (c) 1989 Microsoft Corporation Module Name: atkdep.c Abstract: This module contains the routines interacting with the mac for ethernet tokenring, localtalk and fddi written for NT. Author: Nikhil Kamkolkar (NikhilK) 8-July-1992 Notes: Hardware specific routines for access the various kinds of physical ports that we support. Xxxxx is a port type and may be: Ethernet LocalTalk TokenRing Fddi HalfPort For each port type, the following four routines must be filled in: BOOLEAN InitializeXxxxxController(port, controllerInfo) int port; char *controllerInfo; port - Index into "portDescriptors" for the port we're interested in. controllerInfo - Hardware (or driver) specific information as passed to "Initialize" to identify the port we're supposed to query. For HalfPort this guy is called: InitializeHalfPort. BOOLEAN AddXxxxxMulticastAddresses(port, numberOfAddresses, addressList) int port; int numberOfAddresses; char *addressList; port - Index into "portDescriptors" for the port we're interested in. numberOfAddreses - How many hardware addresses (packed) are in the addressList. addressList - List of hardware multicast addresses. Not used for HalfPort. BOOLEAN RemoveXxxxxMulticastAddresses(port, numberOfAddresses, addressList) int port; int numberOfAddresses; char *addressList; port - Index into "portDescriptors" for the port we're interested in. numberOfAddreses - How many hardware addresses (packed) are in the addressList. addressList - List of hardware multicast addresses. Not used for HalfPort. BOOLEAN FindMyXxxxxAddress(port, controllerInfo, address) int port; char *controllerInfo; char *address; port - Index into "portDescriptors" for the port we're interested in. controllerInfo - Hardware (or driver) specific information as passed to "Initialize" to identify the port we're supposed to query. address - Filled in with host hardware address. Not used for HalfPort. BOOLEAN XxxxxPacketOut(port, chain, length, transmitCompleteHandler, userData) int port; BufferDescriptor chain; int length; port - Index into "portDescriptors" for the port we're sending the packet out on. packet - A descriptor chain of a full, on-the-wire, packet. length - Length, in bytes, of the packet. transmitCompleteHandler - Optional pointer to routine to be called on transmit completion. userData - User data for the above routine. void BuildXxxxHeader(lapPacket, port, destination, protocol) char *lapPacket; int port; char *destination; int routingInfoLength; char *routingInfo; LogicalProtocol protocol; ddpPacket - Address of the start of the LAP packet, we'll put the header IN FRONT of this address. port - Index into "portDescriptors" for the port we're interested in. destination - Destination hardware address (if "empty" the broadcast address will be used). routingInfoLength - For links with routing info (e.g. TokenRing) number of bytes in next field. routingInfo - Optional bytes of routing info. protocol - Logical protocol type (AppleTalk or AddressResolution). void XxxxPacketInAT(port, packet, length) void XxxxPacketInAARP(port, packet, length) int port; char *packet; int length; port - Index into "portDescriptors" for the port the packet arrived from. packet - A full on-the-wire packet. length - Length, in bytes, of the packet. AARP is not used for HalfPorts or AppleTalkRemoteAccess. Revision History: --*/ #include "atalknt.h" // Move to atkdep.h // // Local routines // #define Build802dot2header(Packet, Protocol) \ { \ (Packet)[Ieee802dot2dsapOffset] = SnapSap; \ (Packet)[Ieee802dot2ssapOffset] = SnapSap; \ (Packet)[Ieee802dot2controlOffset] = UnnumberedInformation; \ if (Protocol == AppleTalk) { \ MoveMem( \ (Packet) + Ieee802dot2protocolOffset, \ appleTalkProtocolType, \ Ieee802dot2protocolTypeLength); \ } else { \ MoveMem( \ (Packet) + Ieee802dot2protocolOffset, \ aarpProtocolType, \ Ieee802dot2protocolTypeLength); \ } \ } LOCAL VOID NTAddMulticastAddressesComplete(NDIS_STATUS Status, PVOID Context); VOID NTTransmitComplete( BufferDescriptor Chain ) { if (Chain->transmitCompleteHandler != NULL) (*Chain->transmitCompleteHandler)(ATnoError, Chain->userData, Chain); FreeBufferChain(Chain); return; } VOID NTAddMulticastAddressesComplete( NDIS_STATUS Status, PVOID Context ) { int Port=(int)Context; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_IMPOSSIBLE, ("INFO1: NTAddMulticastAddrComplete - added port %lx status %lx\n", Port, Status)); // // BUGBUG: // If failure, then shutdown port // return; } // // // ETHERNET 802.3 MEDIA ROUTINES // // BOOLEAN NTInitializeEthernetController( int Port, PCHAR ControllerInfo ) /*++ Routine Description: This routine is called by the portable code to intialize the ethernet controller on a port Arguments: Port- The port number associated with the adapter to be initialized ControllerInfo- Pointer currently always NULL Return Value: TRUE- If initialized, FALSE otherwise --*/ { ULONG atalkFilter; BOOLEAN result = FALSE; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("INFO1: NTInitializeEthernetController - port #%d\n", Port)); ASSERT(Port <= NumberOfPorts); ASSERT(NdisPortDesc[Port].PortState == STATE_BOUND); do { // // Check port number // Check to see it we bound successfully to this adapter // if ((Port > NumberOfPorts) || (NdisPortDesc[Port].PortState != STATE_BOUND)) { break; } // // Set the lookahead size- this must be atleast the size of the max // link + ddp + aarp data size (= MaximumAarpPacketSize in portable code // base) // // All MACs are required to support *atleast* 256 bytes of lookahead data // by NDIS 3.0 // if (!AtalkNdisSetLookaheadSize( Port, MAX_AARPPACKETSIZE, NdisMedium802_3, SYNC, NULL, NULL)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeEthernetController - Set lookahead size failed!\n")); break; } // // Set the broadcast address in the multicast list // if (!NTAddEthernetMulticastAddresses( Port, 1, // Number of addresses ethernetBroadcastAddress)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeEthernetController - Add broadcast failed!\n")); break; } // // Initialize the adapter to start receiving packets // atalkFilter = NDIS_PACKET_TYPE_DIRECTED | NDIS_PACKET_TYPE_MULTICAST; if (!AtalkNdisSetPacketFilter( Port, atalkFilter, NdisMedium802_3, SYNC, NULL, NULL)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeEthernetController - Set filter failed!\n")); break; } result = TRUE; break; } while (FALSE); return(result); } BOOLEAN NTAddEthernetMulticastAddresses( int Port, int NumberOfAddresses, PCHAR AddressList ) /*++ Routine Description: This routine is called by the portable code to add an ethernet address. This routine will preserve whatever previous addresses were added N.B The portable code never passes in more than one address though. Both for Add and Remove Arguments: Port- The port number associated with the adapter to be initialized NumberOfAddress- The number of addresses in the list AddressList- The list of addresses Return Value: TRUE- If initialized, FALSE otherwise --*/ { // // The caller could have passed the address of list which could // be on the stack. since the stack is swappable and since we // block, we make a copy of the passed in list after allocating // required amount of space. We assume address list is of the form // ARRAY[X] where each element is (CHAR [6]) // DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("NTAddEthernetMulticastAddresses: PORT #%d NumAddr %d\n", Port, NumberOfAddresses)); if (NumberOfAddresses != 1) { KeBugCheck(0); } // // Check to see it we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: We did not bind to this port!\n")); return(FALSE); } // // Grab the perport spinlock // ACQUIRE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); if (NdisPortDesc[Port].MulticastAddressList == NULL) { // // Allocate space for the address // NdisPortDesc[Port].MulticastAddressList = (PCHAR)AtalkCallocNonPagedMemory(ETHERNET_ADDRESSLENGTH,sizeof(CHAR)); if (NdisPortDesc[Port].MulticastAddressList == NULL) { // // Release the spinlock // RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return(FALSE); } NdisPortDesc[Port].MulticastAddressListSize = 0; NdisPortDesc[Port].MulticastAddressBufferSize = ETHERNET_ADDRESSLENGTH; } else { PCHAR tempList; // // reallocate! // tempList = (PCHAR)AtalkCallocNonPagedMemory( NdisPortDesc[Port].MulticastAddressListSize+ \ ETHERNET_ADDRESSLENGTH, sizeof(CHAR)); if (tempList == NULL) { // // Release the spinlock // RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return(FALSE); } // // Copy the old list over to the new space // RtlMoveMemory( tempList, NdisPortDesc[Port].MulticastAddressList, NdisPortDesc[Port].MulticastAddressListSize); // // Set the proper values back into NdisPortDesc after freeing the old // list // AtalkFreeNonPagedMemory(NdisPortDesc[Port].MulticastAddressList); NdisPortDesc[Port].MulticastAddressBufferSize = NdisPortDesc[Port].MulticastAddressListSize+ETHERNET_ADDRESSLENGTH; NdisPortDesc[Port].MulticastAddressList = tempList; } // // Guaranteed space is available to copy the new address // Ready to copy our new address here and then do the set! // RtlMoveMemory( NdisPortDesc[Port].MulticastAddressList+ \ NdisPortDesc[Port].MulticastAddressListSize, AddressList, ETHERNET_ADDRESSLENGTH); NdisPortDesc[Port].MulticastAddressListSize += ETHERNET_ADDRESSLENGTH; NdisPortDesc[Port].MulticastAddressBufferSize -= ETHERNET_ADDRESSLENGTH; // // Release the spinlock // RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return (AtalkNdisSetMulticastAddressList( Port, NdisPortDesc[Port].MulticastAddressList, NdisPortDesc[Port].MulticastAddressListSize, NdisMedium802_3, SYNC_IF_POSSIBLE, NTAddMulticastAddressesComplete, (PVOID)Port)); // Context value passed in above } BOOLEAN NTRemoveEthernetMulticastAddrs( INT Port, INT NumberOfAddresses, PCHAR AddressList ) /*++ Routine Description: This routine is called by the portable code to remove a list of ethernet addresses. N.B The portable code never passes in more than one address though. Both for Add and Remove Arguments: Port- The port number associated with the adapter to be initialized NumberOfAddress- The number of addresses in the list AddressList- The list of addresses to be removed Return Value: TRUE- If initialized, FALSE otherwise --*/ { PCHAR currentList; INT numberInList, noAddr; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS0, ("INFO0: NTRemoveEthernetMulticastAddrs - called\n")); if (NumberOfAddresses != 1) { KeBugCheck(0); } // // Check to see it we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTRemoveEthernetMulticastAddrs - not bound to this port!\n")); return(FALSE); } if (NdisPortDesc[Port].MulticastAddressList == NULL) { // // Nothing to remove! // DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTRemoveEthernetMulticastAddrs - nothing to remote!\n")); DBGBRK(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR); return(FALSE); } // // Grab the perport spinlock // ACQUIRE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); numberInList = NdisPortDesc[Port].MulticastAddressListSize/ETHERNET_ADDRESSLENGTH; currentList = NdisPortDesc[Port].MulticastAddressList; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("INFO1: NTRemoveEthernetMulticastAddrs - Number of addresses in list %d\n", numberInList)); ASSERT( (NdisPortDesc[Port].MulticastAddressListSize % ETHERNET_ADDRESSLENGTH) == 0); for (noAddr = 0; noAddr < numberInList; noAddr++) { if (RtlCompareMemory( currentList+(noAddr*ETHERNET_ADDRESSLENGTH), AddressList, ETHERNET_ADDRESSLENGTH) == ETHERNET_ADDRESSLENGTH) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS0, ("INFO0: NTRemoveEthernetMulticastAddrs - Match found!\n")); break; } } // // Was this the last address or was address not present in the list? // if (noAddr == numberInList-1) { // // This was the last address, just reduce the size of the list // NdisPortDesc[Port].MulticastAddressListSize -= ETHERNET_ADDRESSLENGTH; NdisPortDesc[Port].MulticastAddressBufferSize += ETHERNET_ADDRESSLENGTH; } else if (noAddr != numberInList) { // // Move the remaining addresses overwriting the address to be removed // RtlMoveMemory( currentList+(noAddr*ETHERNET_ADDRESSLENGTH), currentList+2*(noAddr+ETHERNET_ADDRESSLENGTH), NdisPortDesc[Port].MulticastAddressListSize-(noAddr*ETHERNET_ADDRESSLENGTH)); NdisPortDesc[Port].MulticastAddressListSize -= ETHERNET_ADDRESSLENGTH; NdisPortDesc[Port].MulticastAddressBufferSize += ETHERNET_ADDRESSLENGTH; } else { // // Address was not present in the list // ASSERT(0); } // // Release the spinlock // RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return (AtalkNdisSetMulticastAddressList( Port, NdisPortDesc[Port].MulticastAddressList, NdisPortDesc[Port].MulticastAddressListSize, NdisMedium802_3, SYNC_IF_POSSIBLE, NULL, NULL)); } BOOLEAN NTFindMyEthernetAddress( int Port, PCHAR ControllerInfo, PCHAR Address ) /*++ Routine Description: This routine is called by the portable code to add a list of ethernet addresses. This routine will preserve whatever previous addresses were added N.B The portable code never passes in more than one address though. Both for Add and Remove Arguments: Port- The port number associated with the adapter to be initialized ControllerInfo- Pointer currently always NULL (use global NdisPortDesc) Address- The buffer for the address to be returned (atleast MAX_ETHERNETADDRLEN bytes) Return Value: TRUE- If obtained, FALSE otherwise --*/ { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("NTFindMyEthernetAddresses: PORT #%d\n", Port)); // // Check to see if we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: We did not bind to this port!\n")); return(FALSE); } if (!AtalkNdisGetCurrentStationAddress( Port, Address, ETHERNET_ADDRESSLENGTH, NdisMedium802_3, SYNC, NULL, NULL)) { return(FALSE); } return(TRUE); } BOOLEAN NTEthernetPacketOut( int Port, BufferDescriptor Chain, int Length, TRANSMIT_COMPLETION *TransmitCompleteRoutine, ULONG UserData ) /*++ Routine Description: This routine is called by the portable code to send a packet out on ethernet Arguments: Port- The port number associated with the adapter to be initialized Chain- The portable stacks buffer descriptor chain Length- The cumulative length to send Return Value: TRUE- If sent/pending, FALSE otherwise TransmitComplete is called if this call pended by completion code --*/ { // // Create an NDIS Packet descriptor for all the buffer descriptor and // send the packet // Chain->transmitCompleteHandler = TransmitCompleteRoutine; Chain->userData = UserData; return(AtalkNdisPacketOut(Port, Chain, Length)); } PCHAR NTBuildEthernetHeader( PUCHAR DdpPacket, int DdpLength, int Port, PUCHAR Destination, PUCHAR RoutingInfo, int RoutingInfoLength, LOGICAL_PROTOCOL Protocol ) /*++ Routine Description: Arguments: Return Value: --*/ { PUCHAR packet = DdpPacket - EthernetLinkHeaderLength - Ieee802dot2headerLength; USHORT trueLength = (USHORT)(DdpLength + Ieee802dot2headerLength); UNREFERENCED_PARAMETER(RoutingInfoLength); UNREFERENCED_PARAMETER(RoutingInfo); // Set destination address. if (Destination isnt empty) { MoveMem( packet + EthernetDestinationOffset, Destination, EthernetAddressLength); } else { MoveMem( packet + EthernetDestinationOffset, ethernetBroadcastAddress, EthernetAddressLength); } // Set source address. MoveMem( packet + EthernetSourceOffset, portDescriptors[Port].myAddress, EthernetAddressLength); // Set length, excluding Ethernet hardware header. packet[EthernetLengthOffset] = (UCHAR)(trueLength >> 8); packet[EthernetLengthOffset + 1] = (UCHAR)(trueLength & 0xFF); // Build the 802.2 header. Build802dot2header(packet + EthernetLinkHeaderLength, Protocol); return(packet); } // NTBuildEthernetHeader // // // TOKENRING 802.5 MEDIA ROUTINES // // BOOLEAN NTInitializeTokenRingController( int Port, PCHAR ControllerInfo ) /*++ Routine Description: This routine is called by the portable code to intialize the ethernet controller on a port Arguments: Port- The port number associated with the adapter to be initialized ControllerInfo- Pointer currently always NULL (use global NdisPortDesc) Return Value: TRUE- If initialized, FALSE otherwise --*/ { ULONG atalkFilter; BOOLEAN result=FALSE; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("INFO1: NTInitializeTokenRingController - PORT #%d\n", Port)); ASSERT(Port <= NumberOfPorts); ASSERT(NdisPortDesc[Port].PortState == STATE_BOUND); do { // // Check port number // Check to see it we bound successfully to this adapter // if ((Port > NumberOfPorts) || (NdisPortDesc[Port].PortState != STATE_BOUND)) { break; } // // Set the lookahead size- this must be atleast the size of the max // link + ddp + aarp data size (= MaximumAarpPacketSize in portable code base) // All MACs are required to support *atleast* 256 bytes of lookahead data // by NDIS 3.0 // if (!AtalkNdisSetLookaheadSize( Port, MAX_AARPPACKETSIZE, NdisMedium802_5, SYNC, NULL, NULL)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeTokenRing - Set loolahead failed!\n")); break; } // Set the broadcast address in the multicast list if (!NTAddTokenRingFunctionalAddresses( Port, 1, // Number of addresses tokenRingBroadcastAddress)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeTokenRing - Set functional failed!\n")); break; } // Initialize the adapter to start receiving packets atalkFilter = NDIS_PACKET_TYPE_DIRECTED | NDIS_PACKET_TYPE_FUNCTIONAL; if (!AtalkNdisSetPacketFilter( Port, atalkFilter, NdisMedium802_5, SYNC, NULL, NULL)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeTokenRing - Set filter failed!\n")); break; } result=TRUE; } while (FALSE); return(result); } BOOLEAN NTAddTokenRingFunctionalAddresses( int Port, int NumberOfAddresses, PCHAR Address ) /*++ Routine Description: This routine is called by the portable code to add an ethernet address. This routine will preserve whatever previous addresses were added N.B The portable code never passes in more than one address though. Both for Add and Remove Arguments: Port- The port number associated with the adapter to be initialized NumberOfAddress- The number of addresses in the list AddressList- The list of addresses Return Value: TRUE- If initialized, FALSE otherwise --*/ { // // The caller could have passed the address of list which could // be on the stack. since the stack is swappable and since we // block, we make a copy of the passed in list after allocating // required amount of space. We assume address list is of the form // ARRAY[X] where each element is (CHAR [6]) // DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS0, ("INFO0: NTAddTokenRingMulticastAddresses: PORT #%d NumAddr %d\n", Port, NumberOfAddresses)); if (NumberOfAddresses != 1) { KeBugCheck(0); } // // Check to see it we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTAddTokenRingFunctionalAddresses - not bound to this port!\n")); return(FALSE); } // // Grab the perport spinlock // ACQUIRE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); // // We only need the last four bytes of the address assuming that the // first two bytes always remain the same (C000) and that the MAC assumes // the same- NDIS 3.0 OID length = 4 // NdisPortDesc[Port].FunctionalAddress |= *(PULONG)&Address[2]; RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return (AtalkNdisSetFunctionalAddress( Port, (PCHAR)&NdisPortDesc[Port].FunctionalAddress, sizeof(NdisPortDesc[Port].FunctionalAddress), NdisMedium802_5, SYNC_IF_POSSIBLE, NTAddMulticastAddressesComplete, (PVOID)Port)); // Context value passed in above } BOOLEAN NTRemoveTokenRingFunctionalAddresses( INT Port, INT NumberOfAddresses, PCHAR Address ) /*++ Routine Description: This routine is called by the portable code to remove a list of ethernet addresses. N.B The portable code never passes in more than one address though. Both for Add and Remove Arguments: Port- The port number associated with the adapter to be initialized NumberOfAddress- The number of addresses in the list AddressList- The list of addresses to be removed Return Value: TRUE- If initialized, FALSE otherwise --*/ { if (NumberOfAddresses != 1) { KeBugCheck(0); } // // Check to see it we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTRemoveTokenRingFunctionalAddresses - not bound to this port!\n")); return(FALSE); } // // Grab the perport spinlock // ACQUIRE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); // // We only use the last four bytes of the address assuming that the // first two bytes always remain the same // NdisPortDesc[Port].FunctionalAddress &= ~(*(PULONG)&Address[2]); RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return (AtalkNdisSetFunctionalAddress( Port, (PCHAR)&NdisPortDesc[Port].FunctionalAddress, sizeof(NdisPortDesc[Port].FunctionalAddress), NdisMedium802_5, SYNC_IF_POSSIBLE, NULL, NULL)); } BOOLEAN NTFindMyTokenRingAddress( int Port, PCHAR ControllerInfo, PCHAR Address ) /*++ Routine Description: This routine is called by the portable code to add a list of ethernet addresses. This routine will preserve whatever previous addresses were added N.B The portable code never passes in more than one address though. Both for Add and Remove Arguments: Port- The port number associated with the adapter to be initialized ControllerInfo- Pointer currently always NULL (use global NdisPortDesc) Address- The buffer for the address to be returned (atleast MAX_yyTOKENRINGADDRLEN bytes) Return Value: TRUE- If obtained, FALSE otherwise --*/ { BOOLEAN result=FALSE; do { // // Check to see if we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTFindMyTokenRingAddress - not bound to this port!\n")); break; } if (!AtalkNdisGetCurrentStationAddress( Port, Address, TOKRING_ADDRESSLENGTH, NdisMedium802_5, SYNC, NULL, NULL)) { break; } result = TRUE; } while (FALSE); return(result); } BOOLEAN NTTokenRingPacketOut( int Port, BufferDescriptor Chain, int Length, TRANSMIT_COMPLETION *TransmitCompleteRoutine, ULONG UserData ) /*++ Routine Description: This routine is called by the portable code to send a packet out on ethernet Arguments: Port- The port number associated with the adapter to be initialized Chain- The portable stacks buffer descriptor chain Length- The cumulative length to send Return Value: TRUE- If sent/pending, FALSE otherwise TransmitComplete is called if this call pended by completion code --*/ { // // Create an NDIS Packet descriptor for all the buffer descriptor and // send the packet // Chain->transmitCompleteHandler = TransmitCompleteRoutine; Chain->userData = UserData; return(AtalkNdisPacketOut(Port, Chain, Length)); } PCHAR NTBuildTokenRingHeader( PCHAR DdpPacket, int DdpLength, int port, PCHAR Destination, PCHAR RoutingInfo, int RoutingInfoLength, LOGICAL_PROTOCOL Protocol ) /*++ Routine Description: Arguments: Return Value: --*/ { PCHAR packet; static UCHAR broadcastRoutingInfo[TokenRingMinRoutingBytes] = TokenRingBroadcastRoutingInfo; static UCHAR simpleRoutingInfo[TokenRingMinRoutingBytes] = TokenRingSimpleRoutingInfo; static UCHAR broadcastDestinationHeader[TokenRingBroadcastDestLength] = TokenRingBroadcastDestHeader; UNREFERENCED_PARAMETER(DdpLength); // // Move the "start of packet" back preceeding any headers we'll // prepend. // if (Destination == NULL) { // Broadcast? packet = DdpPacket - TokenRingRoutingInfoOffset - TokenRingMinRoutingBytes - Ieee802dot2headerLength; RoutingInfo = broadcastRoutingInfo; RoutingInfoLength = TokenRingMinRoutingBytes; } else if (RoutingInfoLength isnt 0) { // Known route? packet = DdpPacket - TokenRingRoutingInfoOffset - RoutingInfoLength - Ieee802dot2headerLength; } else if (FixedCompareCaseSensitive( Destination, TokenRingBroadcastDestLength, broadcastDestinationHeader, TokenRingBroadcastDestLength)) { // Multicast? packet = DdpPacket - TokenRingRoutingInfoOffset - TokenRingMinRoutingBytes - Ieee802dot2headerLength; RoutingInfo = broadcastRoutingInfo; RoutingInfoLength = TokenRingMinRoutingBytes; } else { // No routing know; use simple non-broadcast packet = DdpPacket - TokenRingRoutingInfoOffset - TokenRingMinRoutingBytes - Ieee802dot2headerLength; RoutingInfo = simpleRoutingInfo; RoutingInfoLength = TokenRingMinRoutingBytes; } // Set funny header byte values. packet[TokenRingAccessControlOffset] = TokenRingAccessControlValue; packet[TokenRingFrameControlOffset] = TokenRingFrameControlValue; // Set detination address. if (Destination != NULL) MoveMem( packet + TokenRingDestinationOffset, Destination, TokenRingAddressLength); else MoveMem( packet + TokenRingDestinationOffset, tokenRingBroadcastAddress, TokenRingAddressLength); // Set source address. MoveMem( packet + TokenRingSourceOffset, portDescriptors[port].myAddress, TokenRingAddressLength); // Move in routing info. MoveMem( packet + TokenRingRoutingInfoOffset, RoutingInfo, RoutingInfoLength); packet[TokenRingSourceOffset] |= TokenRingSourceRoutingMask; // Build the 802.2 header. Build802dot2header( (PUCHAR)(packet + TokenRingRoutingInfoOffset + RoutingInfoLength), Protocol); // All set! return(packet); } // NTBuildTokenRingHeader // // // LOCALTALK MEDIA ROUTINES // // BOOLEAN NTInitializeLocalTalkController( int Port, PCHAR ControllerInfo ) /*++ Routine Description: This routine is called by the portable code to intialize the LOCALTALK controller on a port Arguments: Port- The port number associated with the adapter to be initialized ControllerInfo- Pointer currently always NULL (use global NdisPortDesc) Return Value: TRUE- If initialized, FALSE otherwise --*/ { ULONG atalkFilter; BOOLEAN result=FALSE; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("INFO1: NTInitializeLocaltalkController: PORT #%d\n", Port)); ASSERT(Port <= NumberOfPorts); ASSERT(NdisPortDesc[Port].PortState == STATE_BOUND); do { // // Check port number // Check to see it we bound successfully to this adapter // if ((Port > NumberOfPorts) || (NdisPortDesc[Port].PortState != STATE_BOUND)) { break; } // // Set the lookahead size- this must be atleast the size of the max // link + ddp + aarp data size (= MaximumAarpPacketSize in portable code base) // All MACs are required to support *atleast* 256 bytes of lookahead data // by NDIS 3.0 // if (!AtalkNdisSetLookaheadSize( Port, MAX_AARPPACKETSIZE, NdisMediumLocalTalk, SYNC, NULL, NULL)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeLocalTalk - Set loolahead failed!\n")); break; } // Initialize the adapter to start receiving packets atalkFilter = NDIS_PACKET_TYPE_DIRECTED | NDIS_PACKET_TYPE_FUNCTIONAL; if (!AtalkNdisSetPacketFilter( Port, atalkFilter, NdisMediumLocalTalk, SYNC, NULL, NULL)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeLocalTalk - Set filter failed!\n")); break; } result=TRUE; } while (FALSE); return(result); } BOOLEAN NTFindMyLocalTalkAddress( int Port, PCHAR ControllerInfo, PCHAR Address ) /*++ Routine Description: Arguments: Port- The port number associated with the adapter to be initialized ControllerInfo- Pointer currently always NULL (use global NdisPortDesc) Address- The buffer for the address to be returned (atleast LOCALTALKNODEID bytes) Return Value: TRUE- If obtained, FALSE otherwise --*/ { BOOLEAN result=FALSE; // // We can't use AARP for LocalTalk, so we have to play the silly ALAP ENQ // games. This is special case, so we'll treat it as hardware specific. // This routine is allowed to block until it comes up with an answer. A // negative return value indicates no luck. // do { // // Check to see if we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTFindMyLocaltalkAddress - not bound to this port!\n")); break; } if (!AtalkNdisGetCurrentStationAddress( Port, (PCHAR)Address, LOCALTALK_NODEIDLENGTH, NdisMediumLocalTalk, SYNC, NULL, NULL)) { break; } result = TRUE; } while (FALSE); return(result); } BOOLEAN NTLocalTalkPacketOut( int Port, BufferDescriptor Chain, int Length, TRANSMIT_COMPLETION *TransmitCompleteRoutine, ULONG UserData ) /*++ Routine Description: This routine is called by the portable code to send a packet out on ethernet Arguments: Port- The port number associated with the adapter to be initialized Chain- The portable stacks buffer descriptor chain Length- The cumulative length to send Return Value: TRUE- If sent/pending, FALSE otherwise TransmitComplete is called if this call pended by completion code --*/ { // // Create an NDIS Packet descriptor for all the buffer descriptor and // send the packet // Chain->transmitCompleteHandler = TransmitCompleteRoutine; Chain->userData = UserData; return(AtalkNdisPacketOut(Port, Chain, Length)); } PCHAR NTBuildLocalTalkHeader( PCHAR DdpPacket, int ExtendedDdpHeaderFlag, int Port, PCHAR Destination, PCHAR RoutingInfo, int RoutingInfoLength, LOGICAL_PROTOCOL Protocol ) /*++ Routine Description: Arguments: Return Value: --*/ { PCHAR packet = DdpPacket - LapHeaderLength; // Touch unused formals. UNREFERENCED_PARAMETER(RoutingInfoLength); UNREFERENCED_PARAMETER(RoutingInfo); UNREFERENCED_PARAMETER(Protocol); // Fill in LAP header. if (Destination != NULL) packet[AlapDestinationOffset] = *Destination; else packet[AlapDestinationOffset] = AppleTalkBroadcastNodeNumber; ASSERT(portDescriptors[Port].activeNodes != NULL); packet[AlapSourceOffset] = portDescriptors[Port].activeNodes->extendedNode.nodeNumber; packet[AlapTypeOffset] = (UCHAR)(ExtendedDdpHeaderFlag ? AlapLongDdpHeaderType : AlapShortDdpHeaderType); return(packet); } // NTBuildLocalTalkHeader // // // FDDI MEDIA ROUTINES // // BOOLEAN NTInitializeFddiController( int Port, PCHAR ControllerInfo ) /*++ Routine Description: This routine is called by the portable code to intialize the fddi controller on a port Arguments: Port- The port number associated with the adapter to be initialized ControllerInfo- Pointer currently always NULL Return Value: TRUE- If initialized, FALSE otherwise --*/ { ULONG atalkFilter; BOOLEAN result = FALSE; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("INFO1: NTInitializeFddiController - port #%d\n", Port)); ASSERT(Port <= NumberOfPorts); ASSERT(NdisPortDesc[Port].PortState == STATE_BOUND); do { // // Check port number // Check to see it we bound successfully to this adapter // if ((Port > NumberOfPorts) || (NdisPortDesc[Port].PortState != STATE_BOUND)) { break; } // // Set the lookahead size- this must be atleast the size of the max // link + ddp + aarp data size (= MaximumAarpPacketSize in portable code // base) // // All MACs are required to support *atleast* 256 bytes of lookahead data // by NDIS 3.0 // if (!AtalkNdisSetLookaheadSize( Port, MAX_AARPPACKETSIZE, NdisMediumFddi, SYNC, NULL, NULL)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeFddiController - Set lookahead size failed!\n")); break; } // // Set the broadcast address in the multicast list // if (!NTAddFddiMulticastAddresses( Port, 1, // Number of addresses ethernetBroadcastAddress)) { // use ethernet multicast address DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeFddiController - Add broadcast failed!\n")); break; } // // Initialize the adapter to start receiving packets // atalkFilter = NDIS_PACKET_TYPE_DIRECTED | NDIS_PACKET_TYPE_MULTICAST; if (!AtalkNdisSetPacketFilter( Port, atalkFilter, NdisMediumFddi, SYNC, NULL, NULL)) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTInitializeFddiController - Set filter failed!\n")); break; } result = TRUE; break; } while (FALSE); return(result); } BOOLEAN NTAddFddiMulticastAddresses( int Port, int NumberOfAddresses, PCHAR AddressList ) /*++ Routine Description: This routine is called by the portable code to add an ethernet address. This routine will preserve whatever previous addresses were added N.B The portable code never passes in more than one address though. Both for Add and Remove Arguments: Port- The port number associated with the adapter to be initialized NumberOfAddress- The number of addresses in the list AddressList- The list of addresses Return Value: TRUE- If initialized, FALSE otherwise --*/ { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("INFO1: NTAddFddiMulticastAddresses - PORT #%d NumAddr %d\n", Port, NumberOfAddresses)); if (NumberOfAddresses != 1) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_IMPOSSIBLE, ("ERROR: NTAddFddiMulticastAddresses - More than one address\n")); DBGBRK(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_IMPOSSIBLE); return(FALSE); } // // Check to see it we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTAddFddiMulticastAddresses - not bound to port %d!\n", Port)); return(FALSE); } // // Grab the perport spinlock // ACQUIRE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); if (NdisPortDesc[Port].MulticastAddressList == NULL) { // // Allocate space for the address // NdisPortDesc[Port].MulticastAddressList = (PCHAR)AtalkCallocNonPagedMemory(ETHERNET_ADDRESSLENGTH,sizeof(CHAR)); if (NdisPortDesc[Port].MulticastAddressList == NULL) { // // Release the spinlock // RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return(FALSE); } NdisPortDesc[Port].MulticastAddressListSize = 0; NdisPortDesc[Port].MulticastAddressBufferSize = ETHERNET_ADDRESSLENGTH; } else { PCHAR tempList; // // reallocate! // tempList = (PCHAR)AtalkCallocNonPagedMemory( NdisPortDesc[Port].MulticastAddressListSize+ \ ETHERNET_ADDRESSLENGTH, sizeof(CHAR)); if (tempList == NULL) { // // Release the spinlock // RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return(FALSE); } // // Copy the old list over to the new space // RtlMoveMemory( tempList, NdisPortDesc[Port].MulticastAddressList, NdisPortDesc[Port].MulticastAddressListSize); // // Set the proper values back into NdisPortDesc after freeing the old // list // AtalkFreeNonPagedMemory(NdisPortDesc[Port].MulticastAddressList); NdisPortDesc[Port].MulticastAddressBufferSize = NdisPortDesc[Port].MulticastAddressListSize+ETHERNET_ADDRESSLENGTH; NdisPortDesc[Port].MulticastAddressList = tempList; } // // Guaranteed space is available to copy the new address // Ready to copy our new address here and then do the set! // RtlMoveMemory( NdisPortDesc[Port].MulticastAddressList+ \ NdisPortDesc[Port].MulticastAddressListSize, AddressList, ETHERNET_ADDRESSLENGTH); NdisPortDesc[Port].MulticastAddressListSize += ETHERNET_ADDRESSLENGTH; NdisPortDesc[Port].MulticastAddressBufferSize -= ETHERNET_ADDRESSLENGTH; // // Release the spinlock // RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return (AtalkNdisSetMulticastAddressList( Port, NdisPortDesc[Port].MulticastAddressList, NdisPortDesc[Port].MulticastAddressListSize, NdisMediumFddi, SYNC_IF_POSSIBLE, NTAddMulticastAddressesComplete, NULL)); } BOOLEAN NTRemoveFddiMulticastAddrs( INT Port, INT NumberOfAddresses, PCHAR AddressList ) /*++ Routine Description: This routine is called by the portable code to remove a list of ethernet addresses. N.B The portable code never passes in more than one address though. Both for Add and Remove Arguments: Port- The port number associated with the adapter to be initialized NumberOfAddress- The number of addresses in the list AddressList- The list of addresses to be removed Return Value: TRUE- If initialized, FALSE otherwise --*/ { PCHAR currentList; INT numberInList, noAddr; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS0, ("INFO0: NTRemoveFddiMulticastAddrs - called\n")); if (NumberOfAddresses != 1) { KeBugCheck(0); } // // Check to see it we bound successfully to this adapter // if (NdisPortDesc[Port].PortState != STATE_BOUND) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTRemoveEthernetMulticastAddrs - not bound to this port!\n")); return(FALSE); } if (NdisPortDesc[Port].MulticastAddressList == NULL) { // // Nothing to remove! // DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR, ("ERROR: NTRemoveEthernetMulticastAddrs - nothing to remote!\n")); DBGBRK(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_ERROR); return(FALSE); } // // Grab the perport spinlock // ACQUIRE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); numberInList = NdisPortDesc[Port].MulticastAddressListSize/ETHERNET_ADDRESSLENGTH; currentList = NdisPortDesc[Port].MulticastAddressList; DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS1, ("INFO1: NTRemoveEthernetMulticastAddrs - Number of addresses in list %d\n", numberInList)); ASSERT( (NdisPortDesc[Port].MulticastAddressListSize % ETHERNET_ADDRESSLENGTH) == 0); for (noAddr = 0; noAddr < numberInList; noAddr++) { if (RtlCompareMemory( currentList+(noAddr*ETHERNET_ADDRESSLENGTH), AddressList, ETHERNET_ADDRESSLENGTH) == ETHERNET_ADDRESSLENGTH) { DBGPRINT(ATALK_DEBUG_DEPEND, DEBUG_LEVEL_INFOCLASS0, ("INFO0: NTRemoveEthernetMulticastAddrs - Match found!\n")); break; } } // // Was this the last address or was address not present in the list? // if (noAddr == numberInList-1) { // // This was the last address, just reduce the size of the list // NdisPortDesc[Port].MulticastAddressListSize -= ETHERNET_ADDRESSLENGTH; NdisPortDesc[Port].MulticastAddressBufferSize += ETHERNET_ADDRESSLENGTH; } else if (noAddr != numberInList) { // // Move the remaining addresses overwriting the address to be removed // RtlMoveMemory( currentList+(noAddr*ETHERNET_ADDRESSLENGTH), currentList+2*(noAddr+ETHERNET_ADDRESSLENGTH), NdisPortDesc[Port].MulticastAddressListSize-(noAddr*ETHERNET_ADDRESSLENGTH)); NdisPortDesc[Port].MulticastAddressListSize -= ETHERNET_ADDRESSLENGTH; NdisPortDesc[Port].MulticastAddressBufferSize += ETHERNET_ADDRESSLENGTH; } else { // // Address was not present in the list // ASSERT(0); } // // Release the spinlock // RELEASE_SPIN_LOCK(&NdisPortDesc[Port].PerPortLock); return (AtalkNdisSetMulticastAddressList( Port, NdisPortDesc[Port].MulticastAddressList, NdisPortDesc[Port].MulticastAddressListSize, NdisMediumFddi, SYNC_IF_POSSIBLE, NULL, NULL)); } BOOLEAN NTFddiPacketOut( int Port, BufferDescriptor Chain, int Length, TRANSMIT_COMPLETION *TransmitCompleteRoutine, ULONG UserData ) /*++ Routine Description: This routine is called by the portable code to send a packet out on fddi Arguments: Port- The port number associated with the adapter to be initialized Chain- The portable stacks buffer descriptor chain Length- The cumulative length to send Return Value: TRUE- If sent/pending, FALSE otherwise TransmitComplete is called if this call pended by completion code --*/ { // // Create an NDIS Packet descriptor for all the buffer descriptor and // send the packet // Chain->transmitCompleteHandler = TransmitCompleteRoutine; Chain->userData = UserData; return(AtalkNdisPacketOut(Port, Chain, Length)); } PCHAR NTBuildFddiHeader( PCHAR DdpPacket, int DdpLength, int port, PCHAR Destination, PCHAR RoutingInfo, int RoutingInfoLength, LOGICAL_PROTOCOL Protocol ) /*++ Routine Description: Arguments: Return Value: --*/ { PCHAR packet = DdpPacket - FddiLinkHeaderLength - Ieee802dot2headerLength; UNREFERENCED_PARAMETER(RoutingInfoLength); UNREFERENCED_PARAMETER(RoutingInfo); // Set Destination address. Use Fddi boardcast address, as needed. if (Destination isnt empty) MoveMem( packet + FddiDestinationOffset, Destination, FddiAddressLength); else MoveMem( packet + FddiDestinationOffset, ethernetBroadcastAddress, FddiAddressLength); // Set source address. MoveMem( packet + FddiSourceOffset, portDescriptors[port].myAddress, FddiAddressLength); // Build the 802.2 header. Build802dot2header( packet + FddiLinkHeaderLength, Protocol); // All set! return(packet); } // NTBuildFddiHeader #if 0 #if (Iam an OS2) or (Iam a DOS) void near _fastcall FddiPacketInAT(int port, PRXBUFDESC RxDesc, int length) #else void FddiPacketInAT(int port, PCHAR packet, int length) #endif /*++ Routine Description: Arguments: Return Value: --*/ { int ddpLength; #if (Iam an OS2) or (Iam a DOS) PCHAR packet; BOOLEAN mustFree; packet = CopyRxDesc(RxDesc, &mustFree); if (packet is empty) return; #endif // Do we at least have a link, 802.2 and DDP headers? if (length < (FddiLinkHeaderLength + Ieee802dot2headerLength + LongDdpHeaderLength) or length > (FddiLinkHeaderLength + Ieee802dot2headerLength + MaximumLongDdpPacketSize)) { ErrorLog("FddiPacketInAT", ISevVerbose, __LINE__, port, IErrDependBadPacketSize, IMsgDependBadPacketSize, Insert0()); #if (Iam an OS2) or (Iam a DOS) if (mustFree) Free(packet - MaximumHeaderLength); #endif return; } // // First, glean any AARP information that we can, then handle the DDP // packet. This guy also makes sure we have a good 802.2 header... // if (not GleanAarpInfo(port, &packet[FddiSourceOffset], FddiAddressLength, empty, 0, packet + FddiLinkHeaderLength, length - FddiLinkHeaderLength)) { #if (Iam an OS2) or (Iam a DOS) if (mustFree) Free(packet - MaximumHeaderLength); #endif return; // Bad packet or not really for us... } // Pull out the DDP length. ddpLength = ((packet[FddiLinkHeaderLength + Ieee802dot2headerLength + LongDdpLengthOffset] & 0x03) << 8); ddpLength += (packet[FddiLinkHeaderLength + Ieee802dot2headerLength + LongDdpLengthOffset + 1] & 0xFF); if (ddpLength < LongDdpHeaderLength or ddpLength > MaximumLongDdpPacketSize) { ErrorLog("FddiPacketInAT", ISevVerbose, __LINE__, port, IErrDependBadDdpSize, IMsgDependBadDdpSize, Insert0()); #if (Iam an OS2) or (Iam a DOS) if (mustFree) Free(packet - MaximumHeaderLength); #endif return; } // Is the DDP length more than we really got? if (ddpLength + FddiLinkHeaderLength + Ieee802dot2headerLength > length) { ErrorLog("FddiPacketInAT", ISevWarning, __LINE__, port, IErrDependDataMissing, IMsgDependDataMissing, Insert0()); #if (Iam an OS2) or (Iam a DOS) if (mustFree) Free(packet - MaximumHeaderLength); #endif return; } DdpPacketIn(port, packet + FddiLinkHeaderLength + Ieee802dot2headerLength, length - FddiLinkHeaderLength - Ieee802dot2headerLength, False, True, 0, 0); #if (Iam an OS2) or (Iam a DOS) if (mustFree) Free(packet - MaximumHeaderLength); #endif return; } // FddiPacketInAT #if (Iam an OS2) or (Iam a DOS) void near _fastcall FddiPacketInAARP(int port, PRXBUFDESC RxDesc, int length) #else void FddiPacketInAARP(int port, PCHAR packet, int length) #endif /*++ Routine Description: Arguments: Return Value: --*/ { int aarpDataLength; #if (Iam an OS2) or (Iam a DOS) PCHAR packet; BOOLEAN mustFree; packet = CopyRxDesc(RxDesc, &mustFree); if (packet is empty) return; #endif // // Do we need to shrink the length due to Fddi padding? Does this concept // even exist for FDDI??? // if (length is MinimumFddiPacketLength) length = FddiLinkHeaderLength + Ieee802dot2headerLength + MaximumAarpDataSize; aarpDataLength = length - FddiLinkHeaderLength - Ieee802dot2headerLength; if (aarpDataLength > MaximumAarpDataSize or aarpDataLength < MinimumAarpDataSize) { ErrorLog("FddiPacketInAARP", ISevVerbose, __LINE__, port, IErrDependBadAarpSize, IMsgDependBadAarpSize, Insert0()); #if (Iam an OS2) or (Iam a DOS) if (mustFree) Free(packet - MaximumHeaderLength); #endif return; } // Pass the 808.2 portion of the packet on to AARP handling. AarpPacketIn(port, empty, 0, packet + FddiLinkHeaderLength, length - FddiLinkHeaderLength); #if (Iam an OS2) or (Iam a DOS) if (mustFree) Free(packet - MaximumHeaderLength); #endif return; } // FddiPacketInAARP #endif // ************************************************ // ********** HalfPort Specific Routines ********** // ************************************************ BOOLEAN InitializeHalfPort(int port, PCHAR controllerInfo) /*++ Routine Description: Arguments: Return Value: --*/ { port, controllerInfo; return(True); } // InitializeHalfPort BOOLEAN far SendHalfPortPacketsTo(int Port, LOGICAL_PROTOCOL Protocol, RawPacketHandler *Routine) /*++ Routine Description: Arguments: Return Value: --*/ { Port, Protocol, Routine; return(True); } extern BOOLEAN _fastcall HalfPortPacketOut(int port, BufferDescriptor chain, int length, TRANSMIT_COMPLETION *transmitCompleteHandler, long unsigned userData) /*++ Routine Description: Arguments: Return Value: --*/ { #if (IamNot an OS2) and (IamNot a DOS) if (dumpOutgoingPackets) { if ((chain = CoalesceBufferChain(chain)) is Empty) return(False); DecodeEthernetPacket("HalfOut", port, chain->data); } #endif // // If our packet is in more than one chunk and the underlying hardware // doesn't allow gather send, coalesce it here. // if ((chain->next isnt Empty or (chain->onBoardDataValid and chain->outBoardDataValid)) and not portSpecificInfo[portDescriptors[port].portType].gatherSendSupport) if ((chain = CoalesceBufferChain(chain)) is Empty) { ErrorLog("HalfPortPacketOut", ISevError, __LINE__, port, IErrDependSourceNotKnown, IMsgDependSourceNotKnown, Insert0()); return(False); } // // Do whatever is needed to actually transmit the packet! Note that if // "length" is less than the computed length of the buffer chain, the // LAST buffer chunk should be truncated accordingly. // chain->transmitCompleteHandler = transmitCompleteHandler; chain->userData = userData; // DoTheTransmit(chain); // // If we don't have asynchronous transmit completion... free up the buffer // now. // if (portSpecificInfo[portDescriptors[port].portType].synchronousTransmits) TransmitComplete(chain); return(True); } // HalfPortPacketOut #if (Iam an OS2) or (Iam a DOS) void near _fastcall HalfPortPacketInAT(int port, PRXBUFDESC RxDesc, int length) #else void HalfPortPacketInAT(int port, PCHAR packet, int length) #endif /*++ Routine Description: Arguments: Return Value: --*/ { port, packet, length; #if (IamNot an OS2) and (IamNot a DOS) if (dumpIncomingPackets) DecodeEthernetPacket("In", port, packet); #endif return; } // HalfPortPacketInAT char *far BuildHalfPortHeader(char far *DdpPacket, int ddpLength, int port, PCHAR DestinationIgnored, PCHAR RoutingInfoIgnored, int RoutingInfoLengthIgnored, LOGICAL_PROTOCOL ProtocolIgnored) /*++ Routine Description: Arguments: Return Value: --*/ { ddpLength, port, DestinationIgnored, ProtocolIgnored; RoutingInfoIgnored, RoutingInfoLengthIgnored; // // If this HalfPort has a loginal Destination address, it can be found // in "portDescriptors[port].controllerInfo". // return(DdpPacket); } // BuildHalfPortHeader // ***************************************************** // ********** Remote Access Specific Routines ********** // ***************************************************** #if ArapIncluded BOOLEAN InitializeRemoteAccess(int port, PCHAR controllerInfo) /*++ Routine Description: Arguments: Return Value: --*/ { port, controllerInfo; return(True); } // InitializeRemoteAccess BOOLEAN far SendRemoteAccessPacketsTo(int port, LOGICAL_PROTOCOL Protocol, RawPacketHandler *routine) /*++ Routine Description: Arguments: Return Value: --*/ { port, protocol, routine; return(True); } // SendRemoteAccessPacketsTo BOOLEAN _fastcall RemoteAccessPacketOut(int port, BufferDescriptor chain, int length, TRANSMIT_COMPLETION *transmitCompleteHandler, long unsigned userData) /*++ Routine Description: Arguments: Return Value: --*/ { // // Always coalesce remote access packets, we have some preprocessing to // do. // if ((chain = CoalesceBufferChain(chain)) is Empty) { ErrorLog("RemoteAccessPacketOut", ISevError, __LINE__, port, IErrDependSourceNotKnown, IMsgDependSourceNotKnown, Insert0()); return(False); } // // There are some packets that we should be hiding from our client (packets // sent by the client, RTMP boardcasts, NBP lookups from the client). Check // this here. // if (ArapCensorPacket(port, chain->data, length)) { chain->transmitCompleteHandler = transmitCompleteHandler; chain->userData = userData; TransmitComplete(chain); return(True); } #if Verbose or (Iam a Primos) DecodeArapPacket("Out", port, chain->data, length); #endif // // Do whatever is needed to actually transmit the packet! Note that if // "length" is less than the computed length of the buffer chain, the // LAST buffer chunk should be truncated accordingly. // chain->transmitCompleteHandler = transmitCompleteHandler; chain->userData = userData; // DoTheTransmit(chain); // // If we don't have asynchronous transmit completion... free up the buffer // now. // #if IncludeVirtualClient VirtualClientPacketIn(port, chain->data); #endif if (portSpecificInfo[portDescriptors[port].portType].synchronousTransmits) TransmitComplete(chain); return(True); } // RemoteAccessPacketOut void RemoteAccessPacketInAT(int port, PCHAR packet, int length) /*++ Routine Description: Arguments: Return Value: --*/ { // Handle the packet. #if Verbose or (Iam a Primos) DecodeArapPacket("In", port, packet, length); #endif ArapIncomingPacket(port, packet, length); } // RemoteAccessPacketInAT char far *far BuildRemoteAccessHeader(char far *DdpPacket, int packetLength, int port, PCHAR Destination, PCHAR RoutingInfo, int RoutingInfoLength, LogicalProtocol arapFlags) /*++ Routine Description: Arguments: Return Value: --*/ { PCHAR packet; int headerLength; short checksum; // Touch unused formals. Destination, RoutingInfoLength, RoutingInfo; // // Complete the length of the header based on what packet type; internal // or data? // if (arapFlags is ArapInternalMessageFlagValue) headerLength = ArapInternalMessageHeaderLength; else headerLength = ArapDataHeaderLength; packet = DdpPacket - headerLength; // Fill in the ARAP header (modem link tool) common to both. MoveShortMachineToWire(packet + ModemLinkToolLengthOffset, headerLength - sizeof(short) + packetLength); packet[ArapFlagsOffset] = (char)arapFlags; // For an internal add int he sequence number, and we're set. if (arapFlags is ArapInternalMessageFlagValue) { packet[ArapSequenceNumberOffset] = portDescriptors[port].remoteAccessInfo->nextOutgoingSequenceNumber; return(packet); } // Otherwise, fill in the LAP header for ARAP. packet[ArapLapHeaderOffset + AlapDestinationOffset] = 0; packet[ArapLapHeaderOffset + AlapSourceOffset] = 0; packet[ArapLapHeaderOffset + AlapTypeOffset] = (char)AlapLongDdpHeaderType; // // Also, set the Ddp length field to zero (don't alter the hop-count), and // if there is a checksum set, reset it to 1. // #if 0 packet[ArapDdpHeaderOffset + LongDdpLengthOffset] |= LongDdpHopCountMask; #else packet[ArapDdpHeaderOffset + LongDdpLengthOffset] &= LongDdpHopCountMask; #endif packet[ArapDdpHeaderOffset + LongDdpLengthOffset + 1] = 0; MoveShortWireToMachine(packet + ArapDdpHeaderOffset + LongDdpChecksumOffset, checksum); if (checksum isnt 0) MoveShortMachineToWire(packet + ArapDdpHeaderOffset + LongDdpChecksumOffset, 1); return(packet); } // BuildRemoteAccessHeader void far RemoteAccessIncomingCall(int port) /*++ Routine Description: Arguments: Return Value: --*/ { // Handle the call. ArapHandleIncomingConnection(port); } // RemoteAccessIncomingCall BOOLEAN far RemoteAccessOutgoingCall(int port, PCHAR modemString) /*++ Routine Description: Arguments: Return Value: --*/ { port, modemString; // Do whatever we need to do to place the outgoing call. return(True); } // RemoteAccessOutgoingCall void far RemoteAccessCallDisconnected(int port) /*++ Routine Description: Arguments: Return Value: --*/ { // Handle the disconnect. ArapHandleConnectionDisconnect(port); } // RemoteAccessCallDisconnected BOOLEAN far RemoteAccessDisconnectCall(int port) /*++ Routine Description: Arguments: Return Value: --*/ { port; // Do whatever is needed to disconnect the connection. return(True); } // RemoteAccessCallDisconnected #endif