You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
659 lines
15 KiB
659 lines
15 KiB
/*++
|
|
|
|
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;
|
|
}
|