0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2025-01-25 10:21:53 +00:00
bird/proto/snmp/snmp.c

476 lines
11 KiB
C
Raw Normal View History

2022-08-01 13:01:49 +02:00
/*
* BIRD -- Simple Network Management Protocol (SNMP)
*
* (c) 2022 Vojtech Vilimek <vojtech.vilimek@nic.cz>
* (c) 2022 CZ.NIC z.s.p.o.
*
* Can be freely distributed and used under the terms of the GNU GPL.
*/
#include "nest/bird.h"
#include "nest/cli.h"
2022-08-10 17:31:32 +02:00
#include "nest/locks.h"
#include "lib/socket.h"
2022-09-20 14:28:57 +02:00
#include "lib/lists.h"
2022-08-01 13:01:49 +02:00
2022-08-10 17:31:32 +02:00
#include "snmp.h"
#include "subagent.h"
2023-07-26 14:34:01 +02:00
#include "snmp_utils.h"
2022-08-10 17:31:32 +02:00
static void snmp_connected(sock *sk);
static void snmp_sock_err(sock *sk, int err);
static void snmp_ping_timer(struct timer *tm);
2022-09-30 09:36:09 +02:00
static void snmp_startup(struct snmp_proto *p);
static void snmp_startup_timeout(timer *t);
static void snmp_start_locked(struct object_lock *lock);
2023-07-26 14:34:01 +02:00
static int snmp_shutdown(struct proto *P);
2022-11-29 16:30:20 +01:00
static const char * const snmp_state[] = {
2022-12-17 18:16:19 +01:00
[SNMP_ERR] = "SNMP ERROR",
[SNMP_DELAY] = "SNMP DELAY",
[SNMP_INIT] = "SNMP INIT",
[SNMP_REGISTER] = "SNMP REGISTERING",
[SNMP_CONN] = "SNMP CONNECTED",
[SNMP_STOP] = "SNMP STOP",
[SNMP_DOWN] = "SNMP DOWN",
[SNMP_LISTEN] = "SNMP LISTEN",
2022-11-29 16:30:20 +01:00
};
2022-08-01 13:01:49 +02:00
static struct proto *
snmp_init(struct proto_config *CF)
{
struct proto *P = proto_new(CF);
2022-08-10 17:31:32 +02:00
struct snmp_proto *p = SKIP_BACK(struct snmp_proto, p, P);
2022-12-10 13:23:50 +01:00
const struct snmp_config *cf = SKIP_BACK(struct snmp_config, cf, CF);
2022-08-01 13:01:49 +02:00
p->rl_gen = (struct tbf) TBF_DEFAULT_LOG_LIMITS;
2022-08-10 17:31:32 +02:00
p->local_ip = cf->local_ip;
p->remote_ip = cf->remote_ip;
p->local_port = cf->local_port;
p->remote_port = cf->remote_port;
2023-03-14 14:09:45 +01:00
p->local_as = cf->local_as;
2023-07-26 14:34:01 +02:00
snmp_log("changing proto_snmp state to INIT");
2022-09-06 18:04:29 +02:00
p->state = SNMP_INIT;
2022-08-10 17:31:32 +02:00
// p->timeout = cf->timeout;
p->timeout = 15;
2022-12-10 13:22:37 +01:00
snmp_log("snmp_reconfigure() lip: %I:%u rip: %I:%u",
2022-08-10 17:31:32 +02:00
cf->local_ip, cf->local_port, cf->remote_ip, cf->remote_port);
2022-08-01 13:01:49 +02:00
return P;
}
2022-12-17 18:16:19 +01:00
static inline void
snmp_cleanup(struct snmp_proto *p)
2022-12-10 13:23:50 +01:00
{
2023-07-26 14:34:01 +02:00
struct additional_buffer *b;
WALK_LIST(b, p->additional_buffers)
{
mb_free(b->buf);
rem_node(&b->n);
mb_free(b);
}
init_list(&p->additional_buffers);
2022-12-17 18:16:19 +01:00
rfree(p->startup_timer);
rfree(p->ping_timer);
2022-12-10 13:23:50 +01:00
if (p->sock != NULL)
2022-12-17 18:16:19 +01:00
rfree(p->sock);
2023-07-26 14:34:01 +02:00
p->sock = NULL;
2022-12-10 13:23:50 +01:00
if (p->lock != NULL)
rfree(p->lock);
2023-07-26 14:34:01 +02:00
p->lock = NULL;
2022-12-10 13:23:50 +01:00
2022-12-17 18:16:19 +01:00
p->state = SNMP_DOWN;
}
void
snmp_down(struct snmp_proto *p)
{
snmp_cleanup(p);
2022-12-10 13:23:50 +01:00
proto_notify_state(&p->p, PS_DOWN);
}
2022-09-30 09:36:09 +02:00
static void
snmp_startup_timeout(timer *t)
{
2022-12-10 13:22:37 +01:00
snmp_log("startup timer triggered");
2022-09-30 09:36:09 +02:00
snmp_startup(t->data);
}
2022-12-17 18:16:19 +01:00
static void
snmp_stop_timeout(timer *t)
{
snmp_log("stop timer triggered");
struct snmp_proto *p = t->data;
snmp_down(p);
}
2022-09-30 09:36:09 +02:00
static void
snmp_startup(struct snmp_proto *p)
{
2022-12-10 13:23:50 +01:00
//snmp_log("changing proto_snmp state to INIT");
if (p->state == SNMP_CONN ||
2022-12-17 18:16:19 +01:00
p->state == SNMP_REGISTER)
2022-12-10 13:23:50 +01:00
{
snmp_log("startup() with invalid state %u", p->state);
return;
}
snmp_log("snmp_startup()");
2022-12-06 16:32:26 +01:00
p->state = SNMP_INIT;
2022-09-30 09:36:09 +02:00
/* starting agentX communicaiton channel */
2022-12-06 16:32:26 +01:00
2022-12-10 13:22:37 +01:00
snmp_log("preparing lock");
2022-09-30 09:36:09 +02:00
struct object_lock *lock;
2022-11-29 16:30:20 +01:00
/* we could have the lock already acquired but be in ERROR state */
2022-09-30 09:36:09 +02:00
lock = p->lock = olock_new(p->p.pool);
2022-12-10 13:23:50 +01:00
// lock->addr
// lock->port
// lock->iface
// lock->vrf
2022-09-30 09:36:09 +02:00
lock->type = OBJLOCK_TCP;
lock->hook = snmp_start_locked;
lock->data = p;
2022-12-10 13:22:37 +01:00
snmp_log("lock acquiring");
2022-09-30 09:36:09 +02:00
olock_acquire(lock);
2022-12-10 13:22:37 +01:00
/*
snmp_log("local ip: %I:%u, remote ip: %I:%u",
2022-09-30 09:36:09 +02:00
p->local_ip, p->local_port, p->remote_ip, p->remote_port);
2022-12-10 13:22:37 +01:00
*/
2022-09-30 09:36:09 +02:00
}
2022-08-10 17:31:32 +02:00
static void
snmp_start_locked(struct object_lock *lock)
{
2022-12-17 18:16:19 +01:00
snmp_log("snmp_start_locked() - lock acquired; preparing socket");
2022-08-10 17:31:32 +02:00
struct snmp_proto *p = lock->data;
sock *s = sk_new(p->p.pool);
s->type = SK_TCP_ACTIVE;
s->saddr = p->local_ip;
s->daddr = p->remote_ip;
s->dport = p->remote_port;
s->rbsize = SNMP_RX_BUFFER_SIZE;
s->tbsize = SNMP_TX_BUFFER_SIZE;
2022-09-30 09:36:09 +02:00
2022-09-06 18:04:29 +02:00
//s->tos = IP_PREC_INTERNET_CONTROL
2022-08-10 17:31:32 +02:00
//s->rx_hook = snmp_connected;
s->tx_hook = snmp_connected;
s->err_hook = snmp_sock_err;
p->sock = s;
s->data = p;
2022-09-30 09:36:09 +02:00
p->to_send = 0;
p->errs = 0;
2022-12-10 13:23:50 +01:00
// snmp_startup(p);
2022-09-30 09:36:09 +02:00
if (sk_open(s) < 0)
2022-12-10 13:23:50 +01:00
{
2022-08-10 17:31:32 +02:00
log(L_ERR "Cannot open listening socket");
2022-12-10 13:23:50 +01:00
snmp_down(p);
}
2022-09-30 09:36:09 +02:00
2022-12-10 13:22:37 +01:00
snmp_log("socket ready!, trying to connect");
2022-08-10 17:31:32 +02:00
}
static void
snmp_connected(sock *sk)
{
struct snmp_proto *p = sk->data;
2022-12-10 13:22:37 +01:00
snmp_log("snmp_connected() connection created");
2023-07-26 14:34:01 +02:00
byte *buf UNUSED = sk->rpos;
uint size = sk->rbuf + sk->rbsize - sk->rpos;
snmp_dump_packet(buf, size);
2022-08-10 17:31:32 +02:00
sk->rx_hook = snmp_rx;
sk->tx_hook = snmp_tx;
snmp_start_subagent(p);
2022-12-10 13:23:50 +01:00
// TODO ping interval
tm_set(p->ping_timer, 15 S);
2022-08-10 17:31:32 +02:00
}
2022-08-10 17:31:32 +02:00
static void
2022-09-30 09:36:09 +02:00
snmp_sock_err(sock *sk, int err)
2022-08-10 17:31:32 +02:00
{
2022-12-10 13:22:37 +01:00
snmp_log("snmp_sock_err() %s - err no: %d", strerror(err), err);
2022-09-20 14:28:57 +02:00
struct snmp_proto *p = sk->data;
2023-07-26 14:34:01 +02:00
p->errs++;
2022-09-30 09:36:09 +02:00
tm_stop(p->ping_timer);
2022-09-20 14:28:57 +02:00
2022-09-30 09:36:09 +02:00
rfree(p->sock);
p->sock = NULL;
2022-09-20 14:28:57 +02:00
2022-11-29 16:30:20 +01:00
rfree(p->lock);
p->lock = NULL;
2022-12-10 13:22:37 +01:00
snmp_log("changing proto_snmp state to ERR[OR]");
2023-07-26 14:34:01 +02:00
if (err)
p->state = SNMP_ERR;
else
{
snmp_shutdown((struct proto *) p);
return;
}
2022-12-17 18:16:19 +01:00
2022-12-10 13:23:50 +01:00
// TODO ping interval
2023-07-26 14:34:01 +02:00
tm_start(p->startup_timer, 4 S);
}
2022-08-01 13:01:49 +02:00
static int
snmp_start(struct proto *P)
{
2022-12-10 13:22:37 +01:00
snmp_log("snmp_start() - starting timer (almost)");
2022-08-10 17:31:32 +02:00
struct snmp_proto *p = (void *) P;
2022-09-20 14:28:57 +02:00
struct snmp_config *cf = (struct snmp_config *) P->cf;
2022-08-10 17:31:32 +02:00
2022-09-30 09:36:09 +02:00
p->startup_timer = tm_new_init(p->p.pool, snmp_startup_timeout, p, 0, 0);
p->to_send = 0;
p->errs = 0;
p->pool = lp_new(p->p.pool);
p->bgp_trie = f_new_trie(p->pool, cf->bonds);
2022-12-10 18:08:00 +01:00
init_list(&p->register_queue);
init_list(&p->bgp_registered);
2022-08-10 17:31:32 +02:00
p->ping_timer = tm_new_init(p->p.pool, snmp_ping_timer, p, 0, 0);
2022-12-10 13:23:50 +01:00
// tm_set(p->ping_timer, current_time() + 2 S);
2022-08-10 17:31:32 +02:00
2022-12-10 13:23:50 +01:00
/* remove duplicate lock acquiring code */
#if 0
2022-08-10 17:31:32 +02:00
/* starting agentX communicaiton channel */
2022-12-10 13:22:37 +01:00
snmp_log("preparing lock");
2022-08-10 17:31:32 +02:00
struct object_lock *lock;
2022-09-20 14:28:57 +02:00
lock = p->lock = olock_new(p->p.pool);
2022-08-10 17:31:32 +02:00
lock->type = OBJLOCK_TCP;
lock->hook = snmp_start_locked;
lock->data = p;
olock_acquire(lock);
2022-12-10 13:22:37 +01:00
snmp_log("lock acquired");
2022-08-10 17:31:32 +02:00
2022-12-10 13:22:37 +01:00
snmp_log("local ip: %I:%u, remote ip: %I:%u",
2022-08-10 17:31:32 +02:00
p->local_ip, p->local_port, p->remote_ip, p->remote_port);
2022-09-06 18:04:29 +02:00
2022-12-10 13:23:50 +01:00
#endif
2022-09-06 18:04:29 +02:00
/* create copy of bonds to bgp */
2022-09-20 14:28:57 +02:00
HASH_INIT(p->bgp_hash, p->p.pool, 10);
2022-09-06 18:04:29 +02:00
2022-09-30 09:36:09 +02:00
struct snmp_bond *b;
2022-09-06 18:04:29 +02:00
WALK_LIST(b, cf->bgp_entries)
{
2022-09-20 14:28:57 +02:00
struct bgp_config *bc = (struct bgp_config *) b->proto;
2022-09-06 18:04:29 +02:00
if (bc && !ipa_zero(bc->remote_ip))
{
2022-09-20 14:28:57 +02:00
struct snmp_bgp_peer *peer =
mb_allocz(p->p.pool, sizeof(struct snmp_bgp_peer));
2022-09-30 09:36:09 +02:00
peer->config = (struct bgp_config *) b->proto;
2022-09-20 14:28:57 +02:00
peer->peer_ip = bc->remote_ip;
2022-09-06 18:04:29 +02:00
2023-07-26 14:34:01 +02:00
struct net_addr net;
net_fill_ip4(&net, ipa_to_ip4(peer->peer_ip), IP4_MAX_PREFIX_LENGTH);
2022-09-30 09:36:09 +02:00
2023-07-26 14:34:01 +02:00
trie_add_prefix(p->bgp_trie, &net, IP4_MAX_PREFIX_LENGTH, IP4_MAX_PREFIX_LENGTH);
2022-09-30 09:36:09 +02:00
2022-09-20 14:28:57 +02:00
HASH_INSERT(p->bgp_hash, SNMP_HASH, peer);
2022-09-06 18:04:29 +02:00
}
}
2022-09-30 09:36:09 +02:00
2023-07-26 14:34:01 +02:00
init_list(&p->additional_buffers);
2022-12-10 13:23:50 +01:00
snmp_startup(p);
2022-09-30 09:36:09 +02:00
return PS_START;
2022-08-01 13:01:49 +02:00
}
static int
snmp_reconfigure(struct proto *P, struct proto_config *CF)
{
2022-08-10 17:31:32 +02:00
struct snmp_proto *p = SKIP_BACK(struct snmp_proto, p, P);
2022-12-10 13:23:50 +01:00
const struct snmp_config *cf = SKIP_BACK(struct snmp_config, cf, CF);
2022-08-10 17:31:32 +02:00
p->local_ip = cf->local_ip;
p->remote_ip = cf->remote_ip;
p->local_port = cf->local_port;
p->remote_port = cf->remote_port;
2022-09-20 14:28:57 +02:00
p->local_as = cf->local_as;
2022-08-10 17:31:32 +02:00
p->timeout = 15;
2022-12-10 18:08:00 +01:00
/* workaround to make the registration happen */
p->register_to_ack = 1;
2022-09-06 18:04:29 +02:00
/* TODO walk all bind protocols and find their (new) IP
to update HASH table */
2022-12-10 13:22:37 +01:00
snmp_log("snmp_reconfigure() lip: %I:%u rip: %I:%u",
2022-08-10 17:31:32 +02:00
p->local_ip, p->local_port, p->remote_ip, p->remote_port);
2022-12-10 13:22:37 +01:00
2022-09-30 09:36:09 +02:00
return 1;
2022-08-01 13:01:49 +02:00
}
2022-09-30 09:36:09 +02:00
2023-07-26 14:34:01 +02:00
static void
snmp_show_proto_info(struct proto *P)
2022-08-01 13:01:49 +02:00
{
2022-11-29 16:30:20 +01:00
struct snmp_proto *sp = (void *) P;
struct snmp_config *c = (void *) P->cf;
2022-08-01 13:01:49 +02:00
2022-11-29 16:30:20 +01:00
cli_msg(-1006, "");
cli_msg(-1006, " snmp status %s", snmp_state[sp->state]);
cli_msg(-1006, "");
cli_msg(-1006, " BGP peers");
struct snmp_bond *bond;
WALK_LIST(bond, c->bgp_entries)
2022-08-01 13:01:49 +02:00
{
struct proto_config *cf = P->cf;
struct bgp_config *bcf = (struct bgp_config *) cf;
struct proto *p = cf->proto;
struct bgp_proto *bp = (struct bgp_proto *) cf->proto;
struct bgp_conn *conn = bp->conn;
cli_msg(-1006, " name: %s", cf->name);
cli_msg(-1006, "");
cli_msg(-1006, " rem. identifier: %u", bp->remote_id);
// learn more !!
2022-12-06 16:32:26 +01:00
cli_msg(-1006, " admin status: %s", (p->disabled) ? "stop" :
"start");
// version ?
2022-09-20 14:28:57 +02:00
cli_msg(-1006, " version: 4");
cli_msg(-1006, " local ip: %u", bcf->local_ip);
cli_msg(-1006, " remote ip: %u", bcf->remote_ip);
cli_msg(-1006, " local port: %u", bcf->local_port);
cli_msg(-1006, " remote port: %u", bcf->remote_port);
2022-08-10 17:31:32 +02:00
/*
if (conn) {
cli_msg(-1006, " state: %u", conn->state);
cli_msg(-1006, " remote as: %u", conn->remote_caps->as4_number);
}
2022-08-10 17:31:32 +02:00
*/
cli_msg(-1006, " in updates: %u", bp->stats.rx_updates);
cli_msg(-1006, " out updates: %u", bp->stats.tx_updates);
cli_msg(-1006, " in total: %u", bp->stats.rx_messages);
cli_msg(-1006, " out total: %u", bp->stats.tx_messages);
cli_msg(-1006, " fsm transitions: %u",
bp->stats.fsm_established_transitions);
// not supported yet
cli_msg(-1006, " fsm total time: --");
cli_msg(-1006, " retry interval: %u", bcf->connect_retry_time);
2022-08-10 17:31:32 +02:00
/*
if (conn) {
cli_msg(-1006, " hold time: %u", conn->hold_time);
cli_msg(-1006, " keep alive: %u", conn->keepalive_time );
2022-08-01 13:01:49 +02:00
}
2022-08-10 17:31:32 +02:00
*/
cli_msg(-1006, " hold configurated: %u", bcf->hold_time );
cli_msg(-1006, " keep alive config: %u", bcf->keepalive_time );
// unknown
cli_msg(-1006, " min AS origin. int.: --");
cli_msg(-1006, " min route advertisement: %u", 0 );
cli_msg(-1006, " in update elapsed time: %u", 0 );
if (!conn)
cli_msg(-1006, " no default connection");
cli_msg(-1006, " outgoinin_conn state %u", bp->outgoing_conn.state + 1);
cli_msg(-1006, " incoming_conn state: %u", bp->incoming_conn.state + 1);
2022-08-01 13:01:49 +02:00
}
}
2022-09-20 14:28:57 +02:00
static void
snmp_postconfig(struct proto_config *CF)
{
2023-07-26 14:34:01 +02:00
// walk the bgp protocols and cache their references
2022-09-20 14:28:57 +02:00
if (((struct snmp_config *) CF)->local_as == 0)
cf_error("local as not specified");
}
2022-08-10 17:31:32 +02:00
static void
snmp_ping_timer(struct timer *tm)
2022-08-01 13:01:49 +02:00
{
2022-12-10 18:08:00 +01:00
// snmp_log("snmp_ping_timer() ");
2022-09-30 09:36:09 +02:00
struct snmp_proto *p = tm->data;
2022-08-10 17:31:32 +02:00
2022-09-20 14:28:57 +02:00
if (p->state == SNMP_CONN)
{
snmp_ping(p);
}
2022-09-06 18:04:29 +02:00
2022-09-30 09:36:09 +02:00
//tm_set(tm, current_time() + (15 S));
tm_set(tm, current_time() + 15 S);
2022-09-20 14:28:57 +02:00
}
2022-08-10 17:31:32 +02:00
static int
snmp_shutdown(struct proto *P)
2022-08-01 13:01:49 +02:00
{
2022-12-10 18:08:00 +01:00
snmp_log("snmp_shutdown()");
2022-08-10 17:31:32 +02:00
struct snmp_proto *p = SKIP_BACK(struct snmp_proto, p, P);
2022-09-20 14:28:57 +02:00
tm_stop(p->ping_timer);
2022-12-17 18:16:19 +01:00
/* connection established => close the connection */
if (p->state == SNMP_CONN)
{
p->state = SNMP_STOP;
/* startup time is reused for connection closing */
p->startup_timer->hook = snmp_stop_timeout;
// TODO timeout duration ??
tm_set(p->startup_timer, 15 S);
snmp_stop_subagent(p);
return PS_STOP;
}
/* no connection to close */
else
{
snmp_cleanup(p);
return PS_DOWN;
}
2022-08-01 13:01:49 +02:00
}
struct protocol proto_snmp = {
.name = "Snmp",
.template = "snmp%d",
.channel_mask = NB_ANY,
.proto_size = sizeof(struct snmp_proto),
.config_size = sizeof(struct snmp_config),
2022-09-20 14:28:57 +02:00
.postconfig = snmp_postconfig,
2022-08-01 13:01:49 +02:00
.init = snmp_init,
.start = snmp_start,
.reconfigure = snmp_reconfigure,
2022-08-10 17:31:32 +02:00
.shutdown = snmp_shutdown,
2022-08-01 13:01:49 +02:00
.show_proto_info = snmp_show_proto_info,
};
void
2022-08-10 17:31:32 +02:00
snmp_build(void)
2022-08-01 13:01:49 +02:00
{
proto_build(&proto_snmp);
}