/*++ Copyright (C) Microsoft Corporation, 1996 - 1999 Module Name: Protocol.hxx Abstract: The Protocol object definitions Author: Kamen Moutafov [KamenM] Revision History: KamenM 12/22/1998 Creation --*/ #if _MSC_VER >= 1200 #pragma once #endif #ifndef __PROTOCOL_HXX_ #define __PROTOCOL_HXX_ // uncomment this to turn on PNP debug spew //#define MAJOR_PNP_DEBUG enum TransportProtocolStates { ProtocolNotLoaded, ProtocolLoadedWithoutAddress, ProtocolWasLoadedOrNeedsActivation, ProtocolLoaded, ProtocolWasLoadedOrNeedsActivationWithoutAddress, ProtocolLoadedAndMonitored }; struct BASE_ASYNC_OBJECT; // forwards extern RPC_ADDRESS_CHANGE_FN * AddressChangeFn; void RPC_ENTRY NullAddressChangeFn( PVOID arg ); class TransportProtocol /* This object encapsulates a protocol on the transport level. Used primarily for PnP support. */ { public: TransportProtocol(void) { State = ProtocolNotLoaded; RpcpInitializeListHead(&ObjectList); addressChangeSocket = 0; // provide arbitrary non-zero code, to indicate that this query has not succeeded addressChangeOverlapped.Internal = (ULONG_PTR)-1; } // If a piece of code detects that a protocol is functional through means other // than PnP, it may call this function to notify the PnP code that the protocol is functional // The only effect will be speedier processing of a potential PnP notification void DetectedAsFunctional(PROTOCOL_ID ProtocolId); // Whenever a PnP notification arrives, this method needs to be called for each protocol. // The method is idempotent - it saves the previous state and will no-op redundant calls // However, this method should not be called before the runtime is ready to accept // calls on transport addresses void HandleProtocolChange(IN WSAPROTOCOL_INFO *lpProtocolBuffer, IN int ProtocolCount, IN PROTOCOL_ID thisProtocolId); // Adds an object to the list of objects for this protocol. The list is used to track down // all instances of objects belonging to this protocol when a PnP notification arrives. void AddObjectToList(IN OUT BASE_ASYNC_OBJECT *pObj); // Removes an object from the list of objects on this protocol void RemoveObjectFromList(IN OUT BASE_ASYNC_OBJECT *pObj); // If a NewAddress query fails asynchronously, this function needs to be called which will // retry submitting all failed query for this protocol. For non-failed, or non-submitted queries, // this is a no-op BOOL ResubmitQueriesIfNecessary(PROTOCOL_ID ProtocolId); // Handle a PnP notification. Does all necessary processing static BOOL HandlePnPStateChange(void); // Calls ResubmitQueriesIfNecessary(PROTOCOL_ID ProtocolId) on all protocols. static BOOL ResubmitQueriesIfNecessary(void); // Same as the non-static FunctionalProtocolDetected, except that this one operates // without this pointer - it will retrieve the appropriate object and will call // FunctionalProtocolDetected on it. static void FunctionalProtocolDetected(PROTOCOL_ID ProtocolId); // Same as AddObjectToList except that it will call AddObjectToList on the appropriate // protocol static void AddObjectToProtocolList(IN OUT BASE_ASYNC_OBJECT *pObj); // Same as RemoveObjectFromList except that it will call RemoveObjectFromList on the appropriate // protocol static void RemoveObjectFromProtocolList(IN OUT BASE_ASYNC_OBJECT *pObj); #if defined(DBG) || defined(_DEBUG) // will ASSERT that all protocols are in a consistent state static void AssertTransportProtocolState(void); // will ASSSERT that this protocol is in a consistent state void AssertState(PROTOCOL_ID ProtocolId); // will ASSERT that a protocols list is in a consistent state static void AssertProtocolListIntegrity(IN BASE_ASYNC_OBJECT *pObj); void AssertListIntegrity (IN PROTOCOL_ID ProtocolId); #endif // Initializes transport address objects that listen on // interfaces that were recently assigned addresses. void InitNewAddresses(PROTOCOL_ID ProtocolId); private: // attempts to verify that a protocol is functional, and if found so, advances the state of // the protocol BOOL VerifyProtocolIsFunctional(PROTOCOL_ID ProtocolId); // checks whether the given protocol has an address BOOL DoesAddressSocketHaveAddress(void); // If there is a pending address change request (WSAIoctl), it cancels it // if fForceCancel is FALSE, monitored protocols don't have their // address change requests canceled. If fForceCancel is TRUE, // all protocols get their address change requests cancelled void CancelAddressChangeRequestIfNecessary(IN BOOL fForceCancel, IN PROTOCOL_ID ProtocolId); // Restarts a protocol that was unloaded. void RestartProtocol(PROTOCOL_ID ProtocolId); // Unloads a protocol that was loaded. void UnloadProtocol(PROTOCOL_ID ProtocolId); // Submits an address change query (WSAIoctl) BOOL SubmitAddressChangeQuery(void); // Sets the state of the given protocol. Other code should never directly set the State - it should // always go through this function. This allows for derivative protocols (like HTTP) to mirror // the state of their base protocols - TCP void SetState(TransportProtocolStates newState, PROTOCOL_ID ProtocolId); // will return whether address change monitoring is required for this protocol. Currently this // is used only by RPCSS for SPX (in order to start sending updated SAP broadcasts) // and to initialize TCP/IP transport addresses listening on the NICs that have had // their IP addresses changed in the selective binding scenarios. BOOL IsAddressChangeMonitoringOn(PROTOCOL_ID ProtocolId) { // If this is TCP & SPX, and somebody actually cares about this (i.e. somebody has // registered a notification callback), then turn on address monitoring. // Also turn it on if the firewall table has not yet been fully initialized. // We will use the address change notification to trigger an update of the table. return ( ( ( #ifdef SPX_ON (ProtocolId == SPX) || #endif (ProtocolId == TCP) ) && IsAddressChangeFnDefined() ) || ( (ProtocolId == TCP) && FirewallTableNeedsUpdating() ) ); } BOOL IsAddressChangeFnDefined(void) { return (AddressChangeFn != NullAddressChangeFn && AddressChangeFn != NULL); } BOOL FirewallTableNeedsUpdating(void) { return (pFirewallTable != NULL && fFirewallTableFullyInitialized == FALSE); } // Trailing protocols are protocols that don't have their own independent transport provider // They use the transport provider of another protocol, but for some reason we present it // to the upper layers as different protocols. We call them trailing, because their // state trails the state of the protocol whose transport they use. // Currently we have only one of those - HTTP which trails TCP BOOL IsTrailingProtocol(PROTOCOL_ID ProtocolId) { return (ProtocolId == HTTP); } void SetStateToLoadedAndMonitorProtocolIfNecessary(PROTOCOL_ID ProtocolId); // checks if there is an address change request pending, and if there isn't // submits one. Will also open an address change socket if necessary BOOL MonitorFunctionalProtocolForAddressChange(PROTOCOL_ID ProtocolId); // checks if these's an address change request socket, and if there // isn't it opens one BOOL OpenAddressChangeRequestSocket(PROTOCOL_ID ProtocolId); // if ThisProtocolsId is in ProtocolLoadedAndMonitored or ProtocolLoaded state // move the Mirrored protocol to the corresponding state void MirrorProtocolState ( IN PROTOCOL_ID MirrorProtocolId ); #ifdef MAJOR_PNP_DEBUG // Dumps the protocol state on the debugger for the current protocol void DumpProtocolState(PROTOCOL_ID ProtocolId); // Dumps the protocol state on the debugger for all protocols static void DumpProtocolState(void); #endif // the current state of this protocol TransportProtocolStates State; // the list of opened objects for this protocol LIST_ENTRY ObjectList; // iff there's address change request submitted on this socket, it will be non-null SOCKET addressChangeSocket; // the OVERLAPPED for the address change request (WSAIoctl). It's Internal member will be -1 // if no request has been submitted. Other data members are as normal. OVERLAPPED addressChangeOverlapped; // NOTE: A TransportProtocol object is not aware of its PROTOCOL_ID - it must be // passed externally to it for all methods that require it. This is done to save memory. }; extern TransportProtocol *TransportProtocolArray; // returns the TransportProtocol object for the ProtocolId. Ownership of the object does not pass // to the caller inline TransportProtocol *GetTransportProtocol(IN PROTOCOL_ID ProtocolId) { ASSERT(ProtocolId >= 1); ASSERT(ProtocolId < MAX_PROTOCOLS); return &TransportProtocolArray[ProtocolId]; } inline void TransportProtocol::FunctionalProtocolDetected(PROTOCOL_ID ProtocolId) { GetTransportProtocol(ProtocolId)->DetectedAsFunctional(ProtocolId); } RPC_STATUS InitTransportProtocols(void); #if defined(DBG) || defined(_DEBUG) #define ASSERT_ALL_TRANSPORT_PROTOCOL_STATE() TransportProtocol::AssertTransportProtocolState() #define ASSERT_TRANSPORT_PROTOCOL_STATE(p) AssertState(p) #else #define ASSERT_ALL_TRANSPORT_PROTOCOL_STATE() #define ASSERT_TRANSPORT_PROTOCOL_STATE(p) #endif #endif