/*++ 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 if (SnmpUtilOidCpy(&pVLE->ResolvedVb.name, &pVb->name) == 0) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: could not copy vb name to working structure.\n" )); // free allocated resources FreeVLE(pVLE); return FALSE; } // 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 if (SnmpUtilAsnAnyCpy(&pVLE->ResolvedVb.value, &pVb->value) == 0) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: could not copy vb value to working structure.\n" )); // free allocated resources FreeVLE(pVLE); return FALSE; } SNMPDBG(( SNMP_LOG_VERBOSE, "SNMP: SVC: variable %d value %s.\n", pVLE->nErrorIndex, "" )); } } 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 if (SnmpUtilVarBindCpy(&pNLE->Pdu.Vbl.list[pVLE->nErrorIndex - 1], &pVLE->ResolvedVb) == 0) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: could not copy resolved vb struct into pdu vbl.\n" )); return FALSE; } // 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; if (nMaxRepetitions > (UINT_MAX/nRepeaters)) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: arithmetic overflow: nMaxRepetitions 0x%x, nRepeaters 0x%x.\n", nMaxRepetitions, nRepeaters )); return FALSE; // bail... } if ((nMaxRepetitions * nRepeaters) > (UINT_MAX - nNonRepeaters)) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: arithmetic overflow: nMaxRepetitions 0x%x, nRepeaters 0x%x, nNonRepeaters 0x%x.\n", nMaxRepetitions, nRepeaters, nNonRepeaters )); return FALSE; // bail... } // calculate maximum number of varbinds possible nMaxVarBinds = nNonRepeaters + (nMaxRepetitions * nRepeaters); if (sizeof(SnmpVarBind) > (UINT_MAX/nMaxVarBinds)) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: arithmetic overflow: sizeof(SnmpVarBind) 0x%x, nMaxVarBinds 0x%x.\n", sizeof(SnmpVarBind), nMaxVarBinds )); return FALSE; // bail... } // BTW, we might want to restrict // (nMaxVarBinds * sizeof(SnmpVarBind)) < 65535 // because this vbl has to be less than the size of udp datagram. // 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 if (SnmpUtilVarBindCpy(&pNLE->Pdu.Vbl.list[nVarBinds], &pVLE->ResolvedVb) == 0) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: could not copy resolved vb into pdu struct at line %d.\n", __LINE__ )); return FALSE; } // increment nVarBinds++; } 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 if (SnmpUtilVarBindCpy(&pNLE->Pdu.Vbl.list[nVarBinds], &pVLE->ResolvedVb) == 0) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: could not copy resolved vb into pdu vbl at line %d.\n", __LINE__ )); // save varbind count processed so far pNLE->Pdu.Vbl.len = nVarBinds; return FALSE; } // 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 if (SnmpUtilVarBindCpy(&pNLE->Pdu.Vbl.list[nVarBinds], &pVLE->ResolvedVbl.list[nIterations]) == 0) { SNMPDBG(( SNMP_LOG_ERROR, "SNMP: SVC: could not copy resolved vb into pdu vbl at line %d.\n", __LINE__ )); // save varbind count processed so far pNLE->Pdu.Vbl.len = nVarBinds; return FALSE; } // 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; }