/*++ Copyright (C) Microsoft Corporation, 1996 - 1999 Module Name: Protocol.cxx Abstract: Transport Protocol abstraction used mainly by PnP. The following is a brief description of the current RPC PnP mechanism: Whenever a non-local transport level object (a socket, or an address) is opened for a given protocol, it is added to the list of objects for the given protocol. This alllows all objects for a given protocol to be tracked down and closed if the protocol is unloaded. By the same token, objects closed are removed from the list of the given protocol. When a PnP notification arrives, the lower level transport code call into this PnP module to tell it that there may be a change of state. For each PnP protocol, the PnP code examines whether the protocol is active, and what was its last state. Depending of the outcome of this comparison, it will take appropriate action. Here's the finite state automaton with the transitions: Currently, RPC differentiates between four networking states of each protocol - protocol is not loaded, protocol is partially loaded (i.e. it is active, but does not have an address on it yet), it is fully loaded (or functional), and it is fully loaded and network address change monitoring is required. The fully loaded state with address change monitoring may be abbreviated to FunctionalMon for the purpose of this document. Note that Partially Loaded, and loaded without address are equivalent terms for the purpose of this module. Also, note that currently the networking code does not allow reloading of a protocol, so some of the code paths will never get exercised. RPC in turn maintains the following RPC (as opposed to networking) states: ProtocolNotLoaded ProtocolLoadedWithoutAddress ProtocolWasLoadedOrNeedsActivation ProtocolLoaded ProtocolWasLoadedOrNeedsActivationWithoutAddress ProtocolLoadedAndMonitored Depending on the last RPC state, and the new networking state, the following changes and state transitions are effected: If a PnP notification fails asynchronously, the completion port code will call into PnP code to try to resubmit the queries. Note that the threads on which overlapped WSAIoctl's are submitted are not protected. If a thread dies, the IO will fail, and the thread that picks the failure will resubmit the WSAIoctl. This strategy allows up to keep the thread pool smaller. Networking State Action RPC State ProtocolNotLoaded NotLoaded No-op PartiallyLoaded Submit a WSAIoctl to be notified when the protocol becomes functional. State = ProtocolLoadedWithoutAddress Functional State = ProtocolLoaded FunctionalMon Submit address change WSAIoctl. State = ProtocolLoadedAndMonitored ProtocolLoadedWithoutAddress NotLoaded Cancel WSAIoctl State = ProtocolNotLoaded PartiallyLoaded No-op Functional CancelWSAIoctl State = ProtocolLoaded FunctionalMon Submit address change WSAIoctl if necessary State = ProtocolLoadedAndMonitored ProtocolWasLoadedOrNeedsActivation NotLoaded No-op PartiallyLoaded Submit a WSAIoctl to be notified when the protocol becomes functional. State = ProtocolWasLoadedOrNeedsActivationWithoutAddress Functional Restart protocol State = ProtocolLoaded FunctionalMon Submit address change WSAIoctl. State = ProtocolLoadedAndMonitored ProtocolLoaded NotLoaded Unload protocol State = ProtocolWasLoadedOrNeedsActivation PartiallyLoaded Invalid Functional No-op FunctionalMon Invalid transition ProtocolWasLoadedOrNeedsActivationWithoutAddress NotLoaded Cancel WSAIoctl State = ProtocolWasLoadedOrNeedsActivation PartiallyLoaded No-op Functional Restart protocol State = ProtocolLoaded FunctionalMon Submit address change WSAIoctl. State = ProtocolLoadedAndMonitored ProtocolLoadedAndMonitored NotLoaded Cancel address change WSAIoctl Unload protocol State = ProtocolWasLoadedOrNeedsActivation PartiallyLoaded Resubmit address change WSAIoclt if necessary Functional Invalid transition FunctionalMon No-op ProtocolNeedToLoadWhenReady NotLoaded No-op PartiallyLoaded Submit a WSAIoctl to be notified when the protocol becomes functional. State = ProtocolNeedToLoadWhenReadyWithoutAddress Functional Restart protocol State = ProtocolLoaded FunctionalMon Submit address change WSAIoctl. State = ProtocolLoadedAndMonitored ProtocolNeedToLoadWhenReadyWithoutAddress NotLoaded Cancel WSAIoctl State = ProtocolNeedToLoadWhenReady PartiallyLoaded No-op Functional Restart protocol State = ProtocolLoaded FunctionalMon Submit address change WSAIoctl. State = ProtocolLoadedAndMonitored Author: Kamen Moutafov [KamenM] Revision History: KamenM 12/22/1998 Creation KamenM 03/05/1999 Adding state ProtocolLoadedAndMonitored and support for it. KamenM 07/17/2000 Adding support for ProtocolWasLoadedOrNeedsActivation/ ProtocolWasLoadedOrNeedsActivationWithoutAddress --*/ #include #include void RPC_ENTRY NullAddressChangeFn( PVOID arg ) { } RPC_ADDRESS_CHANGE_FN * AddressChangeFn = NullAddressChangeFn; #ifdef MAJOR_PNP_DEBUG const char *ProtocolStateNames[]; #endif void TransportProtocol::DetectedAsFunctional(PROTOCOL_ID ProtocolId) { if (IsAddressChangeMonitoringOn(ProtocolId)) { EnterCriticalSection(&AddressListLock); // monitor functional protocols for address change will // set the state MonitorFunctionalProtocolForAddressChange(ProtocolId); LeaveCriticalSection(&AddressListLock); } else { // we also need to take the critical section, and cancel any address change // notification (if any), to avoid race between a successful listen making // the protocol functional, and the address change notification completing EnterCriticalSection(&AddressListLock); CancelAddressChangeRequestIfNecessary(FALSE, ProtocolId); SetState(ProtocolLoaded, ProtocolId); LeaveCriticalSection(&AddressListLock); } } void TransportProtocol::HandleProtocolChange(IN WSAPROTOCOL_INFO *lpProtocolBuffer, IN int ProtocolCount, IN PROTOCOL_ID thisProtocolId) /*++ Function Name: HandleProtocolChange Parameters: lpProtocolBuffer - an array of WSAPROTOCOL_INFO structures as returned by EnumProtocols ProtocolCount - the number of elements in the lpProtocolBuffer array thisProtocolId - the ID of the protocol for this object Description: This handles protocol state change for a particular protocol. The function is idempotent - it can be called many times safely, regardless of previous calls. It will turn into no-op if it is redundant. Returns: --*/ { int i; BOOL fProtocolActive = FALSE; const WS_TRANS_INFO *pInfo; ASSERT(ProtocolCount >= 0); ASSERT(lpProtocolBuffer != NULL); ASSERT_TRANSPORT_PROTOCOL_STATE(thisProtocolId); if ( #ifdef NETBIOS_ON (thisProtocolId == NBF) || (thisProtocolId == NBT) || (thisProtocolId == NBI) || #endif #ifdef NCADG_MQ_ON (thisProtocolId == MSMQ) || #endif (thisProtocolId == CDP) ) return; if (IsTrailingProtocol(thisProtocolId)) return; for (i = 0; i < ProtocolCount; i ++) { // if the enumerated protocol is the current protocol, break out of the loop if (MapProtocolId(lpProtocolBuffer[i].iProtocol, lpProtocolBuffer[i].iAddressFamily) == thisProtocolId) { fProtocolActive = TRUE; break; } } pInfo = &WsTransportTable[thisProtocolId]; switch(State) { case ProtocolNotLoaded: case ProtocolLoadedWithoutAddress: // if the protocol was not loaded, but now it is active, attempt to verify // it is operational if (fProtocolActive) { #ifdef MAJOR_PNP_DEBUG if (State == ProtocolNotLoaded) { DbgPrint("Protocol %d was just loaded\n", thisProtocolId); } #endif // If the protocol is not fully functional, we will submit an address change // request to get notified when it does if (VerifyProtocolIsFunctional(thisProtocolId) == TRUE) { // we succeeded in changing the state of the protocol ASSERT((State == ProtocolLoaded) || (State == ProtocolLoadedAndMonitored)); if (IsAddressChangeMonitoringOn(thisProtocolId)) { if (FirewallTableNeedsUpdating()) { DoFirewallUpdate(); } // If the only reason we were monitoring this protocol // is to finish initializing the firewall table, then // we may not continue monitoring for address change if // the table has finished initializing after DoFirewallUpdate(). if(IsAddressChangeMonitoringOn(thisProtocolId)) { MonitorFunctionalProtocolForAddressChange(thisProtocolId); } if (IsAddressChangeFnDefined()) { (*AddressChangeFn)((PVOID) State); } } } #ifdef MAJOR_PNP_DEBUG if (State == ProtocolLoadedWithoutAddress) { DbgPrint("Protocol %d was without an address\n", thisProtocolId); } #endif } else { if (State == ProtocolLoadedWithoutAddress) { // a protocol was removed without being fully initialized // cancel the pending query if any, and reset the state to not loaded CancelAddressChangeRequestIfNecessary(TRUE, thisProtocolId); SetState(ProtocolNotLoaded, thisProtocolId); #ifdef MAJOR_PNP_DEBUG DbgPrint("Protocol %d was removed without being fully initialized\n", thisProtocolId); #endif } // else - don't care. The protocol state is not loaded, and will remain so } break; case ProtocolWasLoadedOrNeedsActivation: case ProtocolWasLoadedOrNeedsActivationWithoutAddress: if (fProtocolActive) { // If the protocol is not fully functional, we will submit an address change // request to get notified when it does if (VerifyProtocolIsFunctional(thisProtocolId)) { // if a protocol was loaded, and now is active, restart the addresses on it RestartProtocol(thisProtocolId); if (thisProtocolId == TCP) { GetTransportProtocol(HTTP)->RestartProtocol(HTTP); } // we succeeded in changing the state of the protocol ASSERT(State == ProtocolLoaded); if (IsAddressChangeMonitoringOn(thisProtocolId)) { MonitorFunctionalProtocolForAddressChange(thisProtocolId); } } } else { // if the protocol was loaded, but it's not active, we don't care; // if it was trying to get an address, but then it was unloaded, // cancel the request if (State == ProtocolWasLoadedOrNeedsActivationWithoutAddress) { CancelAddressChangeRequestIfNecessary(TRUE, thisProtocolId); SetState(ProtocolWasLoadedOrNeedsActivation, thisProtocolId); #ifdef MAJOR_PNP_DEBUG DbgPrint("Protocol %d was removed without being fully initialized\n", thisProtocolId); #endif } } break; case ProtocolLoaded: ASSERT(IsAddressChangeMonitoringOn(thisProtocolId) == FALSE); // if the protocol was loaded, and it is active, we don't need to do anything; // if it was loaded, but is not active currently, we need to unload the protocol if (!fProtocolActive) { #ifdef MAJOR_PNP_DEBUG DbgPrint("Protocol %d was just unloaded\n", thisProtocolId); #endif UnloadProtocol(thisProtocolId); if (thisProtocolId == TCP) { GetTransportProtocol(HTTP)->UnloadProtocol(HTTP); } SetState(ProtocolWasLoadedOrNeedsActivation, thisProtocolId); } break; case ProtocolLoadedAndMonitored: // if it was loaded, but is not active currently, we need to unload the protocol if (!fProtocolActive) { #ifdef MAJOR_PNP_DEBUG DbgPrint("Protocol %d was just unloaded\n", thisProtocolId); #endif if (FirewallTableNeedsUpdating()) { DoFirewallUpdate(); } CancelAddressChangeRequestIfNecessary(TRUE, thisProtocolId); UnloadProtocol(thisProtocolId); if (thisProtocolId == TCP) { GetTransportProtocol(HTTP)->UnloadProtocol(HTTP); } SetState(ProtocolWasLoadedOrNeedsActivation, thisProtocolId); if (IsAddressChangeFnDefined()) { (*AddressChangeFn)((PVOID) State); } } else { #ifdef MAJOR_PNP_DEBUG DbgPrint("Protocol %d is monitored and received event\n", thisProtocolId); #endif if (FirewallTableNeedsUpdating()) { DoFirewallUpdate(); } if (IsAddressChangeFnDefined()) { (*AddressChangeFn)((PVOID) State); } } break; #if defined(DBG) || defined(_DEBUG) default: ASSERT(!"Invalid State"); #endif } ASSERT_TRANSPORT_PROTOCOL_STATE(thisProtocolId); } void TransportProtocol::AddObjectToList(IN OUT BASE_ASYNC_OBJECT *pObj) /*++ Function Name: AddObjectToList Parameters: pObj - the object to be added Description: Add the object to the list of transport objects for this protocol. Returns: --*/ { EnterCriticalSection(&AddressListLock); RpcpfInsertHeadList(&ObjectList, &pObj->ObjectList); LeaveCriticalSection(&AddressListLock); } void TransportProtocol::RemoveObjectFromList(IN OUT BASE_ASYNC_OBJECT *pObj) /*++ Function Name: RemoveObjectFromList Parameters: pObj - the object to be removed Description: Removes the object from the list of transport objects for this protocol. Returns: --*/ { BASE_ASYNC_OBJECT *Prev, *Cur; Prev = NULL; EnterCriticalSection(&AddressListLock); RpcpfRemoveEntryList(&pObj->ObjectList); LeaveCriticalSection(&AddressListLock); } BOOL TransportProtocol::ResubmitQueriesIfNecessary(PROTOCOL_ID ProtocolId) /*++ Function Name: ResubmitQueriesIfNecessary Parameters: ProtocolId - the ID of the current protocol Description: If there was a WSAIoctl pending that failed, it will be resubmitted. If this protocol was being monitored, try to restart monitoring if necessary Returns: FALSE if the WSAIoctl was not resubmitted successfully. TRUE if the WSAIoctl was resubmitted successfully, or there was no need to resubmit it. --*/ { if (addressChangeSocket) { if ((addressChangeOverlapped.Internal != 0) && (addressChangeOverlapped.Internal != STATUS_PENDING)) { if (SubmitAddressChangeQuery() == FALSE) return FALSE; } } if (State == ProtocolLoadedAndMonitored) { if (MonitorFunctionalProtocolForAddressChange(ProtocolId) == FALSE) return FALSE; } return TRUE; } PROTOCOL_ID MapProtocolId ( IN UINT ProtocolId, IN UINT AddressFamily ) /*++ Function Name: MapProtocolId Parameters: ProtocolId - Winsock protocol ID AddressFamily - Winsock address family Description: Converts a Winsock Protocol ID to a RPC Transport protocol ID. Returns: The RPC Transport Protocol ID if successfull, or -1 if no mapping can be found --*/ { unsigned id; for (id = 1; id < cWsTransportTable; id++) { if ((WsTransportTable[id].Protocol == (int) ProtocolId) && (WsTransportTable[id].AddressFamily == (int) AddressFamily)) { return id; } } return -1; } BOOL TransportProtocol::HandlePnPStateChange(void) /*++ Function Name: HandlePnPNotification Parameters: Description: Whenever a PnP notification (NewAddress) arrives, the completion port will direct it to this routine, which will handle all state management and all handling of the PnP notification. Returns: TRUE if the runtime should be notified that a protocol state change has occurred. FALSE if the run time should not be notified, or need not be notified that a protocol state change has occurred. --*/ { int i; // // Enumerate the currently loaded protocols // WSAPROTOCOL_INFO *lpProtocolBuffer; DWORD dwBufferLength = 512; int ProtocolCount; PROTOCOL_ID ProtocolId; TransportProtocol *pCurrentProtocol; BOOL fRetVal; EnterCriticalSection(&AddressListLock); ASSERT(hWinsock2); while (1) { lpProtocolBuffer = (WSAPROTOCOL_INFO *) I_RpcAllocate(dwBufferLength); if (lpProtocolBuffer == 0) { fRetVal = FALSE; goto CleanupAndReturn; } ProtocolCount = WSAEnumProtocolsT( 0, lpProtocolBuffer, &dwBufferLength ); if (ProtocolCount != SOCKET_ERROR) { break; } I_RpcFree(lpProtocolBuffer); if (GetLastError() != WSAENOBUFS) { fRetVal = FALSE; goto CleanupAndReturn; } } for (i = 1; i < MAX_PROTOCOLS; i++) { pCurrentProtocol = GetTransportProtocol(i); pCurrentProtocol->HandleProtocolChange(lpProtocolBuffer, ProtocolCount, i); } I_RpcFree(lpProtocolBuffer); fRetVal = g_NotifyRt; CleanupAndReturn: LeaveCriticalSection(&AddressListLock); #ifdef MAJOR_PNP_DEBUG DumpProtocolState(); #endif return fRetVal; } BOOL TransportProtocol::ResubmitQueriesIfNecessary(void) /*++ Function Name: ResubmitQueriesIfNecessary Parameters: Description: Iterates through all protocols and calls their ResubmitQueriesIfNecessary Returns: FALSE if at least one protocol needed to resubmit a query, but failed to do so TRUE otherwise --*/ { int i; BOOL fAllResubmitsSucceeded = TRUE; TransportProtocol *pCurrentProtocol; #ifdef MAJOR_PNP_DEBUG DbgPrint("Resubmitting queries for process %d\n", GetCurrentProcessId()); #endif EnterCriticalSection(&AddressListLock); for (i = 1; i < MAX_PROTOCOLS; i++) { pCurrentProtocol = GetTransportProtocol(i); if (!pCurrentProtocol->ResubmitQueriesIfNecessary(i)) fAllResubmitsSucceeded = FALSE; } LeaveCriticalSection(&AddressListLock); return fAllResubmitsSucceeded; } void TransportProtocol::AddObjectToProtocolList(BASE_ASYNC_OBJECT *pObj) { GetTransportProtocol(pObj->id)->AddObjectToList(pObj); } #if defined(DBG) || defined(_DEBUG) void TransportProtocol::AssertProtocolListIntegrity(IN BASE_ASYNC_OBJECT *pObj) { GetTransportProtocol(pObj->id)->AssertListIntegrity(pObj->id); } #endif // DBG || _DEBUG void TransportProtocol::RemoveObjectFromProtocolList(IN OUT BASE_ASYNC_OBJECT *pObj) { // in some cases, we can legally have the id set to INVALID_PROTOCOL_ID // this happens when we have initialized the connection, but have failed // before calling Open on it, and then we attempt to destroy it if (pObj->id != INVALID_PROTOCOL_ID) { GetTransportProtocol(pObj->id)->RemoveObjectFromList(pObj); } } BOOL TransportProtocol::VerifyProtocolIsFunctional(IN PROTOCOL_ID ProtocolId) /*++ Function Name: VerifyProtocolIsFunctional Parameters: ProtocolId - the protocol which we're attempting to verify as functional or not Description: Tries to find out the listening address for a loaded protocol. The address itself is not used anywhere. It just testifies that the protocol is fully operational. Depending on how far it gets with testing whether the protocol is operational, it will change the State to ProtocolLoaded, ProtocolLoadedWithoutAddress or ProtocolWasLoadedOrNeedsActivationWithoutAddress. It will return TRUE iff the state is moved to ProtocolLoaded. Returns: TRUE if the address was found and the protocol was fully operational FALSE if the address could not be found, and the protocol is not fully operational. Note that there are two subcases here. One is when the protocol is not operational (or we cannot confirm that for lack of resources for example) and it has no sign of becoming operational. The second is when the protocol is not operational, but there are chances of it becoming operational. This happens with protocols that take some time to initialize. In the second case, this function will arrange for a retry attempt by posting an async WSAIoctl request to the completion port. In the current code base, we don't differentiate between the two cases because we don't need to. We may need to do so in the future. --*/ { ASSERT((State == ProtocolNotLoaded) || (State == ProtocolWasLoadedOrNeedsActivation) || (State == ProtocolLoadedWithoutAddress) || (State == ProtocolWasLoadedOrNeedsActivationWithoutAddress)); ASSERT_TRANSPORT_PROTOCOL_STATE(ProtocolId); if (OpenAddressChangeRequestSocket(ProtocolId) == FALSE) goto AbortAndCleanup; // if the socket already has an address, skip further checks // NOTE: this check provides a fast way to check initialization state of // protocols that initialize quickly, but more importantly, it provides a // handling path for protocols that have submitted an address list change query // and now are getting back the result if (DoesAddressSocketHaveAddress()) { #ifdef MAJOR_PNP_DEBUG DbgPrint("Protocol %d is functional (1)\n", ProtocolId); #endif CancelAddressChangeRequestIfNecessary(FALSE, ProtocolId); SetStateToLoadedAndMonitorProtocolIfNecessary(ProtocolId); ASSERT_TRANSPORT_PROTOCOL_STATE(ProtocolId); return TRUE; } // if there isn't a pending request, and there isn't a successful request (i.e. there is either // a failed request, or no request has been submitted so far) if ((addressChangeOverlapped.Internal != 0) && (addressChangeOverlapped.Internal != STATUS_PENDING)) { #ifdef MAJOR_PNP_DEBUG DbgPrint("Submitting WSAIoclt for protocol %d\n", ProtocolId); #endif if (!SubmitAddressChangeQuery()) goto AbortAndCleanup; } // check once more whether we have address - this takes care of the race where the // address did arrive between the time we checked for it in the beginning of this // function and the time we submitted the address change query if (DoesAddressSocketHaveAddress()) { #ifdef MAJOR_PNP_DEBUG DbgPrint("Protocol %d is functional (2)\n", ProtocolId); #endif CancelAddressChangeRequestIfNecessary(FALSE, ProtocolId); SetStateToLoadedAndMonitorProtocolIfNecessary(ProtocolId); ASSERT_TRANSPORT_PROTOCOL_STATE(ProtocolId); return TRUE; } // regardless of whether we succeeded immediately or we are pending, advance the // state to ProtocolLoadedWithoutAddress and return not loaded. The completion at the // completion port will take care of the rest if (State == ProtocolWasLoadedOrNeedsActivation) SetState(ProtocolWasLoadedOrNeedsActivationWithoutAddress, ProtocolId); else if (State == ProtocolNotLoaded) SetState(ProtocolLoadedWithoutAddress, ProtocolId); #ifdef MAJOR_PNP_DEBUG else { DbgPrint("VerifyProtocolIsFunctional did not change state for protocol %d, state: %s\n", ProtocolId, ProtocolStateNames[State]); } #endif ASSERT_TRANSPORT_PROTOCOL_STATE(ProtocolId); return FALSE; AbortAndCleanup: // TRUE or FALSE for this argument doesn't matter here - // the operation has failed CancelAddressChangeRequestIfNecessary(FALSE, ProtocolId); ASSERT_TRANSPORT_PROTOCOL_STATE(ProtocolId); return FALSE; } BOOL TransportProtocol::DoesAddressSocketHaveAddress(void) /*++ Function Name: DoesAddressSocketHaveAddress Parameters: Description: Checks if a valid address can be obtained for this protocol. Returns: TRUE - a valid address could be obtained for this protocol FALSE - otherwise --*/ { DWORD byteRet = 0; char buf[40]; ASSERT(addressChangeSocket != 0); if (WSAIoctl(addressChangeSocket, SIO_ADDRESS_LIST_QUERY, 0, 0, buf, sizeof(buf), &byteRet, NULL, NULL) == SOCKET_ERROR) { return FALSE; } // if the result has non-zero length ... if (byteRet != 0) { #if 0 int i; DbgPrint("WSAIoctl returned:\n"); for (i = 0; i < byteRet; i ++) { DbgPrint(" %X", (unsigned long) buf[i]); } DbgPrint("\nWSAIoctl with ADDRESS_LIST_QUERY returned addresses success\n"); #endif // ... and the resulting value is non zero ... if (*(long *)buf != 0) { // ... we have managed to get the true address return TRUE; } } return FALSE; } void TransportProtocol::CancelAddressChangeRequestIfNecessary(BOOL fForceCancel, IN PROTOCOL_ID ProtocolId) /*++ Function Name: CancelAddressChangeRequestIfNecessary Parameters: Description: If there's an active WSAIoctl on the protocol, cancel it by closing the socket. Returns: --*/ { if (addressChangeSocket != 0) { // if the address change monitoring is off, or cancel is // forced, do the actual cancelling. That is, we don't // cancel if this is a monitored protocol and cancel is // optional if (!IsAddressChangeMonitoringOn(ProtocolId) || fForceCancel) { #ifdef MAJOR_PNP_DEBUG DbgPrint("Address change request cancelled\n"); #endif closesocket(addressChangeSocket); addressChangeSocket = 0; addressChangeOverlapped.Internal = -1; } } } void TransportProtocol::RestartProtocol(PROTOCOL_ID ProtocolId) /*++ Function Name: RestartProtocol Parameters: ProtocolId - the protocol to be restarted. Description: Restarts all addresses on this protocol. Returns: --*/ { BASE_ASYNC_OBJECT *Obj; BOOL fAddressFound = 0; RPC_STATUS Status; LIST_ENTRY *CurrentEntry; VALIDATE(ProtocolId) { TCP, #ifdef SPX_ON SPX, #endif #ifdef APPLETALK_ON DSP, #endif HTTP, UDP, #ifdef IPX_ON IPX, #endif TCP_IPv6 } END_VALIDATE; CurrentEntry = ObjectList.Flink; while(CurrentEntry != &ObjectList) { Obj = CONTAINING_RECORD(CurrentEntry, BASE_ASYNC_OBJECT, ObjectList); ASSERT(Obj->id == (int) ProtocolId); if (Obj->type & ADDRESS) { if (Obj->type & DATAGRAM) { Status = DG_ReactivateAddress((WS_DATAGRAM_ENDPOINT *) Obj); } else { Status = WS_ReactivateAddress((WS_ADDRESS *) Obj); } if (Status == RPC_S_OK) { fAddressFound = 1; COMMON_AddressManager((BASE_ADDRESS *) Obj); } } CurrentEntry = CurrentEntry->Flink; } if (fAddressFound) { COMMON_PostNonIoEvent(TRANSPORT, 0, NULL); } } void TransportProtocol::InitNewAddresses(PROTOCOL_ID ProtocolId) /*++ Function Name: RestartProtocol Parameters: ProtocolId - the protocol for which the addresses are to be initialized. Description: Initializes transport address objects that listen on interfaces that were assigned addresses. Works for TCP only. Returns: --*/ { BASE_ASYNC_OBJECT *Obj; WS_ADDRESS *WsAddr; BASE_ADDRESS *pSavedFirstAddress, *pSavedNextAddress; BOOL fAddressFound = FALSE; RPC_STATUS Status; LIST_ENTRY *CurrentEntry; ASSERT(ProtocolId == TCP); // For each new address in the firewall table, find tranport address // objects that have not been initialized and make them listen on // the net address. Make sure that a given transport address list // listens on all the available network addresses. // Go through the firewall table for (DWORD j = 0; j < pFirewallTable->NumAddresses; j++) { // Pick a newly-initialized and enabled address from the table. if (pFirewallTable->Entries[j].fNewAddress == FALSE || pFirewallTable->Entries[j].fEnabled == FALSE) { continue; } // Go through the list of transport addresses. CurrentEntry = ObjectList.Flink; while(CurrentEntry != &ObjectList) { Obj = CONTAINING_RECORD(CurrentEntry, BASE_ASYNC_OBJECT, ObjectList); ASSERT(Obj->id == (int) ProtocolId); WsAddr = (WS_ADDRESS *)Obj; #ifdef MAJOR_TRANS_DEBUG DbgPrint("RPC: TransportProtocol::InitNewAddresses: Processing address 0x%x.\n", WsAddr); #endif // Initialize the object if it is a TCP address that has not yet been initialized // and if the address list that it is a part of is not listening on all of the available // network addresses yet. if (Obj->type & ADDRESS && Obj->id == TCP && WsAddr->fAddressInitialized == FALSE && FirewallTableNumActiveEntries > ((WS_ADDRESS *)WsAddr->pFirstAddress)->NumActiveAddresses) { // Initialize the address from the firewall table entry. // // Note that we do not have to syncronize with the other users of // pFirewallTable since the only possible update would come from a PnP // notification and this function will be executing during a notification // after the table update. UseProtseq* path does not modify an initialized table // and can't race with this path. #ifdef MAJOR_TRANS_DEBUG DbgPrint("RPC: TransportProtocol::InitNewAddresses: Initializing address 0x%x from pFirewalTable entry with index 0x%x.\n", WsAddr, j); #endif WsAddr->ListenAddr.inetaddr.sin_addr.s_addr = pFirewallTable->Entries[j].Address; // Pass in fResetAddressListEntries = FALSE since the address is already in the list // and the fields have been initialized. Status = WS_ReactivateAddress(WsAddr, FALSE); #ifdef MAJOR_TRANS_DEBUG DbgPrint("RPC: TransportProtocol::InitNewAddresses: WS_ReactivateAddress for address 0x%x returned 0x%x.\n", WsAddr, Status); #endif if (Status == RPC_S_OK) { WsAddr->fAddressInitialized = TRUE; fAddressFound = TRUE; ((WS_ADDRESS *)WsAddr->pFirstAddress)->NumActiveAddresses++; COMMON_AddressManager((BASE_ADDRESS *) Obj); } // Re-build the address vector. Status = IP_BuildAddressVector( &(WsAddr->pFirstAddress->pAddressVector), 0, //OsfAddress->NICFlags, NULL, // Not listening on a specific address. (WS_ADDRESS *)WsAddr->pFirstAddress); // If we were not able to find an entry in the pFirewallTable to initialize this address, // there is nothing to do - wait for another address change event when the // NIC may finish initializing. // // Note that if an interface gets disabled, it may disapper from the pFirewallTable during // an update. In this case, we will remain with un-initialized transport address objects, // but there is nothing that we can do about this since the interface on which they should have // listened is not up. } CurrentEntry = CurrentEntry->Flink; } } if (fAddressFound) { COMMON_PostNonIoEvent(TRANSPORT, 0, NULL); } } void TransportProtocol::UnloadProtocol(PROTOCOL_ID ProtocolId) /*++ Function Name: UnloadProtocol Parameters: ProtocolId - the protocol to be unloaded. Description: Walks the list of the protocol transport objects. Closes the connections, and removes the addresses from the list. Thus failing requests on addresses will not be retried. Returns: --*/ { BASE_ASYNC_OBJECT *Obj; LIST_ENTRY *CurrentEntry; VALIDATE(ProtocolId) { TCP, #ifdef SPX_ON SPX, #endif #ifdef APPLETALK_ON DSP, #endif HTTP, UDP, #ifdef IPX_ON IPX, #endif TCP_IPv6, HTTPv2 } END_VALIDATE; CurrentEntry = ObjectList.Flink; // // - Cleanup all the objects (ie: close the socket). // - Remove all objects other than address objects // from the list. // while(CurrentEntry != &ObjectList) { Obj = CONTAINING_RECORD(CurrentEntry, BASE_ASYNC_OBJECT, ObjectList); ASSERT(Obj->id == (int)ProtocolId); if (Obj->type & ADDRESS) { COMMON_RemoveAddress((BASE_ADDRESS *) Obj); } else { RpcpfRemoveEntryList(&Obj->ObjectList); if (ProtocolId != HTTPv2) WS_Abort(Obj); else HTTP_Abort(Obj); } CurrentEntry = CurrentEntry->Flink; } } BOOL TransportProtocol::SubmitAddressChangeQuery(void) /*++ Function Name: SubmitAddressChangeQuery Parameters: Description: Submits an address change query (WSAIoctl) Returns: TRUE if the WSAIoctl was successfully posted. FALSE otherwise --*/ { ASSERT(addressChangeSocket != 0); static DWORD dwBytesReturned; // // Post an address list change request // addressChangeOverlapped.hEvent = 0; addressChangeOverlapped.Offset = 0; addressChangeOverlapped.OffsetHigh = 0; addressChangeOverlapped.Internal = 0; // submit the address change request - it will always complete on the completion port // we don't care about the dwBytesReturned. The provider requires a valid address, or // it rejects the request if (WSAIoctl(addressChangeSocket, SIO_ADDRESS_LIST_CHANGE, 0, 0, 0, 0, &dwBytesReturned, &addressChangeOverlapped, 0) == SOCKET_ERROR) { if (WSAGetLastError() != ERROR_IO_PENDING) { #ifdef MAJOR_PNP_DEBUG DbgPrint("Submitting WSAIoclt failed with: %d\n", GetLastError()); #endif return FALSE; } } return TRUE; } void TransportProtocol::SetState(TransportProtocolStates newState, PROTOCOL_ID ProtocolId) /*++ Function Name: SetState Parameters: newState - the new state that the protocol needs to move to ProtocolId - the protocol number of this protocol Description: Sets the state of the protocol. Nobody should write the state directly, because protocol state mirroring will stop working. Returns: --*/ { TransportProtocol *CurrentProtocol; #ifdef MAJOR_PNP_DEBUG DbgPrint("Protocol %d moved from state: %s to state: %s\n", ProtocolId, ProtocolStateNames[State], ProtocolStateNames[newState]); #endif // either the address change monitoring is not on for this protocol, or // the protocol doesn't move to loaded state, but not both ASSERT(!IsAddressChangeMonitoringOn(ProtocolId) || (newState != ProtocolLoaded)); State = newState; if (ProtocolId == TCP) { // if a base protocols is moved into loaded and monitored, make sure // the trailing protocols doesn't follow it there. In such a case // the trailing protocol becomes only loaded if (newState == ProtocolLoadedAndMonitored) { GetTransportProtocol(HTTP)->State = ProtocolLoaded; } else { GetTransportProtocol(HTTP)->State = newState; } MirrorProtocolState(TCP_IPv6); } else if (ProtocolId == TCP_IPv6) { MirrorProtocolState(TCP); } } void TransportProtocol::SetStateToLoadedAndMonitorProtocolIfNecessary(PROTOCOL_ID ProtocolId) { if (IsAddressChangeMonitoringOn(ProtocolId)) { // monitor functional protocols for address change will // set the state MonitorFunctionalProtocolForAddressChange(ProtocolId); } else SetState(ProtocolLoaded, ProtocolId); } RPC_STATUS InitTransportProtocols(void) /*++ Function Name: InitTransportProtocols Parameters: Description: Initializes all transport level protocols. This function should be called before any of the TransportProtocol functions. Returns: --*/ { TransportProtocolArray = new TransportProtocol[MAX_PROTOCOLS]; if (TransportProtocolArray == NULL) return (RPC_S_OUT_OF_MEMORY); else return RPC_S_OK; } #if defined(DBG) || defined(_DEBUG) void TransportProtocol::AssertTransportProtocolState(void) /*++ Function Name: AssertTransportProtocolState Parameters: Description: Loops through all protocols and calls AssertState on them Returns: --*/ { int i; for (i = 1; i < MAX_PROTOCOLS; i ++) { GetTransportProtocol(i)->AssertState(i); } } void TransportProtocol::AssertState(PROTOCOL_ID ProtocolId) /*++ Function Name: AssertState Parameters: ProtocolId - the protocol number of the this protocol Description: Currently this includes that addressChangeSocket is set only in the *WithoutAddress states and that all the objects in this protocol list are of the same protocol as this protocol. Returns: --*/ { // make sure that the internal state of the object is consistent ASSERT (State >= ProtocolNotLoaded); ASSERT (State <= ProtocolLoadedAndMonitored); if (IsTrailingProtocol(ProtocolId)) { ASSERT(addressChangeSocket == 0); } else { // if we are in one of these states, there should be no address change request pending if ((State == ProtocolNotLoaded) || (State == ProtocolWasLoadedOrNeedsActivation) || (State == ProtocolLoaded)) { ASSERT(addressChangeSocket == 0); } else { // if we are in one of the else states, there must be address change request pending ASSERT(addressChangeSocket != 0); } } AssertListIntegrity(ProtocolId); } void TransportProtocol::AssertListIntegrity ( IN PROTOCOL_ID ProtocolId ) /*++ Routine Description: Verifies the integrity of the protocol list by walking through every entry and making sure it has the same (and expected) protocol id. Arguments: ProtocolId - the protocol id of the current protocol Return Value: --*/ { BASE_ASYNC_OBJECT *pObject; LIST_ENTRY *CurrentEntry; EnterCriticalSection(&AddressListLock); // walk the object list and make sure every object is of the same protocol CurrentEntry = ObjectList.Flink; while (CurrentEntry != &ObjectList) { pObject = CONTAINING_RECORD(CurrentEntry, BASE_ASYNC_OBJECT, ObjectList); ASSERT(pObject->id == ProtocolId); CurrentEntry = CurrentEntry->Flink; } LeaveCriticalSection(&AddressListLock); } #endif BOOL TransportProtocol::MonitorFunctionalProtocolForAddressChange(PROTOCOL_ID ProtocolId) /*++ Function Name: MonitorFunctionalProtocolForAddressChange Parameters: ProtocolId - the protocol id of the current protocol Description: Makes sure that an already functional protocol is monitored for address change. This is done by: - if an address change socket does not exist, one is opened. - if no address change WSAIoctl is pending on the socket, one is posted. Returns: TRUE if the posting of address change was successful FALSE if it wasn't --*/ { BOOL bRetVal = FALSE; // OpenAddressChangeRequestSocket will take care to check whether there's already // a socket opened if (OpenAddressChangeRequestSocket(ProtocolId) == FALSE) goto Cleanup; if (addressChangeOverlapped.Internal != STATUS_PENDING) { if (SubmitAddressChangeQuery() == FALSE) goto Cleanup; } bRetVal = TRUE; Cleanup: if (State != ProtocolLoadedAndMonitored) SetState(ProtocolLoadedAndMonitored, ProtocolId); return bRetVal; } BOOL TransportProtocol::OpenAddressChangeRequestSocket(PROTOCOL_ID ProtocolId) /*++ Function Name: OpenAddressChangeRequestSocket Parameters: ProtocolId - the protocol id of the current protocol Description: Makes sure that an address change request socket is opened. Returns: TRUE if the opening was successful FALSE if it wasn't --*/ { const WS_TRANS_INFO *pInfo = &WsTransportTable[ProtocolId]; HANDLE h; // we may end up with non-zero addressChangeSocket if this is an address change query // request coming back to us. In this case we don't need to open up another socket if (addressChangeSocket == 0) { ASSERT((State == ProtocolNotLoaded) || (State == ProtocolWasLoadedOrNeedsActivation) || (State == ProtocolLoaded)); // // Create a socket // addressChangeSocket = WSASocketT(pInfo->AddressFamily, pInfo->SocketType, pInfo->Protocol, 0, 0, WSA_FLAG_OVERLAPPED); if (addressChangeSocket == INVALID_SOCKET) { // // We should be able to at least open a socket on the protocol, // if not we got a bogus notification or we're out of resources addressChangeSocket = 0; return FALSE; } // // make the handle non-inheritable so it goes away when we close it. // if (FALSE == SetHandleInformation( (HANDLE) addressChangeSocket, HANDLE_FLAG_INHERIT, 0)) { closesocket(addressChangeSocket); addressChangeSocket = 0; return FALSE; } // associate the socket with the completion port so that we get the notification there h = CreateIoCompletionPort((HANDLE)addressChangeSocket, RpcCompletionPort, NewAddress, 0); if (h == 0) { closesocket(addressChangeSocket); addressChangeSocket = 0; return FALSE; } else if (RpcCompletionPort != h) { ASSERT(RpcCompletionPort == h); CloseHandle(h); closesocket(addressChangeSocket); addressChangeSocket = 0; return FALSE; } } else { // we may have a protocol in state ProtocolNotLoaded, because we are just trying // to bring it to loaded state ASSERT((State == ProtocolLoadedWithoutAddress) || (State == ProtocolWasLoadedOrNeedsActivationWithoutAddress) || (State == ProtocolLoadedAndMonitored) || (State == ProtocolNotLoaded)); } #ifdef MAJOR_PNP_DEBUG DbgPrint("Socket was successfully opened for protocol %d\n", ProtocolId); #endif return TRUE; } void TransportProtocol::MirrorProtocolState ( IN PROTOCOL_ID MirrorProtocolId ) /*++ Function Name: MirrorProtocolState Parameters: MirrorProtocolId - the protocol which is mirrored Description: If this is one of the protocols in a dual stack configuration, change the other protocol into appropriate state Returns: --*/ { TransportProtocol *MirrorProtocol; MirrorProtocol = GetTransportProtocol(MirrorProtocolId); if ((State == ProtocolLoadedAndMonitored) || (State == ProtocolLoaded)) { if (MirrorProtocol->State == ProtocolNotLoaded) { MirrorProtocol->SetState(ProtocolWasLoadedOrNeedsActivation, MirrorProtocolId); } else if (MirrorProtocol->State == ProtocolLoadedWithoutAddress) { MirrorProtocol->SetState(ProtocolWasLoadedOrNeedsActivationWithoutAddress, MirrorProtocolId); } } } #ifdef MAJOR_PNP_DEBUG void TransportProtocol::DumpProtocolState(void) /*++ Function Name: DumpProtocolState Parameters: Description: Iterates through all protocol and calls their DumpProtocolState Returns: --*/ { int i; DbgPrint("Dumping protocol state for process %d\n", GetCurrentProcessId()); for (i = 1; i < MAX_PROTOCOLS; i ++) { GetTransportProtocol(i)->DumpProtocolState(i); } } const char *ProtocolStateNames[] = {"ProtocolNotLoaded", "ProtocolLoadedWithoutAddress" , "ProtocolWasLoadedOrNeedsActivation", "ProtocolLoaded", "ProtocolWasLoadedOrNeedsActivationWithoutAddress", "ProtocolLoadedAndMonitored"}; void TransportProtocol::DumpProtocolState(PROTOCOL_ID ProtocolId) /*++ Function Name: DumpProtocolState Parameters: ProtocolId - the protocol number for this protocol. Description: Dumps all significant information for this protcool on the debugger Returns: --*/ { BASE_ASYNC_OBJECT *pCurrentObject; LIST_ENTRY *pCurrentEntry = ObjectList.Flink; int fFirstTime = TRUE; const RPC_CHAR *Protseq; if (TransportTable[ProtocolId].pInfo) Protseq = TransportTable[ProtocolId].pInfo->ProtocolSequence; else Protseq = L"(null)"; DbgPrint("Protocol: %S\n", Protseq); DbgPrint("State: %s, Addr Change Socket: %X, Addr Change Overlapped: %X\n", ProtocolStateNames[State], addressChangeSocket, (ULONG_PTR)&addressChangeOverlapped); while (pCurrentEntry != &ObjectList) { if (fFirstTime) { DbgPrint("Object List:\n"); fFirstTime = FALSE; } pCurrentObject = CONTAINING_RECORD(pCurrentEntry, BASE_ASYNC_OBJECT, ObjectList); DbgPrint("\t%X\n", (ULONG_PTR)pCurrentObject); pCurrentEntry = pCurrentEntry->Flink; } } #endif TransportProtocol *TransportProtocolArray;