Blame agent/agent_handler.c

Packit fcad23
/* Portions of this file are subject to the following copyright(s).  See
Packit fcad23
 * the Net-SNMP's COPYING file for more details and other copyrights
Packit fcad23
 * that may apply:
Packit fcad23
 */
Packit fcad23
/*
Packit fcad23
 * Portions of this file are copyrighted by:
Packit fcad23
 * Copyright © 2003 Sun Microsystems, Inc. All rights reserved.
Packit fcad23
 * Use is subject to license terms specified in the COPYING file
Packit fcad23
 * distributed with the Net-SNMP package.
Packit fcad23
 *
Packit fcad23
 * Portions of this file are copyrighted by:
Packit fcad23
 * Copyright (c) 2016 VMware, Inc. All rights reserved.
Packit fcad23
 * Use is subject to license terms specified in the COPYING file
Packit fcad23
 * distributed with the Net-SNMP package.
Packit fcad23
 */
Packit fcad23
#include <net-snmp/net-snmp-config.h>
Packit fcad23
#include <net-snmp/net-snmp-features.h>
Packit fcad23
Packit fcad23
#include <sys/types.h>
Packit fcad23
Packit fcad23
#if HAVE_STRING_H
Packit fcad23
#include <string.h>
Packit fcad23
#endif
Packit fcad23
Packit fcad23
#include <net-snmp/net-snmp-includes.h>
Packit fcad23
#include <net-snmp/agent/net-snmp-agent-includes.h>
Packit fcad23
Packit fcad23
#include <net-snmp/agent/bulk_to_next.h>
Packit fcad23
Packit fcad23
netsnmp_feature_child_of(agent_handler, libnetsnmpagent)
Packit fcad23
Packit fcad23
netsnmp_feature_child_of(handler_mark_requests_as_delegated, agent_handler)
Packit fcad23
Packit fcad23
static netsnmp_mib_handler *_clone_handler(netsnmp_mib_handler *it);
Packit fcad23
Packit fcad23
/***********************************************************************/
Packit fcad23
/*
Packit fcad23
 * New Handler based API 
Packit fcad23
 */
Packit fcad23
/***********************************************************************/
Packit fcad23
/** @defgroup handler Net-SNMP Agent handler and extensibility API
Packit fcad23
 *  @ingroup agent
Packit fcad23
 *
Packit fcad23
 *  The basic theory goes something like this: In the past, with the
Packit fcad23
 *  original mib module api (which derived from the original CMU SNMP
Packit fcad23
 *  code) the underlying mib modules were passed very little
Packit fcad23
 *  information (only the truly most basic information about a
Packit fcad23
 *  request).  This worked well at the time but in todays world of
Packit fcad23
 *  subagents, device instrumentation, low resource consumption, etc,
Packit fcad23
 *  it just isn't flexible enough.  "handlers" are here to fix all that.
Packit fcad23
 *
Packit fcad23
 *  With the rewrite of the agent internals for the net-snmp 5.0
Packit fcad23
 *  release, we introduce a modular calling scheme that allows agent
Packit fcad23
 *  modules to be written in a very flexible manner, and more
Packit fcad23
 *  importantly allows reuse of code in a decent way (and without the
Packit fcad23
 *  memory and speed overheads of OO languages like C++).
Packit fcad23
 *
Packit fcad23
 *  Functionally, the notion of what a handler does is the same as the
Packit fcad23
 *  older api: A handler is @link netsnmp_create_handler() created@endlink and
Packit fcad23
 *  then @link netsnmp_register_handler() registered@endlink with the main
Packit fcad23
 *  agent at a given OID in the OID tree and gets called any time a
Packit fcad23
 *  request is made that it should respond to.  You probably should
Packit fcad23
 *  use one of the convenience helpers instead of doing anything else
Packit fcad23
 *  yourself though:
Packit fcad23
 *
Packit fcad23
 *  Most importantly, though, is that the handlers are built on the
Packit fcad23
 *  notion of modularity and reuse.  Specifically, rather than do all
Packit fcad23
 *  the really hard work (like parsing table indexes out of an
Packit fcad23
 *  incoming oid request) in each module, the API is designed to make
Packit fcad23
 *  it easy to write "helper" handlers that merely process some aspect
Packit fcad23
 *  of the request before passing it along to the final handler that
Packit fcad23
 *  returns the real answer.  Most people will want to make use of the
Packit fcad23
 *  @link instance instance@endlink, @link table table@endlink, @link
Packit fcad23
 *  table_iterator table_iterator@endlink, @link table_data
Packit fcad23
 *  table_data@endlink, or @link table_dataset table_dataset@endlink
Packit fcad23
 *  helpers to make their life easier.  These "helpers" interpert
Packit fcad23
 *  important aspects of the request and pass them on to you.
Packit fcad23
 *
Packit fcad23
 *  For instance, the @link table table@endlink helper is designed to
Packit fcad23
 *  hand you a list of extracted index values from an incoming
Packit fcad23
 *  request.  THe @link table_iterator table_iterator@endlink helper
Packit fcad23
 *  is built on top of the table helper, and is designed to help you
Packit fcad23
 *  iterate through data stored elsewhere (like in a kernel) that is
Packit fcad23
 *  not in OID lexographical order (ie, don't write your own index/oid
Packit fcad23
 *  sorting routine, use this helper instead).  The beauty of the
Packit fcad23
 *  @link table_iterator table_iterator helper@endlink, as well as the @link
Packit fcad23
 *  instance instance@endlink helper is that they take care of the complex
Packit fcad23
 *  GETNEXT processing entirely for you and hand you everything you
Packit fcad23
 *  need to merely return the data as if it was a GET request.  Much
Packit fcad23
 *  less code and hair pulling.  I've pulled all my hair out to help
Packit fcad23
 *  you so that only one of us has to be bald.
Packit fcad23
 *
Packit fcad23
 * @{
Packit fcad23
 */
Packit fcad23
Packit fcad23
/** Creates a MIB handler structure.
Packit fcad23
 *  The new structure is allocated and filled using the given name
Packit fcad23
 *  and access method.
Packit fcad23
 *  The returned handler should then be @link netsnmp_register_handler()
Packit fcad23
 *  registered @endlink.
Packit fcad23
 *
Packit fcad23
 *  @param name is the handler name and is copied then assigned to
Packit fcad23
 *              netsnmp_mib_handler->handler_name
Packit fcad23
 *
Packit fcad23
 *  @param handler_access_method is a function pointer used as the access
Packit fcad23
 *	   method for this handler registration instance for whatever required
Packit fcad23
 *         needs.
Packit fcad23
 *
Packit fcad23
 *  @return a pointer to a populated netsnmp_mib_handler struct to be
Packit fcad23
 *          registered
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_create_handler_registration()
Packit fcad23
 *  @see netsnmp_register_handler()
Packit fcad23
 */
Packit fcad23
netsnmp_mib_handler *
Packit fcad23
netsnmp_create_handler(const char *name,
Packit fcad23
                       Netsnmp_Node_Handler * handler_access_method)
Packit fcad23
{
Packit fcad23
    netsnmp_mib_handler *ret = SNMP_MALLOC_TYPEDEF(netsnmp_mib_handler);
Packit fcad23
    if (ret) {
Packit fcad23
        ret->access_method = handler_access_method;
Packit fcad23
        if (NULL != name) {
Packit fcad23
            ret->handler_name = strdup(name);
Packit fcad23
            if (NULL == ret->handler_name)
Packit fcad23
                SNMP_FREE(ret);
Packit fcad23
        }
Packit fcad23
    }
Packit fcad23
    return ret;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Creates a MIB handler structure.
Packit fcad23
 *  The new structure is allocated and filled using the given name,
Packit fcad23
 *  access function, registration location OID and list of modes that
Packit fcad23
 *  the handler supports. If modes == 0, then modes will automatically
Packit fcad23
 *  be set to the default value of only HANDLER_CAN_DEFAULT, which is
Packit fcad23
 *  by default read-only GET and GETNEXT requests. A hander which supports
Packit fcad23
 *  sets but not row creation should set us a mode of HANDLER_CAN_SET_ONLY.
Packit fcad23
 *  @note This ends up calling netsnmp_create_handler(name, handler_access_method)
Packit fcad23
 *  @param name is the handler name and is copied then assigned to
Packit fcad23
 *              netsnmp_handler_registration->handlerName.
Packit fcad23
 *
Packit fcad23
 *  @param handler is a function pointer used as the access
Packit fcad23
 *	method for this handler registration instance for whatever required
Packit fcad23
 *	needs.
Packit fcad23
 *
Packit fcad23
 *  @param reg_oid is the registration location oid.
Packit fcad23
 *
Packit fcad23
 *  @param reg_oid_len is the length of reg_oid; can use the macro,
Packit fcad23
 *         OID_LENGTH
Packit fcad23
 *
Packit fcad23
 *  @param modes is used to configure read/write access.  If modes == 0, 
Packit fcad23
 *	then modes will automatically be set to the default 
Packit fcad23
 *	value of only HANDLER_CAN_DEFAULT, which is by default read-only GET 
Packit fcad23
 *	and GETNEXT requests.  The other two mode options are read only, 
Packit fcad23
 *	HANDLER_CAN_RONLY, and read/write, HANDLER_CAN_RWRITE.
Packit fcad23
 *
Packit fcad23
 *		- HANDLER_CAN_GETANDGETNEXT
Packit fcad23
 *		- HANDLER_CAN_SET
Packit fcad23
 *		- HANDLER_CAN_GETBULK      
Packit fcad23
 *
Packit fcad23
 *		- HANDLER_CAN_RONLY   (HANDLER_CAN_GETANDGETNEXT)
Packit fcad23
 *		- HANDLER_CAN_RWRITE  (HANDLER_CAN_GETANDGETNEXT | 
Packit fcad23
 *			HANDLER_CAN_SET)
Packit fcad23
 *		- HANDLER_CAN_DEFAULT HANDLER_CAN_RONLY
Packit fcad23
 *
Packit fcad23
 *  @return Returns a pointer to a netsnmp_handler_registration struct.
Packit fcad23
 *          NULL is returned only when memory could not be allocated for the 
Packit fcad23
 *          netsnmp_handler_registration struct.
Packit fcad23
 *
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_create_handler()
Packit fcad23
 *  @see netsnmp_register_handler()
Packit fcad23
 */
Packit fcad23
netsnmp_handler_registration *
Packit fcad23
netsnmp_handler_registration_create(const char *name,
Packit fcad23
                                    netsnmp_mib_handler *handler,
Packit fcad23
                                    const oid * reg_oid, size_t reg_oid_len,
Packit fcad23
                                    int modes)
Packit fcad23
{
Packit fcad23
    netsnmp_handler_registration *the_reg;
Packit fcad23
    the_reg = SNMP_MALLOC_TYPEDEF(netsnmp_handler_registration);
Packit fcad23
    if (!the_reg)
Packit fcad23
        return NULL;
Packit fcad23
Packit fcad23
    if (modes)
Packit fcad23
        the_reg->modes = modes;
Packit fcad23
    else
Packit fcad23
        the_reg->modes = HANDLER_CAN_DEFAULT;
Packit fcad23
Packit fcad23
    the_reg->handler = handler;
Packit fcad23
    the_reg->priority = DEFAULT_MIB_PRIORITY;
Packit fcad23
    if (name)
Packit fcad23
        the_reg->handlerName = strdup(name);
Packit fcad23
    the_reg->rootoid = snmp_duplicate_objid(reg_oid, reg_oid_len);
Packit fcad23
    the_reg->rootoid_len = reg_oid_len;
Packit fcad23
    return the_reg;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Creates a handler registration structure with a new MIB handler.
Packit fcad23
 *  This function first @link netsnmp_create_handler() creates @endlink
Packit fcad23
 *  a MIB handler, then @link netsnmp_handler_registration_create()
Packit fcad23
 *  makes registation structure @endlink for it.
Packit fcad23
 *
Packit fcad23
 *  @param name is the handler name for netsnmp_create_handler()
Packit fcad23
 *
Packit fcad23
 *  @param handler_access_method is a function pointer used as the access
Packit fcad23
 *     method for netsnmp_create_handler()
Packit fcad23
 *
Packit fcad23
 *  @param reg_oid is the registration location oid.
Packit fcad23
 *
Packit fcad23
 *  @param reg_oid_len is the length of reg_oid; can use the macro,
Packit fcad23
 *         OID_LENGTH
Packit fcad23
 *
Packit fcad23
 *  @param modes is used to configure read/write access, as in
Packit fcad23
 *         netsnmp_handler_registration_create()
Packit fcad23
 *
Packit fcad23
 *  @return Returns a pointer to a netsnmp_handler_registration struct.
Packit fcad23
 *          If the structures creation failed, NULL is returned.
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_create_handler()
Packit fcad23
 *  @see netsnmp_handler_registration_create()
Packit fcad23
 */
Packit fcad23
netsnmp_handler_registration *
Packit fcad23
netsnmp_create_handler_registration(const char *name,
Packit fcad23
                                    Netsnmp_Node_Handler *
Packit fcad23
                                    handler_access_method, const oid * reg_oid,
Packit fcad23
                                    size_t reg_oid_len, int modes)
Packit fcad23
{
Packit fcad23
    netsnmp_handler_registration *rv = NULL;
Packit fcad23
    netsnmp_mib_handler *handler =
Packit fcad23
        netsnmp_create_handler(name, handler_access_method);
Packit fcad23
    if (handler) {
Packit fcad23
        rv = netsnmp_handler_registration_create(
Packit fcad23
            name, handler, reg_oid, reg_oid_len, modes);
Packit fcad23
        if (!rv)
Packit fcad23
            netsnmp_handler_free(handler);
Packit fcad23
    }
Packit fcad23
    return rv;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Registers a MIB handler inside the registration structure.
Packit fcad23
 *  Checks given registation handler for sanity, then
Packit fcad23
 *  @link netsnmp_register_mib() performs registration @endlink
Packit fcad23
 *  in the MIB tree, as defined by the netsnmp_handler_registration
Packit fcad23
 *  pointer. On success, SNMP_CALLBACK_APPLICATION is called.
Packit fcad23
 *  The registration struct may be created by call of
Packit fcad23
 *  netsnmp_create_handler_registration().
Packit fcad23
 *
Packit fcad23
 *  @param reginfo Pointer to a netsnmp_handler_registration struct.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_create_handler_registration()
Packit fcad23
 *  @see netsnmp_register_mib()
Packit fcad23
 */
Packit fcad23
int
Packit fcad23
netsnmp_register_handler(netsnmp_handler_registration *reginfo)
Packit fcad23
{
Packit fcad23
    netsnmp_mib_handler *handler;
Packit fcad23
    int flags = 0;
Packit fcad23
Packit fcad23
    if (reginfo == NULL) {
Packit fcad23
        snmp_log(LOG_ERR, "netsnmp_register_handler() called illegally\n");
Packit fcad23
        netsnmp_assert(reginfo != NULL);
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    DEBUGIF("handler::register") {
Packit fcad23
        DEBUGMSGTL(("handler::register", "Registering %s (", reginfo->handlerName));
Packit fcad23
        for (handler = reginfo->handler; handler; handler = handler->next) {
Packit fcad23
            DEBUGMSG(("handler::register", "::%s", handler->handler_name));
Packit fcad23
        }
Packit fcad23
Packit fcad23
        DEBUGMSG(("handler::register", ") at "));
Packit fcad23
        if (reginfo->rootoid && reginfo->range_subid) {
Packit fcad23
            DEBUGMSGOIDRANGE(("handler::register", reginfo->rootoid,
Packit fcad23
                              reginfo->rootoid_len, reginfo->range_subid,
Packit fcad23
                              reginfo->range_ubound));
Packit fcad23
        } else if (reginfo->rootoid) {
Packit fcad23
            DEBUGMSGOID(("handler::register", reginfo->rootoid,
Packit fcad23
                         reginfo->rootoid_len));
Packit fcad23
        } else {
Packit fcad23
            DEBUGMSG(("handler::register", "[null]"));
Packit fcad23
        }
Packit fcad23
        DEBUGMSG(("handler::register", "\n"));
Packit fcad23
    }
Packit fcad23
Packit fcad23
    /*
Packit fcad23
     * don't let them register for absolutely nothing.  Probably a mistake 
Packit fcad23
     */
Packit fcad23
    if (0 == reginfo->modes) {
Packit fcad23
        reginfo->modes = HANDLER_CAN_DEFAULT;
Packit fcad23
        snmp_log(LOG_WARNING, "no registration modes specified for %s. "
Packit fcad23
                 "Defaulting to 0x%x\n", reginfo->handlerName, reginfo->modes);
Packit fcad23
    }
Packit fcad23
Packit fcad23
    /*
Packit fcad23
     * for handlers that can't GETBULK, force a conversion handler on them 
Packit fcad23
     */
Packit fcad23
    if (!(reginfo->modes & HANDLER_CAN_GETBULK)) {
Packit fcad23
        handler = netsnmp_get_bulk_to_next_handler();
Packit fcad23
        if (!handler ||
Packit fcad23
            (netsnmp_inject_handler(reginfo, handler) != SNMPERR_SUCCESS)) {
Packit fcad23
            snmp_log(LOG_WARNING, "could not inject bulk to next handler\n");
Packit fcad23
            if (handler)
Packit fcad23
                netsnmp_handler_free(handler);
Packit fcad23
            /** should this be a critical error? */
Packit fcad23
            netsnmp_handler_registration_free(reginfo);
Packit fcad23
            return SNMP_ERR_GENERR;
Packit fcad23
        }
Packit fcad23
    }
Packit fcad23
Packit fcad23
    for (handler = reginfo->handler; handler; handler = handler->next) {
Packit fcad23
        if (handler->flags & MIB_HANDLER_INSTANCE)
Packit fcad23
            flags = FULLY_QUALIFIED_INSTANCE;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    return netsnmp_register_mib(reginfo->handlerName,
Packit fcad23
                                NULL, 0, 0,
Packit fcad23
                                reginfo->rootoid, reginfo->rootoid_len,
Packit fcad23
                                reginfo->priority,
Packit fcad23
                                reginfo->range_subid,
Packit fcad23
                                reginfo->range_ubound, NULL,
Packit fcad23
                                reginfo->contextName, reginfo->timeout, flags,
Packit fcad23
                                reginfo, 1);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Unregisters a MIB handler described inside the registration structure.
Packit fcad23
 *  Removes a registration, performed earlier by
Packit fcad23
 *  netsnmp_register_handler(), from the MIB tree.
Packit fcad23
 *  Uses unregister_mib_context() to do the task.
Packit fcad23
 *
Packit fcad23
 *  @param reginfo Pointer to a netsnmp_handler_registration struct.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_register_handler()
Packit fcad23
 *  @see unregister_mib_context()
Packit fcad23
 */
Packit fcad23
int
Packit fcad23
netsnmp_unregister_handler(netsnmp_handler_registration *reginfo)
Packit fcad23
{
Packit fcad23
    if (!reginfo)
Packit fcad23
        return SNMPERR_SUCCESS;
Packit fcad23
    return unregister_mib_context(reginfo->rootoid, reginfo->rootoid_len,
Packit fcad23
                                  reginfo->priority,
Packit fcad23
                                  reginfo->range_subid, reginfo->range_ubound,
Packit fcad23
                                  reginfo->contextName);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Registers a MIB handler inside the registration structure.
Packit fcad23
 *  Checks given registation handler for sanity, then
Packit fcad23
 *  @link netsnmp_register_mib() performs registration @endlink
Packit fcad23
 *  in the MIB tree, as defined by the netsnmp_handler_registration
Packit fcad23
 *  pointer. Never calls SNMP_CALLBACK_APPLICATION.
Packit fcad23
 *  The registration struct may be created by call of
Packit fcad23
 *  netsnmp_create_handler_registration().
Packit fcad23
 *
Packit fcad23
 *  @param reginfo Pointer to a netsnmp_handler_registration struct.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_create_handler_registration()
Packit fcad23
 *  @see netsnmp_register_mib()
Packit fcad23
 */
Packit fcad23
int
Packit fcad23
netsnmp_register_handler_nocallback(netsnmp_handler_registration *reginfo)
Packit fcad23
{
Packit fcad23
    netsnmp_mib_handler *handler;
Packit fcad23
    if (reginfo == NULL) {
Packit fcad23
        snmp_log(LOG_ERR, "netsnmp_register_handler_nocallback() called illegally\n");
Packit fcad23
        netsnmp_assert(reginfo != NULL);
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
    DEBUGIF("handler::register") {
Packit fcad23
        DEBUGMSGTL(("handler::register",
Packit fcad23
                    "Registering (with no callback) "));
Packit fcad23
        for (handler = reginfo->handler; handler; handler = handler->next) {
Packit fcad23
            DEBUGMSG(("handler::register", "::%s", handler->handler_name));
Packit fcad23
        }
Packit fcad23
Packit fcad23
        DEBUGMSG(("handler::register", " at "));
Packit fcad23
        if (reginfo->rootoid && reginfo->range_subid) {
Packit fcad23
            DEBUGMSGOIDRANGE(("handler::register", reginfo->rootoid,
Packit fcad23
                              reginfo->rootoid_len, reginfo->range_subid,
Packit fcad23
                              reginfo->range_ubound));
Packit fcad23
        } else if (reginfo->rootoid) {
Packit fcad23
            DEBUGMSGOID(("handler::register", reginfo->rootoid,
Packit fcad23
                         reginfo->rootoid_len));
Packit fcad23
        } else {
Packit fcad23
            DEBUGMSG(("handler::register", "[null]"));
Packit fcad23
        }
Packit fcad23
        DEBUGMSG(("handler::register", "\n"));
Packit fcad23
    }
Packit fcad23
Packit fcad23
    /*
Packit fcad23
     * don't let them register for absolutely nothing.  Probably a mistake 
Packit fcad23
     */
Packit fcad23
    if (0 == reginfo->modes) {
Packit fcad23
        reginfo->modes = HANDLER_CAN_DEFAULT;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    return netsnmp_register_mib(reginfo->handler->handler_name,
Packit fcad23
                                NULL, 0, 0,
Packit fcad23
                                reginfo->rootoid, reginfo->rootoid_len,
Packit fcad23
                                reginfo->priority,
Packit fcad23
                                reginfo->range_subid,
Packit fcad23
                                reginfo->range_ubound, NULL,
Packit fcad23
                                reginfo->contextName, reginfo->timeout, 0,
Packit fcad23
                                reginfo, 0);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Injects handler into the calling chain of handlers.
Packit fcad23
 *  The given MIB handler is inserted after the handler named before_what.
Packit fcad23
 *  If before_what is NULL, the handler is put at the top of the list,
Packit fcad23
 *  and hence will be the handler to be called first.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_create_handler_registration()
Packit fcad23
 *  @see netsnmp_inject_handler()
Packit fcad23
 */
Packit fcad23
int
Packit fcad23
netsnmp_inject_handler_before(netsnmp_handler_registration *reginfo,
Packit fcad23
                              netsnmp_mib_handler *handler,
Packit fcad23
                              const char *before_what)
Packit fcad23
{
Packit fcad23
    netsnmp_mib_handler *handler2 = handler;
Packit fcad23
Packit fcad23
    if (handler == NULL || reginfo == NULL) {
Packit fcad23
        snmp_log(LOG_ERR, "netsnmp_inject_handler() called illegally\n");
Packit fcad23
        netsnmp_assert(reginfo != NULL);
Packit fcad23
        netsnmp_assert(handler != NULL);
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
    while (handler2->next) {
Packit fcad23
        handler2 = handler2->next;  /* Find the end of a handler sub-chain */
Packit fcad23
    }
Packit fcad23
    if (reginfo->handler == NULL) {
Packit fcad23
        DEBUGMSGTL(("handler:inject", "injecting %s\n", handler->handler_name));
Packit fcad23
    }
Packit fcad23
    else {
Packit fcad23
        DEBUGMSGTL(("handler:inject", "injecting %s before %s\n",
Packit fcad23
                    handler->handler_name, reginfo->handler->handler_name));
Packit fcad23
    }
Packit fcad23
    if (before_what) {
Packit fcad23
        netsnmp_mib_handler *nexth, *prevh = NULL;
Packit fcad23
        if (reginfo->handler == NULL) {
Packit fcad23
            snmp_log(LOG_ERR, "no handler to inject before\n");
Packit fcad23
            return SNMP_ERR_GENERR;
Packit fcad23
        }
Packit fcad23
        for(nexth = reginfo->handler; nexth;
Packit fcad23
            prevh = nexth, nexth = nexth->next) {
Packit fcad23
            if (strcmp(nexth->handler_name, before_what) == 0)
Packit fcad23
                break;
Packit fcad23
        }
Packit fcad23
        if (!nexth) {
Packit fcad23
	    snmp_log(LOG_ERR, "Cannot inject '%s' before '%s': not found\n", handler->handler_name, before_what);
Packit fcad23
	    snmp_log(LOG_ERR, "The handlers are:\n");
Packit fcad23
	    for (nexth = reginfo->handler; nexth; nexth = nexth->next)
Packit fcad23
		snmp_log(LOG_ERR, "  %s\n", nexth->handler_name);
Packit fcad23
            return SNMP_ERR_GENERR;
Packit fcad23
	}
Packit fcad23
        if (prevh) {
Packit fcad23
            /* after prevh and before nexth */
Packit fcad23
            prevh->next = handler;
Packit fcad23
            handler2->next = nexth;
Packit fcad23
            handler->prev = prevh;
Packit fcad23
            nexth->prev = handler2;
Packit fcad23
            return SNMPERR_SUCCESS;
Packit fcad23
        }
Packit fcad23
        /* else we're first, which is what we do next anyway so fall through */
Packit fcad23
    }
Packit fcad23
    handler2->next = reginfo->handler;
Packit fcad23
    if (reginfo->handler)
Packit fcad23
        reginfo->handler->prev = handler2;
Packit fcad23
    reginfo->handler = handler;
Packit fcad23
    return SNMPERR_SUCCESS;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Injects handler into the calling chain of handlers.
Packit fcad23
 *  The given MIB handler is put at the top of the list,
Packit fcad23
 *  and hence will be the handler to be called first.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_create_handler_registration()
Packit fcad23
 *  @see netsnmp_inject_handler_before()
Packit fcad23
 */
Packit fcad23
int
Packit fcad23
netsnmp_inject_handler(netsnmp_handler_registration *reginfo,
Packit fcad23
                       netsnmp_mib_handler *handler)
Packit fcad23
{
Packit fcad23
    return netsnmp_inject_handler_before(reginfo, handler, NULL);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Calls a MIB handlers chain, starting with specific handler.
Packit fcad23
 *  The given arguments and MIB handler are checked
Packit fcad23
 *  for sanity, then the handlers are called, one by one,
Packit fcad23
 *  until next handler is NULL.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE int
Packit fcad23
netsnmp_call_handler(netsnmp_mib_handler *next_handler,
Packit fcad23
                     netsnmp_handler_registration *reginfo,
Packit fcad23
                     netsnmp_agent_request_info *reqinfo,
Packit fcad23
                     netsnmp_request_info *requests)
Packit fcad23
{
Packit fcad23
    Netsnmp_Node_Handler *nh;
Packit fcad23
    int             ret;
Packit fcad23
Packit fcad23
    if (next_handler == NULL || reginfo == NULL || reqinfo == NULL ||
Packit fcad23
        requests == NULL) {
Packit fcad23
        snmp_log(LOG_ERR, "netsnmp_call_handler() called illegally\n");
Packit fcad23
        netsnmp_assert(next_handler != NULL);
Packit fcad23
        netsnmp_assert(reqinfo != NULL);
Packit fcad23
        netsnmp_assert(reginfo != NULL);
Packit fcad23
        netsnmp_assert(requests != NULL);
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    do {
Packit fcad23
        nh = next_handler->access_method;
Packit fcad23
        if (!nh) {
Packit fcad23
            if (next_handler->next) {
Packit fcad23
                snmp_log(LOG_ERR, "no access method specified in handler %s.",
Packit fcad23
                         next_handler->handler_name);
Packit fcad23
                return SNMP_ERR_GENERR;
Packit fcad23
            }
Packit fcad23
            /*
Packit fcad23
             * The final handler registration in the chain may well not need
Packit fcad23
             * to include a handler routine, if the processing of this object
Packit fcad23
             * is handled completely by the agent toolkit helpers.
Packit fcad23
             */
Packit fcad23
            return SNMP_ERR_NOERROR;
Packit fcad23
        }
Packit fcad23
Packit fcad23
        DEBUGMSGTL(("handler:calling", "calling handler %s for mode %s\n",
Packit fcad23
                    next_handler->handler_name,
Packit fcad23
                    se_find_label_in_slist("agent_mode", reqinfo->mode)));
Packit fcad23
Packit fcad23
        /*
Packit fcad23
         * XXX: define acceptable return statuses
Packit fcad23
         */
Packit fcad23
        ret = (*nh) (next_handler, reginfo, reqinfo, requests);
Packit fcad23
Packit fcad23
        DEBUGMSGTL(("handler:returned", "handler %s returned %d\n",
Packit fcad23
                    next_handler->handler_name, ret));
Packit fcad23
Packit fcad23
        if (! (next_handler->flags & MIB_HANDLER_AUTO_NEXT))
Packit fcad23
            break;
Packit fcad23
Packit fcad23
        /*
Packit fcad23
         * did handler signal that it didn't want auto next this time around?
Packit fcad23
         */
Packit fcad23
        if(next_handler->flags & MIB_HANDLER_AUTO_NEXT_OVERRIDE_ONCE) {
Packit fcad23
            next_handler->flags &= ~MIB_HANDLER_AUTO_NEXT_OVERRIDE_ONCE;
Packit fcad23
            break;
Packit fcad23
        }
Packit fcad23
Packit fcad23
        next_handler = next_handler->next;
Packit fcad23
Packit fcad23
    } while(next_handler);
Packit fcad23
Packit fcad23
    return ret;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** @private
Packit fcad23
 *  Calls all the MIB Handlers in registration struct for a given mode.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 */
Packit fcad23
int
Packit fcad23
netsnmp_call_handlers(netsnmp_handler_registration *reginfo,
Packit fcad23
                      netsnmp_agent_request_info *reqinfo,
Packit fcad23
                      netsnmp_request_info *requests)
Packit fcad23
{
Packit fcad23
    netsnmp_request_info *request;
Packit fcad23
    int             status;
Packit fcad23
Packit fcad23
    if (reginfo == NULL || reqinfo == NULL || requests == NULL) {
Packit fcad23
        snmp_log(LOG_ERR, "netsnmp_call_handlers() called illegally\n");
Packit fcad23
        netsnmp_assert(reqinfo != NULL);
Packit fcad23
        netsnmp_assert(reginfo != NULL);
Packit fcad23
        netsnmp_assert(requests != NULL);
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    if (reginfo->handler == NULL) {
Packit fcad23
        snmp_log(LOG_ERR, "no handler specified.");
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    switch (reqinfo->mode) {
Packit fcad23
    case MODE_GETBULK:
Packit fcad23
    case MODE_GET:
Packit fcad23
    case MODE_GETNEXT:
Packit fcad23
        if (!(reginfo->modes & HANDLER_CAN_GETANDGETNEXT))
Packit fcad23
            return SNMP_ERR_NOERROR;    /* legal */
Packit fcad23
        break;
Packit fcad23
Packit fcad23
#ifndef NETSNMP_NO_WRITE_SUPPORT
Packit fcad23
    case MODE_SET_RESERVE1:
Packit fcad23
    case MODE_SET_RESERVE2:
Packit fcad23
    case MODE_SET_ACTION:
Packit fcad23
    case MODE_SET_COMMIT:
Packit fcad23
    case MODE_SET_FREE:
Packit fcad23
    case MODE_SET_UNDO:
Packit fcad23
        if (!(reginfo->modes & HANDLER_CAN_SET)) {
Packit fcad23
            for (; requests; requests = requests->next) {
Packit fcad23
                netsnmp_set_request_error(reqinfo, requests,
Packit fcad23
                                          SNMP_ERR_NOTWRITABLE);
Packit fcad23
            }
Packit fcad23
            return SNMP_ERR_NOERROR;
Packit fcad23
        }
Packit fcad23
        break;
Packit fcad23
#endif /* NETSNMP_NO_WRITE_SUPPORT */
Packit fcad23
Packit fcad23
    default:
Packit fcad23
        snmp_log(LOG_ERR, "unknown mode in netsnmp_call_handlers! bug!\n");
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
    DEBUGMSGTL(("handler:calling", "main handler %s\n",
Packit fcad23
                reginfo->handler->handler_name));
Packit fcad23
Packit fcad23
    for (request = requests ; request; request = request->next) {
Packit fcad23
        request->processed = 0;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    status = netsnmp_call_handler(reginfo->handler, reginfo, reqinfo, requests);
Packit fcad23
Packit fcad23
    return status;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** @private
Packit fcad23
 *  Calls the next MIB handler in the chain, after the current one.
Packit fcad23
 *  The given arguments and MIB handler are checked
Packit fcad23
 *  for sanity, then the next handler is called.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE int
Packit fcad23
netsnmp_call_next_handler(netsnmp_mib_handler *current,
Packit fcad23
                          netsnmp_handler_registration *reginfo,
Packit fcad23
                          netsnmp_agent_request_info *reqinfo,
Packit fcad23
                          netsnmp_request_info *requests)
Packit fcad23
{
Packit fcad23
Packit fcad23
    if (current == NULL || reginfo == NULL || reqinfo == NULL ||
Packit fcad23
        requests == NULL) {
Packit fcad23
        snmp_log(LOG_ERR, "netsnmp_call_next_handler() called illegally\n");
Packit fcad23
        netsnmp_assert(current != NULL);
Packit fcad23
        netsnmp_assert(reginfo != NULL);
Packit fcad23
        netsnmp_assert(reqinfo != NULL);
Packit fcad23
        netsnmp_assert(requests != NULL);
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    return netsnmp_call_handler(current->next, reginfo, reqinfo, requests);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** @private
Packit fcad23
 *  Calls the next MIB handler in the chain, after the current one.
Packit fcad23
 *  The given arguments and MIB handler are not validated before
Packit fcad23
 *  the call, only request is checked.
Packit fcad23
 *
Packit fcad23
 *  @return Returns SNMPERR_SUCCESS or SNMP_ERR_* error code.
Packit fcad23
 */
Packit fcad23
netsnmp_feature_child_of(netsnmp_call_next_handler_one_request,netsnmp_unused)
Packit fcad23
#ifndef NETSNMP_FEATURE_REMOVE_NETSNMP_CALL_NEXT_HANDLER_ONE_REQUEST
Packit fcad23
NETSNMP_INLINE int
Packit fcad23
netsnmp_call_next_handler_one_request(netsnmp_mib_handler *current,
Packit fcad23
                                      netsnmp_handler_registration *reginfo,
Packit fcad23
                                      netsnmp_agent_request_info *reqinfo,
Packit fcad23
                                      netsnmp_request_info *requests)
Packit fcad23
{
Packit fcad23
    netsnmp_request_info *request;
Packit fcad23
    int ret;
Packit fcad23
    
Packit fcad23
    if (!requests) {
Packit fcad23
        snmp_log(LOG_ERR, "netsnmp_call_next_handler_ONE_REQUEST() called illegally\n");
Packit fcad23
        netsnmp_assert(requests != NULL);
Packit fcad23
        return SNMP_ERR_GENERR;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    request = requests->next;
Packit fcad23
    requests->next = NULL;
Packit fcad23
    ret = netsnmp_call_handler(current->next, reginfo, reqinfo, requests);
Packit fcad23
    requests->next = request;
Packit fcad23
    return ret;
Packit fcad23
}
Packit fcad23
#endif /* NETSNMP_FEATURE_REMOVE_NETSNMP_CALL_NEXT_HANDLER_ONE_REQUEST */
Packit fcad23
Packit fcad23
/** Deallocates resources associated with a given handler.
Packit fcad23
 *  The handler is removed from chain and then freed.
Packit fcad23
 *  After calling this function, the handler pointer is invalid
Packit fcad23
 *  and should be set to NULL.
Packit fcad23
 *
Packit fcad23
 *  @param handler is the MIB Handler to be freed
Packit fcad23
 */
Packit fcad23
void
Packit fcad23
netsnmp_handler_free(netsnmp_mib_handler *handler)
Packit fcad23
{
Packit fcad23
    if (handler != NULL) {
Packit fcad23
        if (handler->next != NULL) {
Packit fcad23
            /** make sure we aren't pointing to ourselves.  */
Packit fcad23
            netsnmp_assert(handler != handler->next); /* bugs caught: 1 */
Packit fcad23
            netsnmp_handler_free(handler->next);
Packit fcad23
            handler->next = NULL;
Packit fcad23
        }
Packit fcad23
        if ((handler->myvoid != NULL) && (handler->data_free != NULL))
Packit fcad23
        {
Packit fcad23
            handler->data_free(handler->myvoid);
Packit fcad23
        }
Packit fcad23
        SNMP_FREE(handler->handler_name);
Packit fcad23
        SNMP_FREE(handler);
Packit fcad23
    }
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Duplicates a MIB handler and all subsequent handlers.
Packit fcad23
 *  Creates a copy of all data in given handlers chain.
Packit fcad23
 *
Packit fcad23
 *  @param handler is the MIB Handler to be duplicated
Packit fcad23
 *
Packit fcad23
 *  @return Returns a pointer to the complete copy,
Packit fcad23
 *         or NULL if any problem occured.
Packit fcad23
 *
Packit fcad23
 * @see _clone_handler()
Packit fcad23
 */
Packit fcad23
netsnmp_mib_handler *
Packit fcad23
netsnmp_handler_dup(netsnmp_mib_handler *handler)
Packit fcad23
{
Packit fcad23
    netsnmp_mib_handler *h = NULL;
Packit fcad23
Packit fcad23
    if (!handler)
Packit fcad23
        goto err;
Packit fcad23
Packit fcad23
    h = _clone_handler(handler);
Packit fcad23
    if (!h)
Packit fcad23
        goto err;
Packit fcad23
Packit fcad23
    /*
Packit fcad23
     * Providing a clone function without a free function is asking for
Packit fcad23
     * memory leaks, and providing a free function without clone function
Packit fcad23
     * is asking for memory corruption. Hence the log statement below.
Packit fcad23
     */
Packit fcad23
    if (!!handler->data_clone != !!handler->data_free)
Packit fcad23
        snmp_log(LOG_ERR, "data_clone / data_free inconsistent (%s)\n",
Packit fcad23
                 handler->handler_name);
Packit fcad23
    if (handler->myvoid && handler->data_clone) {
Packit fcad23
        h->myvoid = handler->data_clone(handler->myvoid);
Packit fcad23
        if (!h->myvoid)
Packit fcad23
            goto err;
Packit fcad23
    } else
Packit fcad23
        h->myvoid = handler->myvoid;
Packit fcad23
    h->data_clone = handler->data_clone;
Packit fcad23
    h->data_free = handler->data_free;
Packit fcad23
Packit fcad23
    if (handler->next != NULL) {
Packit fcad23
        h->next = netsnmp_handler_dup(handler->next);
Packit fcad23
        if (!h->next)
Packit fcad23
            goto err;
Packit fcad23
        h->next->prev = h;
Packit fcad23
    }
Packit fcad23
    h->prev = NULL;
Packit fcad23
    return h;
Packit fcad23
Packit fcad23
err:
Packit fcad23
    netsnmp_handler_free(h);
Packit fcad23
    return NULL;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Free resources associated with a handler registration object.
Packit fcad23
 *  The registration object and all MIB Handlers in the chain are
Packit fcad23
 *  freed. When the function ends, given pointer is no longer valid
Packit fcad23
 *  and should be set to NULL.
Packit fcad23
 *
Packit fcad23
 *  @param reginfo is the handler registration object to be freed
Packit fcad23
 */
Packit fcad23
void
Packit fcad23
netsnmp_handler_registration_free(netsnmp_handler_registration *reginfo)
Packit fcad23
{
Packit fcad23
    if (reginfo != NULL) {
Packit fcad23
        netsnmp_handler_free(reginfo->handler);
Packit fcad23
        SNMP_FREE(reginfo->handlerName);
Packit fcad23
        SNMP_FREE(reginfo->contextName);
Packit fcad23
        SNMP_FREE(reginfo->rootoid);
Packit fcad23
        reginfo->rootoid_len = 0;
Packit fcad23
        SNMP_FREE(reginfo);
Packit fcad23
    }
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Duplicates handler registration object and all subsequent handlers.
Packit fcad23
 *  Creates a copy of the handler registration object and all its data.
Packit fcad23
 *
Packit fcad23
 *  @param reginfo is the handler registration object to be duplicated
Packit fcad23
 *
Packit fcad23
 *  @return Returns a pointer to the complete copy,
Packit fcad23
 *         or NULL if any problem occured.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_handler_dup()
Packit fcad23
 */
Packit fcad23
netsnmp_handler_registration *
Packit fcad23
netsnmp_handler_registration_dup(netsnmp_handler_registration *reginfo)
Packit fcad23
{
Packit fcad23
    netsnmp_handler_registration *r = NULL;
Packit fcad23
Packit fcad23
    if (reginfo == NULL) {
Packit fcad23
        return NULL;
Packit fcad23
    }
Packit fcad23
Packit fcad23
Packit fcad23
    r = (netsnmp_handler_registration *) calloc(1,
Packit fcad23
                                                sizeof
Packit fcad23
                                                (netsnmp_handler_registration));
Packit fcad23
Packit fcad23
    if (r != NULL) {
Packit fcad23
        r->modes = reginfo->modes;
Packit fcad23
        r->priority = reginfo->priority;
Packit fcad23
        r->range_subid = reginfo->range_subid;
Packit fcad23
        r->timeout = reginfo->timeout;
Packit fcad23
        r->range_ubound = reginfo->range_ubound;
Packit fcad23
        r->rootoid_len = reginfo->rootoid_len;
Packit fcad23
Packit fcad23
        if (reginfo->handlerName != NULL) {
Packit fcad23
            r->handlerName = strdup(reginfo->handlerName);
Packit fcad23
            if (r->handlerName == NULL) {
Packit fcad23
                netsnmp_handler_registration_free(r);
Packit fcad23
                return NULL;
Packit fcad23
            }
Packit fcad23
        }
Packit fcad23
Packit fcad23
        if (reginfo->contextName != NULL) {
Packit fcad23
            r->contextName = strdup(reginfo->contextName);
Packit fcad23
            if (r->contextName == NULL) {
Packit fcad23
                netsnmp_handler_registration_free(r);
Packit fcad23
                return NULL;
Packit fcad23
            }
Packit fcad23
        }
Packit fcad23
Packit fcad23
        if (reginfo->rootoid != NULL) {
Packit fcad23
            r->rootoid =
Packit fcad23
                snmp_duplicate_objid(reginfo->rootoid, reginfo->rootoid_len);
Packit fcad23
            if (r->rootoid == NULL) {
Packit fcad23
                netsnmp_handler_registration_free(r);
Packit fcad23
                return NULL;
Packit fcad23
            }
Packit fcad23
        }
Packit fcad23
Packit fcad23
        r->handler = netsnmp_handler_dup(reginfo->handler);
Packit fcad23
        if (r->handler == NULL) {
Packit fcad23
            netsnmp_handler_registration_free(r);
Packit fcad23
            return NULL;
Packit fcad23
        }
Packit fcad23
        return r;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    return NULL;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Creates a cache of information which can be saved for future reference.
Packit fcad23
 *  The cache is filled with pointers from parameters. Note that
Packit fcad23
 *  the input structures are not duplicated, but put directly into
Packit fcad23
 *  the new cache struct.
Packit fcad23
 *  Use netsnmp_handler_check_cache() later to make sure it's still
Packit fcad23
 *  valid before referencing it in the future.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_handler_check_cache()
Packit fcad23
 * @see netsnmp_free_delegated_cache()
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE netsnmp_delegated_cache *
Packit fcad23
netsnmp_create_delegated_cache(netsnmp_mib_handler *handler,
Packit fcad23
                               netsnmp_handler_registration *reginfo,
Packit fcad23
                               netsnmp_agent_request_info *reqinfo,
Packit fcad23
                               netsnmp_request_info *requests,
Packit fcad23
                               void *localinfo)
Packit fcad23
{
Packit fcad23
    netsnmp_delegated_cache *ret;
Packit fcad23
Packit fcad23
    ret = SNMP_MALLOC_TYPEDEF(netsnmp_delegated_cache);
Packit fcad23
    if (ret) {
Packit fcad23
        ret->transaction_id = reqinfo->asp->pdu->transid;
Packit fcad23
        ret->handler = handler;
Packit fcad23
        ret->reginfo = reginfo;
Packit fcad23
        ret->reqinfo = reqinfo;
Packit fcad23
        ret->requests = requests;
Packit fcad23
        ret->localinfo = localinfo;
Packit fcad23
    }
Packit fcad23
    return ret;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Check if a given delegated cache is still valid.
Packit fcad23
 *  The cache is valid if it's a part of transaction
Packit fcad23
 *  (ie, the agent still considers it to be an outstanding request).
Packit fcad23
 *
Packit fcad23
 *  @param dcache is the delegated cache to be checked.
Packit fcad23
 *
Packit fcad23
 *  @return Returns the cache pointer if it is still valid, NULL otherwise.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_create_delegated_cache()
Packit fcad23
 * @see netsnmp_free_delegated_cache()
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE netsnmp_delegated_cache *
Packit fcad23
netsnmp_handler_check_cache(netsnmp_delegated_cache *dcache)
Packit fcad23
{
Packit fcad23
    if (!dcache)
Packit fcad23
        return dcache;
Packit fcad23
Packit fcad23
    if (netsnmp_check_transaction_id(dcache->transaction_id) ==
Packit fcad23
        SNMPERR_SUCCESS)
Packit fcad23
        return dcache;
Packit fcad23
Packit fcad23
    return NULL;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Free a cache once it's no longer used.
Packit fcad23
 *  Deletes the data allocated by netsnmp_create_delegated_cache().
Packit fcad23
 *  Structures which were not allocated by netsnmp_create_delegated_cache()
Packit fcad23
 *  are not freed (pointers are dropped).
Packit fcad23
 *
Packit fcad23
 *  @param dcache is the delegated cache to be freed.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_create_delegated_cache()
Packit fcad23
 * @see netsnmp_handler_check_cache()
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE void
Packit fcad23
netsnmp_free_delegated_cache(netsnmp_delegated_cache *dcache)
Packit fcad23
{
Packit fcad23
    /*
Packit fcad23
     * right now, no extra data is there that needs to be freed 
Packit fcad23
     */
Packit fcad23
    if (dcache)
Packit fcad23
        SNMP_FREE(dcache);
Packit fcad23
Packit fcad23
    return;
Packit fcad23
}
Packit fcad23
Packit fcad23
Packit fcad23
#ifndef NETSNMP_FEATURE_REMOVE_HANDLER_MARK_REQUESTS_AS_DELEGATED
Packit fcad23
/** Sets a list of requests as delegated or not delegated.
Packit fcad23
 *  Sweeps through given chain of requests and sets 'delegated'
Packit fcad23
 *  flag accordingly to the isdelegaded parameter.
Packit fcad23
 *
Packit fcad23
 *  @param requests Request list.
Packit fcad23
 *  @param isdelegated New value of the 'delegated' flag.
Packit fcad23
 */
Packit fcad23
void
Packit fcad23
netsnmp_handler_mark_requests_as_delegated(netsnmp_request_info *requests,
Packit fcad23
                                           int isdelegated)
Packit fcad23
{
Packit fcad23
    while (requests) {
Packit fcad23
        requests->delegated = isdelegated;
Packit fcad23
        requests = requests->next;
Packit fcad23
    }
Packit fcad23
}
Packit fcad23
#endif /* NETSNMP_FEATURE_REMOVE_HANDLER_MARK_REQUESTS_AS_DELEGATED */
Packit fcad23
Packit fcad23
/** Adds data from node list to the request information structure.
Packit fcad23
 *  Data in the request can be later extracted and used by submodules.
Packit fcad23
 *
Packit fcad23
 * @param request Destination request information structure.
Packit fcad23
 *
Packit fcad23
 * @param node The data to be added to the linked list
Packit fcad23
 *             request->parent_data.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_request_remove_list_data()
Packit fcad23
 * @see netsnmp_request_get_list_data()
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE void
Packit fcad23
netsnmp_request_add_list_data(netsnmp_request_info *request,
Packit fcad23
                              netsnmp_data_list *node)
Packit fcad23
{
Packit fcad23
    if (request) {
Packit fcad23
        if (request->parent_data)
Packit fcad23
            netsnmp_add_list_data(&request->parent_data, node);
Packit fcad23
        else
Packit fcad23
            request->parent_data = node;
Packit fcad23
    }
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Removes all data from a request.
Packit fcad23
 *
Packit fcad23
 * @param request the netsnmp request info structure
Packit fcad23
 *
Packit fcad23
 * @param name this is the name of the previously added data
Packit fcad23
 *
Packit fcad23
 * @return 0 on successful find-and-delete, 1 otherwise.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_request_add_list_data()
Packit fcad23
 * @see netsnmp_request_get_list_data()
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE int
Packit fcad23
netsnmp_request_remove_list_data(netsnmp_request_info *request,
Packit fcad23
                                 const char *name)
Packit fcad23
{
Packit fcad23
    if ((NULL == request) || (NULL ==request->parent_data))
Packit fcad23
        return 1;
Packit fcad23
Packit fcad23
    return netsnmp_remove_list_node(&request->parent_data, name);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Extracts data from a request.
Packit fcad23
 *  Retrieves data that was previously added to the request,
Packit fcad23
 *  usually by a parent module.
Packit fcad23
 *
Packit fcad23
 * @param request Source request information structure.
Packit fcad23
 *
Packit fcad23
 * @param name Used to compare against the request->parent_data->name value;
Packit fcad23
 *             if a match is found, request->parent_data->data is returned.
Packit fcad23
 *
Packit fcad23
 * @return Gives a void pointer(request->parent_data->data); NULL is
Packit fcad23
 *         returned if source data is NULL or the object is not found.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_request_add_list_data()
Packit fcad23
 * @see netsnmp_request_remove_list_data()
Packit fcad23
 */
Packit fcad23
void    *
Packit fcad23
netsnmp_request_get_list_data(netsnmp_request_info *request,
Packit fcad23
                              const char *name)
Packit fcad23
{
Packit fcad23
    if (request)
Packit fcad23
        return netsnmp_get_list_data(request->parent_data, name);
Packit fcad23
    return NULL;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Free the extra data stored in a request.
Packit fcad23
 *  Deletes the data in given request only. Other chain items
Packit fcad23
 *  are unaffected.
Packit fcad23
 *
Packit fcad23
 * @param request Request information structure to be modified.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_request_add_list_data()
Packit fcad23
 * @see netsnmp_free_list_data()
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE void
Packit fcad23
netsnmp_free_request_data_set(netsnmp_request_info *request)
Packit fcad23
{
Packit fcad23
    if (request)
Packit fcad23
        netsnmp_free_list_data(request->parent_data);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Free the extra data stored in a bunch of requests.
Packit fcad23
 *  Deletes all data in the chain inside request.
Packit fcad23
 *
Packit fcad23
 * @param request Request information structure to be modified.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_request_add_list_data()
Packit fcad23
 * @see netsnmp_free_request_data_set()
Packit fcad23
 */
Packit fcad23
NETSNMP_INLINE void
Packit fcad23
netsnmp_free_request_data_sets(netsnmp_request_info *request)
Packit fcad23
{
Packit fcad23
    if (request && request->parent_data) {
Packit fcad23
        netsnmp_free_all_list_data(request->parent_data);
Packit fcad23
        request->parent_data = NULL;
Packit fcad23
    }
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Returns a MIB handler from a chain based on the name.
Packit fcad23
 *
Packit fcad23
 * @param reginfo Handler registration struct which contains the chain.
Packit fcad23
 *
Packit fcad23
 * @param name Target MIB Handler name string. The name is case
Packit fcad23
 *        sensitive.
Packit fcad23
 *
Packit fcad23
 * @return The MIB Handler is returned, or NULL if not found.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_request_add_list_data()
Packit fcad23
 */
Packit fcad23
netsnmp_mib_handler *
Packit fcad23
netsnmp_find_handler_by_name(netsnmp_handler_registration *reginfo,
Packit fcad23
                             const char *name)
Packit fcad23
{
Packit fcad23
    netsnmp_mib_handler *it;
Packit fcad23
    if (reginfo == NULL || name == NULL )
Packit fcad23
        return NULL;
Packit fcad23
    for (it = reginfo->handler; it; it = it->next) {
Packit fcad23
        if (strcmp(it->handler_name, name) == 0) {
Packit fcad23
            return it;
Packit fcad23
        }
Packit fcad23
    }
Packit fcad23
    return NULL;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Returns a handler's void pointer from a chain based on the name.
Packit fcad23
 *
Packit fcad23
 * @warning The void pointer data may change as a handler evolves.
Packit fcad23
 *        Handlers should really advertise some function for you
Packit fcad23
 *        to use instead.
Packit fcad23
 *
Packit fcad23
 * @param reginfo Handler registration struct which contains the chain.
Packit fcad23
 *
Packit fcad23
 * @param name Target MIB Handler name string. The name is case
Packit fcad23
 *        sensitive.
Packit fcad23
 *
Packit fcad23
 * @return The MIB Handler's void * pointer is returned,
Packit fcad23
 *        or NULL if the handler is not found.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_find_handler_by_name()
Packit fcad23
 */
Packit fcad23
void           *
Packit fcad23
netsnmp_find_handler_data_by_name(netsnmp_handler_registration *reginfo,
Packit fcad23
                                  const char *name)
Packit fcad23
{
Packit fcad23
    netsnmp_mib_handler *it = netsnmp_find_handler_by_name(reginfo, name);
Packit fcad23
    if (it)
Packit fcad23
        return it->myvoid;
Packit fcad23
    return NULL;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** @private
Packit fcad23
 *  Clones a MIB Handler with its basic properties.
Packit fcad23
 *  Creates a copy of the given MIB Handler. Copies name, flags and
Packit fcad23
 *  access methods only; not myvoid.
Packit fcad23
 *
Packit fcad23
 *  @see netsnmp_handler_dup()
Packit fcad23
 */
Packit fcad23
static netsnmp_mib_handler *
Packit fcad23
_clone_handler(netsnmp_mib_handler *it)
Packit fcad23
{
Packit fcad23
    netsnmp_mib_handler *dup;
Packit fcad23
Packit fcad23
    if(NULL == it)
Packit fcad23
        return NULL;
Packit fcad23
Packit fcad23
    dup = netsnmp_create_handler(it->handler_name, it->access_method);
Packit fcad23
    if(NULL != dup)
Packit fcad23
        dup->flags = it->flags;
Packit fcad23
Packit fcad23
    return dup;
Packit fcad23
}
Packit fcad23
Packit fcad23
static netsnmp_data_list *handler_reg = NULL;
Packit fcad23
Packit fcad23
void
Packit fcad23
handler_free_callback(void *handler)
Packit fcad23
{
Packit fcad23
    netsnmp_handler_free((netsnmp_mib_handler *)handler);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Registers a given handler by name, so that it can be found easily later.
Packit fcad23
 *  Pointer to the handler is put into a list where it can be easily located
Packit fcad23
 *  at any time.
Packit fcad23
 *
Packit fcad23
 * @param name Name string to be associated with the handler.
Packit fcad23
 *
Packit fcad23
 * @param handler Pointer the MIB Handler.
Packit fcad23
 *
Packit fcad23
 * @see netsnmp_clear_handler_list()
Packit fcad23
 */
Packit fcad23
void
Packit fcad23
netsnmp_register_handler_by_name(const char *name,
Packit fcad23
                                 netsnmp_mib_handler *handler)
Packit fcad23
{
Packit fcad23
    netsnmp_add_list_data(&handler_reg,
Packit fcad23
                          netsnmp_create_data_list(name, (void *) handler,
Packit fcad23
                                                   handler_free_callback));
Packit fcad23
    DEBUGMSGTL(("handler_registry", "registering helper %s\n", name));
Packit fcad23
}
Packit fcad23
Packit fcad23
/** Clears the entire MIB Handlers registration list.
Packit fcad23
 *  MIB Handlers registration list is used to access any MIB Handler by
Packit fcad23
 *  its name. The function frees the list memory and sets pointer to NULL.
Packit fcad23
 *  Instead of calling this function directly, use shutdown_agent().
Packit fcad23
 *
Packit fcad23
 *  @see shutdown_agent()
Packit fcad23
 *  @see netsnmp_register_handler_by_name()
Packit fcad23
 */
Packit fcad23
void
Packit fcad23
netsnmp_clear_handler_list(void)
Packit fcad23
{
Packit fcad23
    DEBUGMSGTL(("agent_handler", "netsnmp_clear_handler_list() called\n"));
Packit fcad23
    netsnmp_free_all_list_data(handler_reg);
Packit fcad23
    handler_reg = NULL;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** @private
Packit fcad23
 *  Injects a handler into a subtree, peers and children when a given
Packit fcad23
 *  subtrees name matches a passed in name.
Packit fcad23
 */
Packit fcad23
void
Packit fcad23
netsnmp_inject_handler_into_subtree(netsnmp_subtree *tp, const char *name,
Packit fcad23
                                    netsnmp_mib_handler *handler,
Packit fcad23
                                    const char *before_what)
Packit fcad23
{
Packit fcad23
    netsnmp_subtree *tptr;
Packit fcad23
    netsnmp_mib_handler *mh;
Packit fcad23
Packit fcad23
    for (tptr = tp; tptr != NULL; tptr = tptr->next) {
Packit fcad23
        /*  if (tptr->children) { 
Packit fcad23
              netsnmp_inject_handler_into_subtree(tptr->children,name,handler);
Packit fcad23
	    }   */
Packit fcad23
        if (strcmp(tptr->label_a, name) == 0) {
Packit fcad23
            DEBUGMSGTL(("injectHandler", "injecting handler %s into %s\n",
Packit fcad23
                        handler->handler_name, tptr->label_a));
Packit fcad23
            netsnmp_inject_handler_before(tptr->reginfo, _clone_handler(handler),
Packit fcad23
                                          before_what);
Packit fcad23
        } else if (tptr->reginfo != NULL &&
Packit fcad23
		   tptr->reginfo->handlerName != NULL &&
Packit fcad23
                   strcmp(tptr->reginfo->handlerName, name) == 0) {
Packit fcad23
            DEBUGMSGTL(("injectHandler", "injecting handler into %s/%s\n",
Packit fcad23
                        tptr->label_a, tptr->reginfo->handlerName));
Packit fcad23
            netsnmp_inject_handler_before(tptr->reginfo, _clone_handler(handler),
Packit fcad23
                                          before_what);
Packit fcad23
        } else if (tptr->reginfo != NULL) {
Packit fcad23
            for (mh = tptr->reginfo->handler; mh != NULL; mh = mh->next) {
Packit fcad23
                if (mh->handler_name && strcmp(mh->handler_name, name) == 0) {
Packit fcad23
                    DEBUGMSGTL(("injectHandler", "injecting handler into %s\n",
Packit fcad23
                                tptr->label_a));
Packit fcad23
                    netsnmp_inject_handler_before(tptr->reginfo,
Packit fcad23
                                                  _clone_handler(handler),
Packit fcad23
                                                  before_what);
Packit fcad23
                    break;
Packit fcad23
                } else {
Packit fcad23
                    DEBUGMSGTL(("injectHandler",
Packit fcad23
                                "not injecting handler into %s\n",
Packit fcad23
                                mh->handler_name));
Packit fcad23
                }
Packit fcad23
            }
Packit fcad23
        }
Packit fcad23
    }
Packit fcad23
}
Packit fcad23
Packit fcad23
static int      doneit = 0;
Packit fcad23
/** @private
Packit fcad23
 *  Parses the "injectHandler" token line.
Packit fcad23
 */
Packit fcad23
void
Packit fcad23
parse_injectHandler_conf(const char *token, char *cptr)
Packit fcad23
{
Packit fcad23
    char            handler_to_insert[256], reg_name[256];
Packit fcad23
    subtree_context_cache *stc;
Packit fcad23
    netsnmp_mib_handler *handler;
Packit fcad23
Packit fcad23
    /*
Packit fcad23
     * XXXWWW: ensure instead that handler isn't inserted twice 
Packit fcad23
     */
Packit fcad23
    if (doneit)                 /* we only do this once without restart the agent */
Packit fcad23
        return;
Packit fcad23
Packit fcad23
    cptr = copy_nword(cptr, handler_to_insert, sizeof(handler_to_insert));
Packit fcad23
    handler = (netsnmp_mib_handler*)netsnmp_get_list_data(handler_reg, handler_to_insert);
Packit fcad23
    if (!handler) {
Packit fcad23
	netsnmp_config_error("no \"%s\" handler registered.",
Packit fcad23
			     handler_to_insert);
Packit fcad23
        return;
Packit fcad23
    }
Packit fcad23
Packit fcad23
    if (!cptr) {
Packit fcad23
        config_perror("no INTONAME specified.  Can't do insertion.");
Packit fcad23
        return;
Packit fcad23
    }
Packit fcad23
    cptr = copy_nword(cptr, reg_name, sizeof(reg_name));
Packit fcad23
Packit fcad23
    for (stc = get_top_context_cache(); stc; stc = stc->next) {
Packit fcad23
        DEBUGMSGTL(("injectHandler", "Checking context tree %s (before=%s)\n",
Packit fcad23
                    stc->context_name, (cptr)?cptr:"null"));
Packit fcad23
        netsnmp_inject_handler_into_subtree(stc->first_subtree, reg_name,
Packit fcad23
                                            handler, cptr);
Packit fcad23
    }
Packit fcad23
}
Packit fcad23
Packit fcad23
/** @private
Packit fcad23
 *  Callback to ensure injectHandler parser doesn't do things twice.
Packit fcad23
 *  @todo replace this with a method to check the handler chain instead.
Packit fcad23
 */
Packit fcad23
static int
Packit fcad23
handler_mark_inject_handler_done(int majorID, int minorID,
Packit fcad23
                    void *serverarg, void *clientarg)
Packit fcad23
{
Packit fcad23
    doneit = 1;
Packit fcad23
    return 0;
Packit fcad23
}
Packit fcad23
Packit fcad23
/** @private
Packit fcad23
 *  Registers the injectHandle parser token.
Packit fcad23
 *  Used in init_agent_read_config().
Packit fcad23
 *
Packit fcad23
 *  @see init_agent_read_config()
Packit fcad23
 */
Packit fcad23
void
Packit fcad23
netsnmp_init_handler_conf(void)
Packit fcad23
{
Packit fcad23
    snmpd_register_config_handler("injectHandler",
Packit fcad23
                                  parse_injectHandler_conf,
Packit fcad23
                                  NULL, "injectHandler NAME INTONAME [BEFORE_OTHER_NAME]");
Packit fcad23
    snmp_register_callback(SNMP_CALLBACK_LIBRARY,
Packit fcad23
                           SNMP_CALLBACK_POST_READ_CONFIG,
Packit fcad23
                           handler_mark_inject_handler_done, NULL);
Packit fcad23
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("GET"), MODE_GET);
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("GETNEXT"), MODE_GETNEXT);
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("GETBULK"), MODE_GETBULK);
Packit fcad23
#ifndef NETSNMP_NO_WRITE_SUPPORT
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("SET_BEGIN"),
Packit fcad23
                         MODE_SET_BEGIN);
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("SET_RESERVE1"),
Packit fcad23
                         MODE_SET_RESERVE1);
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("SET_RESERVE2"),
Packit fcad23
                         MODE_SET_RESERVE2);
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("SET_ACTION"),
Packit fcad23
                         MODE_SET_ACTION);
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("SET_COMMIT"),
Packit fcad23
                         MODE_SET_COMMIT);
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("SET_FREE"), MODE_SET_FREE);
Packit fcad23
    se_add_pair_to_slist("agent_mode", strdup("SET_UNDO"), MODE_SET_UNDO);
Packit fcad23
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("pre-request"),
Packit fcad23
                         MODE_BSTEP_PRE_REQUEST);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("object_lookup"),
Packit fcad23
                         MODE_BSTEP_OBJECT_LOOKUP);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("check_value"),
Packit fcad23
                         MODE_BSTEP_CHECK_VALUE);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("row_create"),
Packit fcad23
                         MODE_BSTEP_ROW_CREATE);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("undo_setup"),
Packit fcad23
                         MODE_BSTEP_UNDO_SETUP);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("set_value"),
Packit fcad23
                         MODE_BSTEP_SET_VALUE);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("check_consistency"),
Packit fcad23
                         MODE_BSTEP_CHECK_CONSISTENCY);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("undo_set"),
Packit fcad23
                         MODE_BSTEP_UNDO_SET);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("commit"),
Packit fcad23
                         MODE_BSTEP_COMMIT);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("undo_commit"),
Packit fcad23
                         MODE_BSTEP_UNDO_COMMIT);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("irreversible_commit"),
Packit fcad23
                         MODE_BSTEP_IRREVERSIBLE_COMMIT);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("undo_cleanup"),
Packit fcad23
                         MODE_BSTEP_UNDO_CLEANUP);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("post_request"),
Packit fcad23
                         MODE_BSTEP_POST_REQUEST);
Packit fcad23
    se_add_pair_to_slist("babystep_mode", strdup("original"), 0xffff);
Packit fcad23
#endif /* NETSNMP_NO_WRITE_SUPPORT */
Packit fcad23
Packit fcad23
    /*
Packit fcad23
     * xxx-rks: hmmm.. will this work for modes which are or'd together?
Packit fcad23
     *          I'm betting not...
Packit fcad23
     */
Packit fcad23
    se_add_pair_to_slist("handler_can_mode", strdup("GET/GETNEXT"),
Packit fcad23
                         HANDLER_CAN_GETANDGETNEXT);
Packit fcad23
    se_add_pair_to_slist("handler_can_mode", strdup("SET"),
Packit fcad23
                         HANDLER_CAN_SET);
Packit fcad23
    se_add_pair_to_slist("handler_can_mode", strdup("GETBULK"),
Packit fcad23
                         HANDLER_CAN_GETBULK);
Packit fcad23
    se_add_pair_to_slist("handler_can_mode", strdup("BABY_STEP"),
Packit fcad23
                         HANDLER_CAN_BABY_STEP);
Packit fcad23
}
Packit fcad23
Packit fcad23
/** @} */