Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

951 lines
23 KiB

/*++
Copyright (c) 1992-1997 Microsoft Corporation
Module Name:
varbinds.c
Abstract:
Contains routines for manipulating varbinds.
Environment:
User Mode - Win32
Revision History:
10-Feb-1997 DonRyan
Rewrote to implement SNMPv2 support.
--*/
///////////////////////////////////////////////////////////////////////////////
// //
// Include files //
// //
///////////////////////////////////////////////////////////////////////////////
#include "globals.h"
#include "varbinds.h"
#include "query.h"
#include "snmpmgmt.h"
///////////////////////////////////////////////////////////////////////////////
// //
// Private procedures //
// //
///////////////////////////////////////////////////////////////////////////////
BOOL
LoadVarBind(
PNETWORK_LIST_ENTRY pNLE,
UINT iVb
)
/*++
Routine Description:
Creates varbind list entry from varbind structure.
Arguments:
pNLE - pointer to network list entry.
iVb - index of variable binding.
Return Values:
Returns true if successful.
--*/
{
SnmpVarBind * pVb;
PVARBIND_LIST_ENTRY pVLE = NULL;
PMIB_REGION_LIST_ENTRY pRLE = NULL;
BOOL fAnyOk;
BOOL fOk;
// allocate list entry
if (fOk = AllocVLE(&pVLE)) {
// save varbind list index
pVLE->nErrorIndex = iVb + 1;
// retrieve varbind pointer
pVb = &pNLE->Pdu.Vbl.list[iVb];
SNMPDBG((
SNMP_LOG_TRACE,
"SNMP: SVC: variable %d name %s.\n",
pVLE->nErrorIndex,
SnmpUtilOidToA(&pVb->name)
));
// initialize type of resolved variable
pVLE->ResolvedVb.value.asnType = ASN_NULL;
// copy varbind name to working structure
SnmpUtilOidCpy(&pVLE->ResolvedVb.name, &pVb->name);
// see if specific object asked for
fAnyOk = ((pNLE->Pdu.nType == SNMP_PDU_GETNEXT) ||
(pNLE->Pdu.nType == SNMP_PDU_GETBULK));
// attempt to lookup variable name in supported regions
if (FindSupportedRegion(&pRLE, &pVb->name, fAnyOk)) {
// save pointer to region
pVLE->pCurrentRLE = pRLE;
// structure has been initialized
pVLE->nState = VARBIND_INITIALIZED;
SNMPDBG((
SNMP_LOG_TRACE,
"SNMP: SVC: variable %d assigned to %s.\n",
pVLE->nErrorIndex,
pVLE->pCurrentRLE->pSLE->pPathname
));
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d state '%s'.\n",
pVLE->nErrorIndex,
VARBINDSTATESTRING(pVLE->nState)
));
// see if this is a getnext request
if (pNLE->Pdu.nType == SNMP_PDU_GETNEXT) {
// only need single rep
pVLE->nMaxRepetitions = 1;
} else if (pNLE->Pdu.nType == SNMP_PDU_GETBULK) {
// see if this non-repeater which is in bounds
if (pNLE->Pdu.Pdu.BulkPdu.nNonRepeaters > (int)iVb) {
// only need single rep
pVLE->nMaxRepetitions = 1;
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d max repetitions %d.\n",
pVLE->nErrorIndex,
pVLE->nMaxRepetitions
));
// see if max-repetitions is non-zero
} else if (pNLE->Pdu.Pdu.BulkPdu.nMaxRepetitions > 0) {
// set max-repetitions to value in getbulk pdu
pVLE->nMaxRepetitions = pNLE->Pdu.Pdu.BulkPdu.nMaxRepetitions;
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d max repetitions %d.\n",
pVLE->nErrorIndex,
pVLE->nMaxRepetitions
));
} else {
// modify state to resolved
pVLE->nState = VARBIND_RESOLVED;
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d state '%s'.\n",
pVLE->nErrorIndex,
VARBINDSTATESTRING(pVLE->nState)
));
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d value NULL.\n",
pVLE->nErrorIndex
));
}
} else if (pNLE->Pdu.nType == SNMP_PDU_SET) {
// copy varbind value to working structure
SnmpUtilAsnAnyCpy(&pVLE->ResolvedVb.value, &pVb->value);
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d value %s.\n",
pVLE->nErrorIndex,
"<TBD>"
));
}
} else {
// null pointer to region
pVLE->pCurrentRLE = NULL;
SNMPDBG((
SNMP_LOG_TRACE,
"SNMP: SVC: variable %d unable to be assigned.\n",
pVLE->nErrorIndex
));
// getbulk
if (fAnyOk) {
// modify state to resolved
pVLE->nState = VARBIND_RESOLVED;
// set the exception in the variable's type field
pVLE->ResolvedVb.value.asnType = SNMP_EXCEPTION_ENDOFMIBVIEW;
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d state '%s'.\n",
pVLE->nErrorIndex,
VARBINDSTATESTRING(pVLE->nState)
));
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d value ENDOFMIBVIEW.\n",
pVLE->nErrorIndex
));
} else if (pNLE->Pdu.nType == SNMP_PDU_GET) {
// modify state to resolved
pVLE->nState = VARBIND_RESOLVED;
// set the exception in the variable's type field
pVLE->ResolvedVb.value.asnType = SNMP_EXCEPTION_NOSUCHOBJECT;
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d state '%s'.\n",
pVLE->nErrorIndex,
VARBINDSTATESTRING(pVLE->nState)
));
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d value NOSUCHOBJECT.\n",
pVLE->nErrorIndex
));
} else {
// modify state to resolved
//pVLE->nState = VARBIND_ABORTED;
pVLE->nState = VARBIND_RESOLVED;
// save error status in network list entry
pNLE->Pdu.Pdu.NormPdu.nErrorStatus = SNMP_ERRORSTATUS_NOTWRITABLE;
pNLE->Pdu.Pdu.NormPdu.nErrorIndex = pVLE->nErrorIndex;
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d state '%s'.\n",
pVLE->nErrorIndex,
VARBINDSTATESTRING(pVLE->nState)
));
SNMPDBG((
SNMP_LOG_VERBOSE,
"SNMP: SVC: variable %d error %s.\n",
pVLE->nErrorIndex,
SNMPERRORSTRING(pNLE->Pdu.Pdu.NormPdu.nErrorStatus)
));
// failure
//fOk = FALSE;
}
}
// add to existing varbind list
InsertTailList(&pNLE->Bindings, &pVLE->Link);
}
return fOk;
}
BOOL
LoadVarBinds(
PNETWORK_LIST_ENTRY pNLE
)
/*++
Routine Description:
Creates list of varbind entries from varbind structure.
Arguments:
pNLE - pointer to network list entry.
Return Values:
Returns true if successful.
--*/
{
UINT iVb;
BOOL fOk = TRUE;
// process each varbind in list
for (iVb = 0; (fOk && (iVb < pNLE->Pdu.Vbl.len)); iVb++) {
// load individual varbind
fOk = LoadVarBind(pNLE, iVb);
}
return fOk;
}
BOOL
UnloadVarBinds(
PNETWORK_LIST_ENTRY pNLE
)
/*++
Routine Description:
Destroys list of varbind entries.
Arguments:
pNLE - pointer to network list entry.
Return Values:
Returns true if successful.
--*/
{
PLIST_ENTRY pLE;
PVARBIND_LIST_ENTRY pVLE;
// process each varbind entry
while (!IsListEmpty(&pNLE->Bindings)) {
// point to first varbind
pLE = RemoveHeadList(&pNLE->Bindings);
// retrieve pointer to varbind entry from link
pVLE = CONTAINING_RECORD(pLE, VARBIND_LIST_ENTRY, Link);
// release
FreeVLE(pVLE);
}
return TRUE;
}
BOOL
ValidateVarBinds(
PNETWORK_LIST_ENTRY pNLE
)
/*++
Routine Description:
Updates error status based on query results and version.
Arguments:
pNLE - pointer to network list entry.
Return Values:
Returns true if successful.
--*/
{
PLIST_ENTRY pLE;
PVARBIND_LIST_ENTRY pVLE;
// see if error has already report during processing
if (pNLE->Pdu.Pdu.NormPdu.nErrorStatus == SNMP_ERRORSTATUS_NOERROR) {
// point to first varbind
pLE = pNLE->Bindings.Flink;
// process each varbind entry
while (pLE != &pNLE->Bindings) {
// retrieve pointer to varbind entry from link
pVLE = CONTAINING_RECORD(pLE, VARBIND_LIST_ENTRY, Link);
// see if varbind has been resolved
if (pVLE->nState != VARBIND_RESOLVED) {
SNMPDBG((
SNMP_LOG_WARNING,
"SNMP: SVC: variable %d unresolved.\n",
pVLE->nErrorIndex
));
// report internal error has occurred
pNLE->Pdu.Pdu.NormPdu.nErrorStatus = SNMP_ERRORSTATUS_GENERR;
pNLE->Pdu.Pdu.NormPdu.nErrorIndex = pVLE->nErrorIndex;
break; // bail...
} else if (pNLE->nVersion == SNMP_VERSION_1) {
// report error if exceptions are present instead of values
if ((pVLE->ResolvedVb.value.asnType == SNMP_EXCEPTION_NOSUCHOBJECT) ||
(pVLE->ResolvedVb.value.asnType == SNMP_EXCEPTION_NOSUCHINSTANCE) ||
(pVLE->ResolvedVb.value.asnType == SNMP_EXCEPTION_ENDOFMIBVIEW)) {
SNMPDBG((
SNMP_LOG_WARNING,
"SNMP: SVC: variable %d unresolved in SNMPv1.\n",
pVLE->nErrorIndex
));
// report that variable could not be found
pNLE->Pdu.Pdu.NormPdu.nErrorStatus = SNMP_ERRORSTATUS_NOSUCHNAME;
pNLE->Pdu.Pdu.NormPdu.nErrorIndex = pVLE->nErrorIndex;
break; // bail...
}
}
// next entry
pLE = pLE->Flink;
}
}
// see if this is first version
if (pNLE->nVersion == SNMP_VERSION_1) {
// adjust status code
switch (pNLE->Pdu.Pdu.NormPdu.nErrorStatus) {
case SNMP_ERRORSTATUS_NOERROR:
case SNMP_ERRORSTATUS_TOOBIG:
case SNMP_ERRORSTATUS_NOSUCHNAME:
case SNMP_ERRORSTATUS_BADVALUE:
case SNMP_ERRORSTATUS_READONLY:
case SNMP_ERRORSTATUS_GENERR:
break;
case SNMP_ERRORSTATUS_NOACCESS:
case SNMP_ERRORSTATUS_NOCREATION:
case SNMP_ERRORSTATUS_NOTWRITABLE:
case SNMP_ERRORSTATUS_AUTHORIZATIONERROR:
case SNMP_ERRORSTATUS_INCONSISTENTNAME:
pNLE->Pdu.Pdu.NormPdu.nErrorStatus = SNMP_ERRORSTATUS_NOSUCHNAME;
break;
case SNMP_ERRORSTATUS_WRONGTYPE:
case SNMP_ERRORSTATUS_WRONGLENGTH:
case SNMP_ERRORSTATUS_WRONGENCODING:
case SNMP_ERRORSTATUS_WRONGVALUE:
case SNMP_ERRORSTATUS_INCONSISTENTVALUE:
pNLE->Pdu.Pdu.NormPdu.nErrorStatus = SNMP_ERRORSTATUS_BADVALUE;
break;
case SNMP_ERRORSTATUS_RESOURCEUNAVAILABLE:
case SNMP_ERRORSTATUS_COMMITFAILED:
case SNMP_ERRORSTATUS_UNDOFAILED:
default:
pNLE->Pdu.Pdu.NormPdu.nErrorStatus = SNMP_ERRORSTATUS_GENERR;
break;
}
}
return (pNLE->Pdu.Pdu.NormPdu.nErrorStatus == SNMP_ERRORSTATUS_NOERROR);
}
BOOL
UpdateVarBindsFromResolvedVb(
PNETWORK_LIST_ENTRY pNLE
)
/*++
Routine Description:
Updates varbinds with results containing single varbinds.
Arguments:
pNLE - pointer to network list entry.
Return Values:
Returns true if successful.
--*/
{
PLIST_ENTRY pLE;
PVARBIND_LIST_ENTRY pVLE;
// point to first varbind
pLE = pNLE->Bindings.Flink;
// process each varbind entry
while (pLE != &pNLE->Bindings) {
// retrieve pointer to varbind entry from link
pVLE = CONTAINING_RECORD(pLE, VARBIND_LIST_ENTRY, Link);
SNMPDBG((
SNMP_LOG_TRACE,
"SNMP: SVC: variable %d resolved name %s.\n",
pVLE->nErrorIndex,
SnmpUtilOidToA(&pVLE->ResolvedVb.name)
));
// release memory for original varbind
SnmpUtilVarBindFree(&pNLE->Pdu.Vbl.list[pVLE->nErrorIndex - 1]);
// copy resolved varbind structure into pdu varbindlist
SnmpUtilVarBindCpy(&pNLE->Pdu.Vbl.list[pVLE->nErrorIndex - 1],
&pVLE->ResolvedVb
);
// next entry
pLE = pLE->Flink;
}
// success
return TRUE;
}
BOOL
UpdateVarBindsFromResolvedVbl(
PNETWORK_LIST_ENTRY pNLE
)
/*++
Routine Description:
Updates varbinds with results containing multiple varbinds.
Arguments:
pNLE - pointer to network list entry.
Return Values:
Returns true if successful.
--*/
{
UINT nRepeaters;
UINT nNonRepeaters;
UINT nMaxRepetitions;
UINT nIterations;
UINT nVarBindsLast;
UINT nVarBinds = 0;
SnmpVarBind * pVarBind;
PVARBIND_LIST_ENTRY pVLE;
PLIST_ENTRY pLE1;
PLIST_ENTRY pLE2;
// retrieve getbulk parameters from pdu
nNonRepeaters = pNLE->Pdu.Pdu.BulkPdu.nNonRepeaters;
nMaxRepetitions = pNLE->Pdu.Pdu.BulkPdu.nMaxRepetitions;
nRepeaters = (pNLE->Pdu.Vbl.len >= nNonRepeaters)
? (pNLE->Pdu.Vbl.len - nNonRepeaters)
: 0
;
// see if we need to expand size of varbind list
if ((nRepeaters > 0) && (nMaxRepetitions > 1)) {
UINT nMaxVarBinds;
SnmpVarBind * pVarBinds;
// calculate maximum number of varbinds possible
nMaxVarBinds = nNonRepeaters + (nMaxRepetitions * nRepeaters);
// reallocate varbind list to fit maximum
pVarBinds = SnmpUtilMemReAlloc(pNLE->Pdu.Vbl.list,
nMaxVarBinds * sizeof(SnmpVarBind)
);
// validate pointer
if (pVarBinds == NULL) {
SNMPDBG((
SNMP_LOG_ERROR,
"SNMP: SVC: Could not re-allocate varbind list.\n"
));
return FALSE; // bail...
}
// restore varbind pointer
pNLE->Pdu.Vbl.list = pVarBinds;
}
// point to first varbind
pLE1 = pNLE->Bindings.Flink;
// process each varbind entry
while (pLE1 != &pNLE->Bindings) {
// retrieve pointer to varbind entry from link
pVLE = CONTAINING_RECORD(pLE1, VARBIND_LIST_ENTRY, Link);
// see if this is non-repeater
if (pVLE->nMaxRepetitions == 1) {
// release memory for original varbind
SnmpUtilVarBindFree(&pNLE->Pdu.Vbl.list[nVarBinds]);
// copy resolved varbind into pdu structure
SnmpUtilVarBindCpy(&pNLE->Pdu.Vbl.list[nVarBinds++],
&pVLE->ResolvedVb
);
} else {
//
// finished processing non-repeaters
//
break;
}
// next entry
pLE1 = pLE1->Flink;
}
// initialize
nIterations = 0;
// store
pLE2 = pLE1;
// process any repeaters until max
while (nIterations < nMaxRepetitions) {
// restore
pLE1 = pLE2;
// process each varbind entry
while (pLE1 != &pNLE->Bindings) {
// retrieve pointer to varbind entry from link
pVLE = CONTAINING_RECORD(pLE1, VARBIND_LIST_ENTRY, Link);
// see if value stored in default
if (pVLE->ResolvedVbl.len == 0) {
// release memory for original varbind
SnmpUtilVarBindFree(&pNLE->Pdu.Vbl.list[nVarBinds]);
// copy resolved varbind into pdu varbind list
SnmpUtilVarBindCpy(&pNLE->Pdu.Vbl.list[nVarBinds],
&pVLE->ResolvedVb
);
// increment
nVarBinds++;
// see if value available in this iteration
} else if (pVLE->ResolvedVbl.len > nIterations) {
// release memory for original varbind
SnmpUtilVarBindFree(&pNLE->Pdu.Vbl.list[nVarBinds]);
// copy resolved varbind into pdu varbind list
SnmpUtilVarBindCpy(&pNLE->Pdu.Vbl.list[nVarBinds],
&pVLE->ResolvedVbl.list[nIterations]
);
// increment
nVarBinds++;
}
// next entry
pLE1 = pLE1->Flink;
}
// increment
nIterations++;
}
// save new varbind count
pNLE->Pdu.Vbl.len = nVarBinds;
// success
return TRUE;
}
BOOL
UpdatePdu(
PNETWORK_LIST_ENTRY pNLE,
BOOL fOk
)
/*++
Routine Description:
Updates pdu with query results.
Arguments:
pNLE - pointer to network list entry.
fOk - true if process succeeded to this point.
Return Values:
Returns true if successful.
--*/
{
PLIST_ENTRY pLE;
PVARBIND_LIST_ENTRY pVLE;
// validate
if (fOk) {
// make sure varbinds valid
fOk = ValidateVarBinds(pNLE);
// validate
if (fOk) {
// see if pdu type is getnext or getbulk
if (pNLE->Pdu.nType != SNMP_PDU_GETBULK) {
// update varbinds with single result
fOk = UpdateVarBindsFromResolvedVb(pNLE);
} else {
// update varbinds with multiple results
fOk = UpdateVarBindsFromResolvedVbl(pNLE);
}
}
}
// trap internal errors that have not been accounted for as of yet
if (!fOk && (pNLE->Pdu.Pdu.NormPdu.nErrorStatus == SNMP_ERRORSTATUS_NOERROR)) {
// report status that was determined above
pNLE->Pdu.Pdu.NormPdu.nErrorStatus = SNMP_ERRORSTATUS_GENERR;
pNLE->Pdu.Pdu.NormPdu.nErrorIndex = 0;
}
if (pNLE->Pdu.Pdu.NormPdu.nErrorStatus == SNMP_ERRORSTATUS_NOERROR)
{
switch(pNLE->Pdu.nType)
{
case SNMP_PDU_GETNEXT:
case SNMP_PDU_GETBULK:
case SNMP_PDU_GET:
// update counter for successful GET-NEXT GET-BULK
mgmtCAdd(CsnmpInTotalReqVars, pNLE->Pdu.Vbl.len);
break;
case SNMP_PDU_SET:
// update counter for successful SET
mgmtCAdd(CsnmpInTotalSetVars, pNLE->Pdu.Vbl.len);
break;
}
}
else
{
// update here counters for all OUT errors
mgmtUtilUpdateErrStatus(OUT_errStatus, pNLE->Pdu.Pdu.NormPdu.nErrorStatus);
}
return TRUE;
}
///////////////////////////////////////////////////////////////////////////////
// //
// Public procedures //
// //
///////////////////////////////////////////////////////////////////////////////
BOOL
AllocVLE(
PVARBIND_LIST_ENTRY * ppVLE
)
/*++
Routine Description:
Allocates varbind structure and initializes.
Arguments:
ppVLE - pointer to receive pointer to entry.
Return Values:
Returns true if successful.
--*/
{
BOOL fOk = FALSE;
PVARBIND_LIST_ENTRY pVLE = NULL;
// attempt to allocate structure
pVLE = AgentMemAlloc(sizeof(VARBIND_LIST_ENTRY));
// validate
if (pVLE != NULL) {
// success
fOk = TRUE;
} else {
SNMPDBG((
SNMP_LOG_ERROR,
"SNMP: SVC: could not allocate varbind entry.\n"
));
}
// transfer
*ppVLE = pVLE;
return fOk;
}
BOOL
FreeVLE(
PVARBIND_LIST_ENTRY pVLE
)
/*++
Routine Description:
Releases varbind structure.
Arguments:
pVLE - pointer to list entry to be freed.
Return Values:
Returns true if successful.
--*/
{
BOOL fOk = TRUE;
// validate pointer
if (pVLE != NULL) {
// release current varbind
SnmpUtilVarBindFree(&pVLE->ResolvedVb);
// release current varbind list
SnmpUtilVarBindListFree(&pVLE->ResolvedVbl);
// release structure
AgentMemFree(pVLE);
}
return TRUE;
}
BOOL
ProcessVarBinds(
PNETWORK_LIST_ENTRY pNLE
)
/*++
Routine Description:
Creates list of varbind entries from varbind structure.
Arguments:
pNLE - pointer to network list entry.
Return Values:
Returns true if successful.
--*/
{
BOOL fOk = FALSE;
// validate type before processing
if ((pNLE->Pdu.nType == SNMP_PDU_SET) ||
(pNLE->Pdu.nType == SNMP_PDU_GET) ||
(pNLE->Pdu.nType == SNMP_PDU_GETNEXT) ||
(pNLE->Pdu.nType == SNMP_PDU_GETBULK)) {
// initialize varbinds
if (LoadVarBinds(pNLE)) {
// process queries
fOk = ProcessQueries(pNLE);
}
// transfer results
UpdatePdu(pNLE, fOk);
// unload varbinds
UnloadVarBinds(pNLE);
// update management counters for accepted and processed requests
switch(pNLE->Pdu.nType)
{
case SNMP_PDU_GET:
mgmtCTick(CsnmpInGetRequests);
break;
case SNMP_PDU_GETNEXT:
case SNMP_PDU_GETBULK:
mgmtCTick(CsnmpInGetNexts);
break;
case SNMP_PDU_SET:
mgmtCTick(CsnmpInSetRequests);
break;
}
} else {
SNMPDBG((
SNMP_LOG_ERROR,
"SNMP: SVC: ignoring unknown pdu type %d.\n",
pNLE->Pdu.nType
));
}
return fOk;
}