/*++ Copyright (c) 1992-1997 Microsoft Corporation Module Name: network.c Abstract: Contains routines for manipulating transport structures. Environment: User Mode - Win32 Revision History: 10-Feb-1997 DonRyan Rewrote to implement SNMPv2 support. --*/ /////////////////////////////////////////////////////////////////////////////// // // // Include files // // // /////////////////////////////////////////////////////////////////////////////// #include "globals.h" #include "network.h" #include "varbinds.h" #include "query.h" /////////////////////////////////////////////////////////////////////////////// // // // Public procedures // // // /////////////////////////////////////////////////////////////////////////////// BOOL IsValidSockAddr( struct sockaddr *pAddress ) /*++ Routine Description: Verifies if an IP or IPX address is valid. An IP address is valid if it is AF_INET and it is not 0.0.0.0 An IPX address is valid if is AF_IPX and the node-number is not null: xxxxxx.000000000000 Arguments: pAddress - pointer to a generic network address to be tested Return Values: Returns true if the address is valid. --*/ { if (pAddress == NULL) return FALSE; if (pAddress->sa_family == AF_INET) { return (((struct sockaddr_in *)pAddress)->sin_addr.s_addr != 0); } else if (pAddress->sa_family == AF_IPX) { char zeroBuff[6] = {0, 0, 0, 0, 0, 0}; return memcmp(((struct sockaddr_ipx *)pAddress)->sa_nodenum, zeroBuff, sizeof(zeroBuff)) != 0; } // the address is neither IP nor IPX hence it is definitely an invalid address return FALSE; } BOOL AllocNLE( PNETWORK_LIST_ENTRY * ppNLE ) /*++ Routine Description: Allocates transport structure and initializes. Arguments: ppNLE - pointer to receive pointer to list entry. Return Values: Returns true if successful. --*/ { BOOL fOk = FALSE; PNETWORK_LIST_ENTRY pNLE; // attempt to allocate structure pNLE = AgentMemAlloc(sizeof(NETWORK_LIST_ENTRY)); // validate pointer if (pNLE != NULL) { // allocate buffer to be used for io pNLE->Buffer.buf = AgentMemAlloc(NLEBUFLEN); // validate pointer if (pNLE->Buffer.buf != NULL) { // initialize socket to socket pNLE->Socket = INVALID_SOCKET; // initialize buffer length pNLE->Buffer.len = NLEBUFLEN; // initialize subagent query list InitializeListHead(&pNLE->Queries); // initialize variable bindings list InitializeListHead(&pNLE->Bindings); // success fOk = TRUE; } else { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: could not allocate network io buffer.\n" )); // release FreeNLE(pNLE); // re-init pNLE = NULL; } } else { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: could not allocate network entry.\n" )); } // transfer *ppNLE = pNLE; return fOk; } BOOL FreeNLE( PNETWORK_LIST_ENTRY pNLE ) /*++ Routine Description: Releases transport structure. Arguments: pNLE - pointer to transport structure. Return Values: Returns true if successful. --*/ { // validate pointer if (pNLE != NULL) { // check to see if socket valid if (pNLE->Socket != INVALID_SOCKET) { // release socket closesocket(pNLE->Socket); } // release pdu UnloadPdu(pNLE); // release query list UnloadQueries(pNLE); // release bindings list UnloadVarBinds(pNLE); // release network buffer AgentMemFree(pNLE->Buffer.buf); // release memory AgentMemFree(pNLE); } return TRUE; } BOOL LoadIncomingTransports( ) /*++ Routine Description: Creates entries for each incoming interface. Arguments: None. Return Values: Returns true if successful. --*/ { BOOL fUdpOk = FALSE; BOOL fIpxOk = FALSE; PNETWORK_LIST_ENTRY pNLE = NULL; INT nStatus; // allocate tcpip if (AllocNLE(&pNLE)) { struct servent * pServEnt; struct sockaddr_in * pSockAddr; // initialize sockaddr structure size pNLE->SockAddrLen = sizeof(struct sockaddr_in); // obtain pointer to sockaddr structure pSockAddr = (struct sockaddr_in *)&pNLE->SockAddr; // attempt to get server information pServEnt = getservbyname("snmp","udp"); // initialize address structure pSockAddr->sin_family = AF_INET; pSockAddr->sin_addr.s_addr = INADDR_ANY; pSockAddr->sin_port = (pServEnt != NULL) ? (SHORT)pServEnt->s_port : htons(DEFAULT_SNMP_PORT_UDP) ; // allocate tpcip socket pNLE->Socket = WSASocket( AF_INET, SOCK_DGRAM, 0, NULL, 0, WSA_FLAG_OVERLAPPED ); // validate socket if (pNLE->Socket != INVALID_SOCKET) { // BUG# 553100 SNMP Agent should not respond to requests to a // subnet broadcast address DWORD dwOption = 0; nStatus = setsockopt(pNLE->Socket, IPPROTO_IP, IP_RECEIVE_BROADCAST, (PCHAR)&dwOption, sizeof(dwOption) ); if (nStatus == SOCKET_ERROR) { SNMPDBG(( SNMP_LOG_WARNING, "SNMP: SVC: ignore error %d, setsockopt with IP_RECEIVE_BROADCAST on socket %p.\n", WSAGetLastError(), pNLE->Socket )); } // attempt to bind nStatus = bind(pNLE->Socket, &pNLE->SockAddr, pNLE->SockAddrLen ); // validate return code if (nStatus != SOCKET_ERROR) { SNMPDBG(( SNMP_LOG_TRACE, "SNMP: SVC: successfully bound to udp port %d.\n", ntohs(pSockAddr->sin_port) )); // insert transport into list of incoming InsertTailList(&g_IncomingTransports, &pNLE->Link); // success fUdpOk = TRUE; } else { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: error %d binding to udp port %d.\n", WSAGetLastError(), ntohs(pSockAddr->sin_port) )); } } else { SNMPDBG(( SNMP_LOG_WARNING, "SNMP: SVC: error %d creating udp socket.\n", WSAGetLastError() )); } if (!fUdpOk) { // release FreeNLE(pNLE); } } // allocate ipx if (AllocNLE(&pNLE)) { struct sockaddr_ipx * pSockAddr; // initialize sockaddr structure size pNLE->SockAddrLen = sizeof(struct sockaddr_ipx); // obtain pointer to sockaddr structure pSockAddr = (struct sockaddr_ipx *)&pNLE->SockAddr; // initialize address structure pSockAddr->sa_family = AF_IPX; pSockAddr->sa_socket = htons(DEFAULT_SNMP_PORT_IPX); // allocate ipx socket pNLE->Socket = WSASocket( AF_IPX, SOCK_DGRAM, NSPROTO_IPX, NULL, 0, WSA_FLAG_OVERLAPPED ); // validate socket if (pNLE->Socket != INVALID_SOCKET) { // attempt to bind nStatus = bind(pNLE->Socket, &pNLE->SockAddr, pNLE->SockAddrLen ); // validate return code if (nStatus != SOCKET_ERROR) { SNMPDBG(( SNMP_LOG_TRACE, "SNMP: SVC: successfully bound to ipx port %d.\n", ntohs(pSockAddr->sa_socket) )); // insert transport into list of incoming InsertTailList(&g_IncomingTransports, &pNLE->Link); // success fIpxOk = TRUE; } else { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: error %d binding to ipx port %d.\n", WSAGetLastError(), ntohs(pSockAddr->sa_socket) )); } } else { SNMPDBG(( SNMP_LOG_WARNING, "SNMP: SVC: error %d creating ipx socket.\n", WSAGetLastError() )); } if (!fIpxOk) { // release FreeNLE(pNLE); } } // need one transport min return (fUdpOk || fIpxOk); } BOOL UnloadTransport( PNETWORK_LIST_ENTRY pNLE ) { // make sure the parameter is valid, otherwise the macro below AVs if (pNLE == NULL) return FALSE; // remove the entry from the list RemoveEntryList(&(pNLE->Link)); // release the memory FreeNLE(pNLE); return TRUE; } BOOL UnloadIncomingTransports( ) /*++ Routine Description: Destroys entries for each outgoing interface. Arguments: None. Return Values: Returns true if successful. --*/ { PLIST_ENTRY pLE; PNETWORK_LIST_ENTRY pNLE; // process entries until empty while (!IsListEmpty(&g_IncomingTransports)) { // extract next entry from head pLE = RemoveHeadList(&g_IncomingTransports); // retrieve pointer to mib region structure pNLE = CONTAINING_RECORD(pLE, NETWORK_LIST_ENTRY, Link); // release FreeNLE(pNLE); } return TRUE; } BOOL LoadOutgoingTransports( ) /*++ Routine Description: Creates entries for each outgoing interface. Arguments: None. Return Values: Returns true if successful. --*/ { BOOL fUdpOk = FALSE; BOOL fIpxOk = FALSE; PNETWORK_LIST_ENTRY pNLE = NULL; // allocate tcpip if (AllocNLE(&pNLE)) { // allocate tpcip socket pNLE->Socket = WSASocket( AF_INET, SOCK_DGRAM, 0, NULL, 0, WSA_FLAG_OVERLAPPED ); // validate socket if (pNLE->Socket != INVALID_SOCKET) { pNLE->SockAddr.sa_family = AF_INET; // insert transport into list of incoming InsertTailList(&g_OutgoingTransports, &pNLE->Link); // success fUdpOk = TRUE; } else { SNMPDBG(( SNMP_LOG_WARNING, "SNMP: SVC: error %d creating udp socket.\n", WSAGetLastError() )); // release FreeNLE(pNLE); } } // allocate ipx if (AllocNLE(&pNLE)) { // allocate ipx socket pNLE->Socket = WSASocket( AF_IPX, SOCK_DGRAM, NSPROTO_IPX, NULL, 0, WSA_FLAG_OVERLAPPED ); // validate socket if (pNLE->Socket != INVALID_SOCKET) { pNLE->SockAddr.sa_family = AF_IPX; // insert transport into list of incoming InsertTailList(&g_OutgoingTransports, &pNLE->Link); // success fIpxOk = TRUE; } else { SNMPDBG(( SNMP_LOG_WARNING, "SNMP: SVC: error %d creating ipx socket.\n", WSAGetLastError() )); // release FreeNLE(pNLE); } } // need one transport min return (fUdpOk || fIpxOk); } BOOL UnloadOutgoingTransports( ) /*++ Routine Description: Destroys entries for each outgoing interface. Arguments: None. Return Values: Returns true if successful. --*/ { PLIST_ENTRY pLE; PNETWORK_LIST_ENTRY pNLE; // process entries until empty while (!IsListEmpty(&g_OutgoingTransports)) { // extract next entry from head pLE = RemoveHeadList(&g_OutgoingTransports); // retrieve pointer to mib region structure pNLE = CONTAINING_RECORD(pLE, NETWORK_LIST_ENTRY, Link); // release FreeNLE(pNLE); } return TRUE; } BOOL UnloadPdu( PNETWORK_LIST_ENTRY pNLE ) /*++ Routine Description: Releases resources allocated in pdu structure. Arguments: pNLE - pointer to network list entry. Return Values: Returns true if successful. --*/ { // release community string SnmpUtilOctetsFree(&pNLE->Community); // release varbinds in pdu SnmpUtilVarBindListFree(&pNLE->Pdu.Vbl); return TRUE; }