2024-07-26 08:43:00 +00:00
|
|
|
/*
|
|
|
|
* BIRD -- Simple Network Management Procotol (SNMP)
|
|
|
|
*
|
|
|
|
* (c) 2024 Vojtech Vilimek <vojtech.vilimek@nic.cz>
|
|
|
|
* (c) 2024 CZ.NIC z.s.p.o.
|
2022-08-01 11:01:49 +00:00
|
|
|
*
|
|
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
|
|
*/
|
|
|
|
|
2023-09-04 07:25:51 +00:00
|
|
|
/**
|
2023-11-16 06:11:14 +00:00
|
|
|
* DOC: Simple Network Management Protocol
|
|
|
|
*
|
|
|
|
* The SNMP protocol is divided into several parts: |snmp.c| which implements
|
2024-08-20 09:11:58 +00:00
|
|
|
* integration with BIRD core, |subagent.c| provides AgentX subagent behaviour
|
|
|
|
* as well as functions for creating and parsing packets. In file |mib_tree.c|
|
|
|
|
* is implemented OID prefix tree for storing supported MIBs. File |bgp4_mib.c|
|
|
|
|
* implements parts of BGP4-MIB, |snmp_utils.c| is collection of helper
|
|
|
|
* functions for whole SNMP protocol.
|
2023-11-16 06:11:14 +00:00
|
|
|
*
|
|
|
|
* Althrough called SNMP the BIRD does not implement SNMP directly but acts as
|
|
|
|
* an AgentX subagent. AgentX subagent connects to AgentX master agent that
|
|
|
|
* processes incomming SNMP requests and passes them down to the correct
|
|
|
|
* subagent. Therefore you need also a running master agent somewhere.
|
|
|
|
* Advantages of this design are that you are capable of doing aggregation of
|
|
|
|
* statuses of multiple BIRDs at the master agent level and much simpler
|
|
|
|
* implementation.
|
|
|
|
*
|
|
|
|
* Before any of the SNMP request could be processed, the SNMP need to
|
|
|
|
* established AgentX session with the master agent and need to register all
|
|
|
|
* subtrees to make them accessible from the master agent. The establishement of
|
|
|
|
* the of session is handled by snmp_start(), snmp_start_locked() and
|
|
|
|
* snmp_start_subagent(). Then we register all MIBs from configuration in
|
|
|
|
* snmp_register_mibs().
|
|
|
|
*
|
|
|
|
* The AgentX request are handled only during MIB subtree registrations and
|
|
|
|
* after then on established session (in states SNMP_REGISTER and SNMP_CONN, see
|
|
|
|
* below). It is also guaranteed that no request is received before MIB subtree
|
|
|
|
* registration because the specific subagent is not authoratitave and also the
|
|
|
|
* master agent has no info about MIB subtree supported by subagent. The AgentX
|
|
|
|
* requests are handled by function snmp_rx() in |subagent.c|.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*
|
2024-08-20 09:11:58 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2023-11-16 06:11:14 +00:00
|
|
|
* SNMP State Machine
|
2023-09-04 07:25:51 +00:00
|
|
|
*
|
|
|
|
* States with main transitions
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* +-----------------+
|
|
|
|
* | SNMP_INIT | entry state after call snmp_start()
|
|
|
|
* +-----------------+
|
|
|
|
* |
|
2024-08-09 19:05:32 +00:00
|
|
|
* | acquiring object lock for tcp communication socket
|
2023-09-04 07:25:51 +00:00
|
|
|
* V
|
|
|
|
* +-----------------+
|
|
|
|
* | SNMP_LOCKED | object lock aquired
|
|
|
|
* +-----------------+
|
|
|
|
* |
|
2023-10-25 10:41:23 +00:00
|
|
|
* | opening communication socket
|
2023-09-04 07:25:51 +00:00
|
|
|
* V
|
|
|
|
* +-----------------+
|
|
|
|
* | SNMP_OPEN | socket created, starting subagent
|
|
|
|
* +-----------------+
|
|
|
|
* |
|
2024-08-09 19:05:32 +00:00
|
|
|
* | BIRD receive response for agentx-Open-PDU
|
2023-09-04 07:25:51 +00:00
|
|
|
* V
|
|
|
|
* +-----------------+
|
|
|
|
* | SNMP_REGISTER | session was established, subagent registers MIBs
|
|
|
|
* +-----------------+
|
|
|
|
* |
|
2024-08-09 19:05:32 +00:00
|
|
|
* | subagent received response for any registration requests
|
2023-09-04 07:25:51 +00:00
|
|
|
* V
|
|
|
|
* +-----------------+
|
|
|
|
* | SNMP_CONN | everything is set
|
|
|
|
* +-----------------+
|
|
|
|
* |
|
2024-08-09 19:05:32 +00:00
|
|
|
* | received malformed PDU, protocol disabled,
|
|
|
|
* | BIRD sends agentx-Close-PDU or agentx-Response-PDU with an error
|
2023-09-04 07:25:51 +00:00
|
|
|
* V
|
|
|
|
* +-----------------+
|
2024-08-09 19:05:32 +00:00
|
|
|
* | SNMP_STOP | waiting until the prepared PDUs are sent
|
2023-09-04 07:25:51 +00:00
|
|
|
* +-----------------+
|
|
|
|
* |
|
2024-08-09 19:05:32 +00:00
|
|
|
* | cleaning protocol state
|
2023-09-04 07:25:51 +00:00
|
|
|
* V
|
|
|
|
* +-----------------+
|
|
|
|
* | SNMP_DOWN | session is closed
|
|
|
|
* +-----------------+
|
|
|
|
*
|
|
|
|
*
|
2023-10-25 10:56:23 +00:00
|
|
|
*
|
2023-09-04 07:25:51 +00:00
|
|
|
* Erroneous transitions:
|
2024-08-09 19:05:32 +00:00
|
|
|
* SNMP is UP (PS_UP) in states SNMP_CONN and also in SNMP_REGISTER because
|
|
|
|
* the session is establised and the GetNext request should be responsed
|
|
|
|
* without regards to MIB registration.
|
2023-09-04 07:25:51 +00:00
|
|
|
*
|
|
|
|
* Reconfiguration is done in similar fashion to BGP, the reconfiguration
|
|
|
|
* request is declined, the protocols is stoped and started with new
|
|
|
|
* configuration.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2022-08-01 11:01:49 +00:00
|
|
|
#include "nest/bird.h"
|
|
|
|
#include "nest/cli.h"
|
2022-08-10 15:31:32 +00:00
|
|
|
#include "nest/locks.h"
|
|
|
|
#include "lib/socket.h"
|
2022-09-20 12:28:57 +00:00
|
|
|
#include "lib/lists.h"
|
2022-08-01 11:01:49 +00:00
|
|
|
|
2022-08-10 15:31:32 +00:00
|
|
|
#include "snmp.h"
|
|
|
|
#include "subagent.h"
|
2023-07-26 12:34:01 +00:00
|
|
|
#include "snmp_utils.h"
|
2024-07-04 14:33:44 +00:00
|
|
|
#include "mib_tree.h"
|
2024-07-26 08:43:00 +00:00
|
|
|
#include "bgp4_mib.h"
|
2022-08-10 15:31:32 +00:00
|
|
|
|
2024-07-22 16:17:35 +00:00
|
|
|
const char agentx_master_addr[] = AGENTX_MASTER_ADDR;
|
2023-11-15 14:03:55 +00:00
|
|
|
|
2024-08-09 19:05:32 +00:00
|
|
|
static const char *snmp_state_str[] = {
|
|
|
|
[SNMP_INIT] = "acquiring address lock",
|
|
|
|
[SNMP_LOCKED] = "address lock acquired",
|
|
|
|
[SNMP_OPEN] = "starting AgentX subagent",
|
|
|
|
[SNMP_REGISTER] = "registering MIBs",
|
|
|
|
[SNMP_CONN] = "AgentX session established",
|
|
|
|
[SNMP_STOP] = "stopping AgentX subagent",
|
|
|
|
[SNMP_DOWN] = "protocol down",
|
|
|
|
};
|
|
|
|
|
2024-08-15 13:52:15 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Callbacks
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* snmp_sock_err - handle errors on socket by reopenning the socket
|
|
|
|
* @sk: socket owned by SNMP protocol instance
|
|
|
|
* @err: socket error code
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
snmp_sock_err(sock *sk, int UNUSED err)
|
|
|
|
{
|
|
|
|
struct snmp_proto *p = sk->data;
|
|
|
|
if (err != 0)
|
|
|
|
TRACE(D_EVENTS, "SNMP socket error (%d)", err);
|
|
|
|
snmp_set_state(p, SNMP_DOWN);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* snmp_ping_timeout - send a agentx-Ping-PDU
|
|
|
|
* @tm: the ping_timer holding the SNMP protocol instance.
|
|
|
|
*
|
|
|
|
* Send an agentx-Ping-PDU. This function is periodically called by ping
|
|
|
|
* timer.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
snmp_ping_timeout(timer *tm)
|
|
|
|
{
|
|
|
|
struct snmp_proto *p = tm->data;
|
|
|
|
snmp_ping(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* snmp_stop_timeout - a timeout for non-responding master agent
|
|
|
|
* @tm: the startup_timer holding the SNMP protocol instance.
|
|
|
|
*
|
|
|
|
* We are trying to empty the TX buffer of communication socket. But if it is
|
|
|
|
* not done in reasonable amount of time, the function is called by timeout
|
|
|
|
* timer. We down the whole SNMP protocol with cleanup of associated data
|
|
|
|
* structures.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
snmp_stop_timeout(timer *tm)
|
|
|
|
{
|
|
|
|
struct snmp_proto *p = tm->data;
|
|
|
|
snmp_set_state(p, SNMP_DOWN);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* snmp_connected - start AgentX session on created socket
|
|
|
|
* @sk: socket owned by SNMP protocol instance
|
|
|
|
*
|
|
|
|
* Starts the AgentX communication by sending an agentx-Open-PDU.
|
|
|
|
* This function is internal and shouldn't be used outside the SNMP module.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
snmp_connected(sock *sk)
|
|
|
|
{
|
|
|
|
struct snmp_proto *p = sk->data;
|
|
|
|
snmp_set_state(p, SNMP_OPEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* snmp_start_locked - open the socket on locked address
|
|
|
|
* @lock: object lock guarding the communication mean (address, ...)
|
|
|
|
*
|
|
|
|
* This function is called when the object lock is acquired. Main goal is to set
|
|
|
|
* socket parameters and try to open configured socket. Function
|
|
|
|
* snmp_connected() handles next stage of SNMP protocol start. When the socket
|
|
|
|
* coundn't be opened, a new try is scheduled after a small delay.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
snmp_start_locked(struct object_lock *lock)
|
|
|
|
{
|
|
|
|
struct snmp_proto *p = lock->data;
|
|
|
|
if (p->startup_delay)
|
|
|
|
{
|
|
|
|
ASSERT(p->startup_timer);
|
|
|
|
p->startup_timer->hook = snmp_startup_timeout;
|
|
|
|
tm_start(p->startup_timer, p->startup_delay);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
snmp_set_state(p, SNMP_LOCKED);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* snmp_startup_timeout - start the initiliazed SNMP protocol
|
|
|
|
* @tm: the startup_timer holding the SNMP protocol instance.
|
|
|
|
*
|
|
|
|
* When the timer rings, the function snmp_startup() is invoked.
|
|
|
|
* This function is internal and shouldn't be used outside the SNMP module.
|
|
|
|
* Used when we delaying the start procedure, or we want to retry opening
|
|
|
|
* the communication socket.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
snmp_startup_timeout(timer *tm)
|
|
|
|
{
|
|
|
|
struct snmp_proto *p = tm->data;
|
|
|
|
snmp_set_state(p, SNMP_LOCKED);
|
|
|
|
}
|
|
|
|
|
2023-11-15 14:03:55 +00:00
|
|
|
/*
|
|
|
|
* snmp_rx_skip - skip all received data
|
|
|
|
* @sk: communication socket
|
|
|
|
* @size: size of received PDUs
|
|
|
|
*
|
|
|
|
* Socket rx_hook used when we are reseting the connection due to malformed PDU.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
snmp_rx_skip(sock UNUSED *sk, uint UNUSED size)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2024-08-09 19:05:32 +00:00
|
|
|
* snmp_tx_skip - handle empty TX buffer during session reset
|
2023-11-15 14:03:55 +00:00
|
|
|
* @sk: communication socket
|
|
|
|
*
|
2024-08-09 19:05:32 +00:00
|
|
|
* The socket tx_hook is called when the TX buffer is empty, i.e. all data was
|
2023-11-15 14:03:55 +00:00
|
|
|
* send. This function is used only when we found malformed PDU and we are
|
2024-08-09 19:05:32 +00:00
|
|
|
* resetting the established session. If called, we perform a SNMP protocol
|
|
|
|
* state change.
|
2023-11-15 14:03:55 +00:00
|
|
|
*/
|
|
|
|
static void
|
2024-01-10 11:21:46 +00:00
|
|
|
snmp_tx_skip(sock *sk)
|
2023-11-15 14:03:55 +00:00
|
|
|
{
|
|
|
|
struct snmp_proto *p = sk->data;
|
2024-08-13 15:50:09 +00:00
|
|
|
snmp_set_state(p, SNMP_STOP);
|
2023-11-15 14:03:55 +00:00
|
|
|
}
|
|
|
|
|
2024-08-15 13:52:15 +00:00
|
|
|
/*
|
|
|
|
* snmp_cleanup - free all resources allocated by SNMP protocol
|
|
|
|
* @p: SNMP protocol instance
|
|
|
|
*
|
|
|
|
* This function forcefully stops and cleans all resources and memory acqiured
|
|
|
|
* by given SNMP protocol instance, such as timers, lists, hash tables etc.
|
|
|
|
*/
|
|
|
|
static inline void
|
|
|
|
snmp_cleanup(struct snmp_proto *p)
|
|
|
|
{
|
|
|
|
/* Function tm_stop() is called inside rfree() */
|
|
|
|
rfree(p->startup_timer);
|
|
|
|
p->startup_timer = NULL;
|
|
|
|
|
|
|
|
rfree(p->ping_timer);
|
|
|
|
p->ping_timer = NULL;
|
|
|
|
|
|
|
|
rfree(p->sock);
|
|
|
|
p->sock = NULL;
|
|
|
|
|
|
|
|
rfree(p->lock);
|
|
|
|
p->lock = NULL;
|
|
|
|
|
|
|
|
struct snmp_registration *r, *r2;
|
|
|
|
WALK_LIST_DELSAFE(r, r2, p->registration_queue)
|
|
|
|
{
|
|
|
|
rem_node(&r->n);
|
|
|
|
mb_free(r);
|
|
|
|
r = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HASH_FREE(p->bgp_hash);
|
|
|
|
rfree(p->lp);
|
|
|
|
p->lp = NULL;
|
|
|
|
/* bgp_trie is allocated exclusively from linpool lp */
|
|
|
|
p->bgp_trie = NULL;
|
|
|
|
|
|
|
|
struct mib_walk_state *walk = tmp_alloc(sizeof(struct mib_walk_state));
|
|
|
|
mib_tree_walk_init(walk, p->mib_tree);
|
|
|
|
(void) mib_tree_delete(p->mib_tree, walk);
|
|
|
|
p->mib_tree = NULL;
|
|
|
|
|
|
|
|
p->state = SNMP_DOWN;
|
|
|
|
}
|
|
|
|
|
2023-11-15 14:03:55 +00:00
|
|
|
/*
|
|
|
|
* snmp_set_state - change state with associated actions
|
2024-08-09 19:05:32 +00:00
|
|
|
* @p: SNMP protocol instance
|
|
|
|
* @state: new SNMP protocol state
|
2024-01-23 23:23:31 +00:00
|
|
|
*
|
2024-08-09 19:05:32 +00:00
|
|
|
* This function does not notify the bird about protocol state. Return current
|
|
|
|
* protocol state (PS_UP, ...).
|
2023-11-15 14:03:55 +00:00
|
|
|
*/
|
2024-01-23 23:23:31 +00:00
|
|
|
int
|
2023-11-15 14:03:55 +00:00
|
|
|
snmp_set_state(struct snmp_proto *p, enum snmp_proto_state state)
|
|
|
|
{
|
|
|
|
enum snmp_proto_state last = p->state;
|
2024-07-22 16:17:35 +00:00
|
|
|
const struct snmp_config *cf = (struct snmp_config *) p->p.cf;
|
2023-11-15 14:03:55 +00:00
|
|
|
|
2024-01-23 23:23:31 +00:00
|
|
|
p->state = state;
|
|
|
|
|
2023-11-15 14:03:55 +00:00
|
|
|
switch (state)
|
|
|
|
{
|
|
|
|
case SNMP_INIT:
|
2024-08-15 08:54:13 +00:00
|
|
|
/* We intentionally do not log anything */
|
2023-11-15 14:03:55 +00:00
|
|
|
ASSERT(last == SNMP_DOWN);
|
2024-07-22 16:17:35 +00:00
|
|
|
|
2024-08-13 15:50:09 +00:00
|
|
|
proto_notify_state(&p->p, PS_START);
|
2024-07-22 16:17:35 +00:00
|
|
|
if (cf->trans_type == SNMP_TRANS_TCP)
|
|
|
|
{
|
|
|
|
/* We need to lock the IP address */
|
|
|
|
struct object_lock *lock;
|
|
|
|
lock = p->lock = olock_new(p->pool);
|
2024-08-16 08:50:46 +00:00
|
|
|
lock->addr = p->master_ip;
|
|
|
|
lock->port = p->master_port;
|
2024-07-22 16:17:35 +00:00
|
|
|
lock->type = OBJLOCK_TCP;
|
|
|
|
lock->hook = snmp_start_locked;
|
|
|
|
lock->data = p;
|
|
|
|
olock_acquire(lock);
|
|
|
|
return PS_START;
|
|
|
|
}
|
|
|
|
|
2024-08-09 19:05:32 +00:00
|
|
|
last = SNMP_INIT;
|
2024-07-22 16:17:35 +00:00
|
|
|
p->state = state = SNMP_LOCKED;
|
|
|
|
/* Fall thru */
|
2023-11-15 14:03:55 +00:00
|
|
|
|
|
|
|
case SNMP_LOCKED:
|
2024-08-15 08:54:13 +00:00
|
|
|
TRACE(D_EVENTS, "SNMP Address lock acquired");
|
2024-08-09 19:05:32 +00:00
|
|
|
ASSERT(last == SNMP_INIT);
|
2023-11-15 14:03:55 +00:00
|
|
|
sock *s = sk_new(p->pool);
|
2024-07-22 16:17:35 +00:00
|
|
|
|
|
|
|
if (cf->trans_type == SNMP_TRANS_TCP)
|
|
|
|
{
|
|
|
|
s->type = SK_TCP_ACTIVE;
|
2024-08-16 08:50:46 +00:00
|
|
|
s->daddr = p->master_ip;
|
|
|
|
s->dport = p->master_port;
|
2024-07-22 16:17:35 +00:00
|
|
|
s->rbsize = SNMP_RX_BUFFER_SIZE;
|
|
|
|
s->tbsize = SNMP_TX_BUFFER_SIZE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
s->type = SK_UNIX_ACTIVE;
|
2024-08-16 08:50:46 +00:00
|
|
|
s->host = cf->master_path; /* daddr */
|
2024-07-22 16:17:35 +00:00
|
|
|
s->rbsize = SNMP_RX_BUFFER_SIZE;
|
|
|
|
s->tbsize = SNMP_TX_BUFFER_SIZE;
|
|
|
|
}
|
2023-11-15 14:03:55 +00:00
|
|
|
|
|
|
|
s->tx_hook = snmp_connected;
|
|
|
|
s->err_hook = snmp_sock_err;
|
|
|
|
|
|
|
|
p->sock = s;
|
|
|
|
s->data = p;
|
|
|
|
|
|
|
|
/* Try opening the socket, schedule a retry on fail */
|
|
|
|
if (sk_open(s) < 0)
|
|
|
|
{
|
2024-08-15 08:54:13 +00:00
|
|
|
TRACE(D_EVENTS, "SNMP Opening of communication socket failed");
|
2023-11-15 14:03:55 +00:00
|
|
|
rfree(s);
|
|
|
|
p->sock = NULL;
|
2024-08-09 19:05:32 +00:00
|
|
|
// TODO handle 0 timeout
|
2023-11-15 14:03:55 +00:00
|
|
|
tm_start(p->startup_timer, p->timeout);
|
|
|
|
}
|
2024-01-23 23:23:31 +00:00
|
|
|
return PS_START;
|
2023-11-15 14:03:55 +00:00
|
|
|
|
|
|
|
case SNMP_OPEN:
|
2024-08-15 08:54:13 +00:00
|
|
|
TRACE(D_EVENTS, "SNMP Communication socket opened, starting AgentX subagent");
|
2023-11-15 14:03:55 +00:00
|
|
|
ASSERT(last == SNMP_LOCKED);
|
2024-08-09 19:05:32 +00:00
|
|
|
|
2023-11-15 14:03:55 +00:00
|
|
|
p->sock->rx_hook = snmp_rx;
|
|
|
|
p->sock->tx_hook = NULL;
|
2024-08-09 19:05:32 +00:00
|
|
|
|
2023-11-15 14:03:55 +00:00
|
|
|
snmp_start_subagent(p);
|
2024-08-09 19:05:32 +00:00
|
|
|
|
2024-07-22 16:17:35 +00:00
|
|
|
p->startup_timer->hook = snmp_stop_timeout;
|
|
|
|
tm_start(p->startup_timer, 1 S);
|
2024-01-23 23:23:31 +00:00
|
|
|
return PS_START;
|
2023-11-15 14:03:55 +00:00
|
|
|
|
|
|
|
case SNMP_REGISTER:
|
2024-08-15 08:54:13 +00:00
|
|
|
TRACE(D_EVENTS, "SNMP Registering MIBs");
|
2023-11-15 14:03:55 +00:00
|
|
|
ASSERT(last == SNMP_OPEN);
|
2024-08-09 19:05:32 +00:00
|
|
|
|
2024-07-22 16:17:35 +00:00
|
|
|
tm_stop(p->startup_timer); /* stop timeout */
|
2024-08-09 19:05:32 +00:00
|
|
|
|
|
|
|
p->sock->rx_hook = snmp_rx;
|
|
|
|
p->sock->tx_hook = snmp_tx;
|
|
|
|
|
2023-11-15 14:03:55 +00:00
|
|
|
snmp_register_mibs(p);
|
2024-08-15 09:52:38 +00:00
|
|
|
|
|
|
|
// TODO timer for CONN
|
|
|
|
|
2024-01-23 23:23:31 +00:00
|
|
|
return PS_START;
|
2023-11-15 14:03:55 +00:00
|
|
|
|
|
|
|
case SNMP_CONN:
|
2024-08-14 16:24:25 +00:00
|
|
|
TRACE(D_EVENTS, "MIBs registered, AgentX session established");
|
2023-11-15 14:03:55 +00:00
|
|
|
ASSERT(last == SNMP_REGISTER);
|
2024-08-13 15:50:09 +00:00
|
|
|
proto_notify_state(&p->p, PS_UP);
|
2024-01-23 23:23:31 +00:00
|
|
|
return PS_UP;
|
2023-11-15 14:03:55 +00:00
|
|
|
|
|
|
|
case SNMP_STOP:
|
2024-08-13 15:50:09 +00:00
|
|
|
if (p->sock && p->state != SNMP_OPEN && !sk_tx_buffer_empty(p->sock))
|
2024-08-09 19:05:32 +00:00
|
|
|
{
|
2024-08-15 08:54:13 +00:00
|
|
|
TRACE(D_EVENTS, "SNMP Closing AgentX session");
|
2024-08-09 19:05:32 +00:00
|
|
|
if (p->state == SNMP_OPEN || p->state == SNMP_REGISTER ||
|
|
|
|
p->state == SNMP_CONN)
|
|
|
|
snmp_stop_subagent(p);
|
2024-07-24 11:38:36 +00:00
|
|
|
|
2024-08-09 19:05:32 +00:00
|
|
|
p->sock->rx_hook = snmp_rx_skip;
|
|
|
|
p->sock->tx_hook = snmp_tx_skip;
|
|
|
|
|
|
|
|
p->startup_timer->hook = snmp_stop_timeout;
|
|
|
|
tm_start(p->startup_timer, 150 MS);
|
2024-08-13 15:50:09 +00:00
|
|
|
proto_notify_state(&p->p, PS_STOP);
|
2024-08-09 19:05:32 +00:00
|
|
|
return PS_STOP;
|
|
|
|
}
|
|
|
|
|
|
|
|
p->state = state = SNMP_DOWN;
|
|
|
|
/* Fall thru */
|
2023-11-15 14:03:55 +00:00
|
|
|
|
|
|
|
case SNMP_DOWN:
|
2024-08-15 08:54:13 +00:00
|
|
|
TRACE(D_EVENTS, "SNMP AgentX session closed");
|
2023-11-15 14:03:55 +00:00
|
|
|
snmp_cleanup(p);
|
2024-08-13 15:50:09 +00:00
|
|
|
proto_notify_state(&p->p, PS_DOWN);
|
2024-01-23 23:23:31 +00:00
|
|
|
return PS_DOWN;
|
2023-11-15 14:03:55 +00:00
|
|
|
|
|
|
|
default:
|
2024-08-14 16:24:25 +00:00
|
|
|
die("unknown SNMP state transition");
|
2024-01-23 23:23:31 +00:00
|
|
|
return PS_DOWN;
|
2023-11-15 14:03:55 +00:00
|
|
|
}
|
|
|
|
}
|
2022-11-29 15:30:20 +00:00
|
|
|
|
2023-11-08 09:55:42 +00:00
|
|
|
/*
|
2024-07-24 11:38:36 +00:00
|
|
|
* snmp_reset - reset AgentX session
|
|
|
|
* @p: SNMP protocol instance
|
2023-11-08 09:55:42 +00:00
|
|
|
*
|
2024-07-24 11:38:36 +00:00
|
|
|
* We wait until the last PDU written into the socket is send while ignoring all
|
|
|
|
* incomming PDUs. Then we hard reset the connection by socket closure. The
|
|
|
|
* protocol instance is automatically restarted by nest.
|
|
|
|
*
|
2024-08-09 19:05:32 +00:00
|
|
|
* Return protocol state (PS_STOP, ...).
|
2024-07-24 11:38:36 +00:00
|
|
|
*/
|
2024-08-09 19:05:32 +00:00
|
|
|
int
|
|
|
|
snmp_reset(struct snmp_proto *p)
|
2024-07-24 11:38:36 +00:00
|
|
|
{
|
2024-08-13 15:50:09 +00:00
|
|
|
return snmp_set_state(p, SNMP_STOP);
|
2022-12-17 17:16:19 +00:00
|
|
|
}
|
|
|
|
|
2024-08-13 15:50:09 +00:00
|
|
|
/*
|
|
|
|
* snmp_up - AgentX session has registered all MIBs, protocols is up
|
|
|
|
* @p: SNMP protocol instance
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
snmp_up(struct snmp_proto *p)
|
|
|
|
{
|
|
|
|
if (p->state == SNMP_REGISTER)
|
|
|
|
snmp_set_state(p, SNMP_CONN);
|
|
|
|
}
|
2024-07-24 11:38:36 +00:00
|
|
|
|
2023-11-08 09:55:42 +00:00
|
|
|
/*
|
2024-08-15 13:52:15 +00:00
|
|
|
* snmp_shutdown - Forcefully stop the SNMP protocol instance
|
|
|
|
* @P: SNMP protocol generic handle
|
2023-11-08 09:55:42 +00:00
|
|
|
*
|
2024-08-15 13:52:15 +00:00
|
|
|
* Simple cast-like wrapper around snmp_reset(), see more info there.
|
2023-11-08 09:55:42 +00:00
|
|
|
*/
|
2024-08-15 13:52:15 +00:00
|
|
|
static int
|
|
|
|
snmp_shutdown(struct proto *P)
|
2023-09-04 11:48:28 +00:00
|
|
|
{
|
2024-08-15 13:52:15 +00:00
|
|
|
struct snmp_proto *p = SKIP_BACK(struct snmp_proto, p, P);
|
|
|
|
return snmp_reset(p);
|
2022-08-02 14:04:25 +00:00
|
|
|
}
|
|
|
|
|
2023-11-08 09:55:42 +00:00
|
|
|
/*
|
2024-08-15 13:52:15 +00:00
|
|
|
* snmp_show_proto_info - print basic information about SNMP protocol instance
|
2024-08-09 19:05:32 +00:00
|
|
|
* @P: SNMP protocol generic handle
|
2023-11-08 09:55:42 +00:00
|
|
|
*/
|
2024-08-15 13:52:15 +00:00
|
|
|
static void
|
|
|
|
snmp_show_proto_info(struct proto *P)
|
2022-08-01 11:01:49 +00:00
|
|
|
{
|
2022-08-10 15:31:32 +00:00
|
|
|
struct snmp_proto *p = (void *) P;
|
2022-09-30 07:36:09 +00:00
|
|
|
|
2024-08-15 13:52:15 +00:00
|
|
|
cli_msg(-1006, " SNMP state: %s", snmp_state_str[p->state]);
|
|
|
|
cli_msg(-1006, " MIBs");
|
2022-09-06 16:04:29 +00:00
|
|
|
|
2024-08-15 13:52:15 +00:00
|
|
|
snmp_bgp4_show_info(p);
|
2022-08-01 11:01:49 +00:00
|
|
|
}
|
|
|
|
|
2024-08-14 14:16:10 +00:00
|
|
|
/*
|
|
|
|
* snmp_reconfigure_logic - find changes in configuration
|
|
|
|
* @p: SNMP protocol instance
|
|
|
|
* @new: new SNMP protocol configuration
|
|
|
|
*
|
|
|
|
* Return 1 if only minor changes have occured, 0 if we need full down-up cycle.
|
|
|
|
*/
|
2024-01-10 11:21:46 +00:00
|
|
|
static inline int
|
|
|
|
snmp_reconfigure_logic(struct snmp_proto *p, const struct snmp_config *new)
|
2022-08-01 11:01:49 +00:00
|
|
|
{
|
2023-09-04 11:58:59 +00:00
|
|
|
const struct snmp_config *old = SKIP_BACK(struct snmp_config, cf, p->p.cf);
|
2022-08-10 15:31:32 +00:00
|
|
|
|
2024-08-14 14:16:10 +00:00
|
|
|
if ((old->trans_type != SNMP_TRANS_TCP) && (new->trans_type == SNMP_TRANS_TCP)
|
|
|
|
|| (old->trans_type == SNMP_TRANS_TCP) && (new->trans_type != SNMP_TRANS_TCP))
|
2024-01-10 11:21:46 +00:00
|
|
|
return 0;
|
|
|
|
|
2024-08-14 14:16:10 +00:00
|
|
|
if (old->trans_type == SNMP_TRANS_TCP &&
|
2024-08-16 08:50:46 +00:00
|
|
|
(ipa_compare(old->master_ip, new->master_ip)
|
|
|
|
|| old->master_port != new->master_port))
|
2023-09-11 11:06:20 +00:00
|
|
|
return 0;
|
|
|
|
|
2024-08-14 14:16:10 +00:00
|
|
|
if (old->trans_type != SNMP_TRANS_TCP &&
|
2024-08-16 08:50:46 +00:00
|
|
|
bstrcmp(old->master_path, new->master_path))
|
2024-01-10 11:21:46 +00:00
|
|
|
return 0;
|
|
|
|
|
2024-08-16 08:50:46 +00:00
|
|
|
return (old->bgp4_local_id != new->bgp4_local_id
|
|
|
|
|| old->bgp4_local_as != new->bgp4_local_as
|
2024-08-14 14:16:10 +00:00
|
|
|
|| old->timeout != new->timeout // TODO distinguish message timemout
|
|
|
|
//(Open.timeout and timeout for timer)
|
2024-08-09 19:05:32 +00:00
|
|
|
|| old->priority != new->priority
|
2024-08-14 16:24:25 +00:00
|
|
|
|| strncmp(old->description, new->description, UINT16_MAX - 1));
|
2022-08-01 11:01:49 +00:00
|
|
|
}
|
2022-09-30 07:36:09 +00:00
|
|
|
|
2023-11-08 09:55:42 +00:00
|
|
|
/*
|
2024-04-22 10:55:33 +00:00
|
|
|
* snmp_reconfigure - Indicate instance reconfigurability
|
2024-01-10 11:21:46 +00:00
|
|
|
* @P - SNMP protocol generic handle, current state
|
|
|
|
* @CF - SNMP protocol configuration generic handle carring new values
|
|
|
|
*
|
|
|
|
* We accept the reconfiguration if the new configuration @CF is identical with
|
2024-01-23 23:23:31 +00:00
|
|
|
* the currently deployed configuration. Otherwise we deny reconfiguration because
|
2024-01-10 11:21:46 +00:00
|
|
|
* the implementation would be cumbersome.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
snmp_reconfigure(struct proto *P, struct proto_config *CF)
|
|
|
|
{
|
|
|
|
struct snmp_proto *p = SKIP_BACK(struct snmp_proto, p, P);
|
|
|
|
const struct snmp_config *new = SKIP_BACK(struct snmp_config, cf, CF);
|
|
|
|
|
2024-01-23 23:23:31 +00:00
|
|
|
/* We are searching for configuration changes */
|
2024-08-14 14:16:10 +00:00
|
|
|
int reconfigurable = snmp_reconfigure_logic(p, new);
|
2024-01-10 11:21:46 +00:00
|
|
|
|
2024-08-14 14:16:10 +00:00
|
|
|
if (reconfigurable)
|
2024-01-23 23:23:31 +00:00
|
|
|
{
|
2024-08-14 14:16:10 +00:00
|
|
|
/* copy possibly changed values */
|
|
|
|
p->startup_delay = new->startup_delay;
|
2024-08-14 16:24:25 +00:00
|
|
|
p->verbose = new->verbose;
|
2024-08-14 14:16:10 +00:00
|
|
|
|
|
|
|
ASSERT(p->ping_timer);
|
|
|
|
int active = tm_active(p->ping_timer);
|
|
|
|
rfree(p->ping_timer);
|
|
|
|
p->ping_timer = tm_new_init(p->pool, snmp_ping_timeout, p, p->timeout, 0);
|
|
|
|
|
|
|
|
if (active)
|
|
|
|
tm_start(p->ping_timer, p->timeout);
|
|
|
|
|
|
|
|
HASH_FREE(p->bgp_hash);
|
2024-01-10 11:21:46 +00:00
|
|
|
HASH_INIT(p->bgp_hash, p->pool, 10);
|
2024-08-14 14:16:10 +00:00
|
|
|
|
|
|
|
rfree(p->lp);
|
|
|
|
p->lp = lp_new(p->pool);
|
|
|
|
p->bgp_trie = f_new_trie(p->lp, 0);
|
|
|
|
|
|
|
|
/* We repopulate BGP related data structures (bgp_hash, bgp_trie). */
|
|
|
|
snmp_bgp4_start(p, 0);
|
2024-01-10 11:21:46 +00:00
|
|
|
}
|
|
|
|
|
2024-08-14 14:16:10 +00:00
|
|
|
return reconfigurable;
|
2024-01-10 11:21:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2024-08-15 13:52:15 +00:00
|
|
|
* snmp_start - Initialize the SNMP protocol instance
|
2024-01-10 11:21:46 +00:00
|
|
|
* @P: SNMP protocol generic handle
|
2024-08-15 13:52:15 +00:00
|
|
|
*
|
|
|
|
* The first step in AgentX subagent startup is protocol initialition.
|
|
|
|
* We must prepare lists, find BGP peers and finally asynchronously start
|
|
|
|
* a AgentX subagent session.
|
2023-11-08 09:55:42 +00:00
|
|
|
*/
|
2024-08-15 13:52:15 +00:00
|
|
|
static int
|
|
|
|
snmp_start(struct proto *P)
|
2022-08-01 11:01:49 +00:00
|
|
|
{
|
2023-11-15 14:03:55 +00:00
|
|
|
struct snmp_proto *p = (void *) P;
|
2024-08-15 13:52:15 +00:00
|
|
|
struct snmp_config *cf = (struct snmp_config *) P->cf;
|
2022-08-02 14:04:25 +00:00
|
|
|
|
2024-08-15 13:52:15 +00:00
|
|
|
p->local_ip = cf->local_ip;
|
2024-08-16 08:50:46 +00:00
|
|
|
p->master_ip = cf->master_ip;
|
|
|
|
p->master_port = cf->master_port;
|
|
|
|
p->bgp4_local_as = cf->bgp4_local_as;
|
|
|
|
p->bgp4_local_id = cf->bgp4_local_id;
|
2024-08-15 13:52:15 +00:00
|
|
|
p->timeout = cf->timeout;
|
|
|
|
p->startup_delay = cf->startup_delay;
|
|
|
|
p->verbose = cf->verbose;
|
2022-08-02 14:04:25 +00:00
|
|
|
|
2024-08-15 13:52:15 +00:00
|
|
|
p->pool = p->p.pool;
|
|
|
|
p->lp = lp_new(p->pool);
|
|
|
|
p->bgp_trie = f_new_trie(p->lp, 0);
|
|
|
|
p->mib_tree = mb_alloc(p->pool, sizeof(struct mib_tree));
|
|
|
|
|
|
|
|
p->startup_timer = tm_new_init(p->pool, snmp_startup_timeout, p, 0, 0);
|
|
|
|
p->ping_timer = tm_new_init(p->pool, snmp_ping_timeout, p, p->timeout, 0);
|
|
|
|
|
|
|
|
init_list(&p->registration_queue);
|
|
|
|
|
|
|
|
/* We create copy of bonds to BGP protocols. */
|
|
|
|
HASH_INIT(p->bgp_hash, p->pool, 10);
|
|
|
|
|
|
|
|
mib_tree_init(p->pool, p->mib_tree);
|
|
|
|
snmp_bgp4_start(p, 1);
|
|
|
|
|
|
|
|
return snmp_set_state(p, SNMP_INIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* snmp_init - preinitialize SNMP instance
|
|
|
|
* @CF: SNMP configuration generic handle
|
|
|
|
*
|
|
|
|
* Returns a generic handle pointing to preinitialized SNMP procotol
|
|
|
|
* instance.
|
|
|
|
*/
|
|
|
|
static struct proto *
|
|
|
|
snmp_init(struct proto_config *CF)
|
|
|
|
{
|
|
|
|
struct proto *P = proto_new(CF);
|
|
|
|
struct snmp_proto *p = SKIP_BACK(struct snmp_proto, p, P);
|
|
|
|
|
|
|
|
p->rl_gen = (struct tbf) TBF_DEFAULT_LOG_LIMITS;
|
|
|
|
p->state = SNMP_DOWN;
|
|
|
|
|
|
|
|
return P;
|
2022-08-01 11:01:49 +00:00
|
|
|
}
|
|
|
|
|
2023-11-08 09:55:42 +00:00
|
|
|
/*
|
|
|
|
* snmp_postconfig - Check configuration correctness
|
2024-01-10 11:21:46 +00:00
|
|
|
* @CF: SNMP procotol configuration generic handle
|
2023-11-08 09:55:42 +00:00
|
|
|
*/
|
2022-09-20 12:28:57 +00:00
|
|
|
static void
|
|
|
|
snmp_postconfig(struct proto_config *CF)
|
|
|
|
{
|
2024-07-22 16:17:35 +00:00
|
|
|
const struct snmp_config *cf = (struct snmp_config *) CF;
|
|
|
|
|
2023-09-04 12:01:08 +00:00
|
|
|
/* Walk the BGP protocols and cache their references. */
|
2024-08-16 08:50:46 +00:00
|
|
|
if (cf->bgp4_local_as == 0)
|
2022-09-20 12:28:57 +00:00
|
|
|
cf_error("local as not specified");
|
|
|
|
}
|
|
|
|
|
2023-11-08 09:55:42 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Protocol infrastructure
|
|
|
|
*/
|
|
|
|
|
2022-08-01 11:01:49 +00:00
|
|
|
struct protocol proto_snmp = {
|
2023-11-08 12:10:13 +00:00
|
|
|
.name = "SNMP",
|
2022-08-01 11:01:49 +00:00
|
|
|
.template = "snmp%d",
|
2023-11-08 12:10:13 +00:00
|
|
|
.channel_mask = 0,
|
2022-08-01 11:01:49 +00:00
|
|
|
.proto_size = sizeof(struct snmp_proto),
|
|
|
|
.config_size = sizeof(struct snmp_config),
|
2022-09-20 12:28:57 +00:00
|
|
|
.postconfig = snmp_postconfig,
|
2022-08-01 11:01:49 +00:00
|
|
|
.init = snmp_init,
|
|
|
|
.start = snmp_start,
|
|
|
|
.reconfigure = snmp_reconfigure,
|
|
|
|
.show_proto_info = snmp_show_proto_info,
|
2024-08-15 13:52:15 +00:00
|
|
|
.shutdown = snmp_shutdown,
|
2022-08-01 11:01:49 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
void
|
2022-08-10 15:31:32 +00:00
|
|
|
snmp_build(void)
|
2022-08-01 11:01:49 +00:00
|
|
|
{
|
|
|
|
proto_build(&proto_snmp);
|
|
|
|
}
|
2023-09-04 07:25:51 +00:00
|
|
|
|