0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2024-12-31 22:21:54 +00:00

BMP: Remove superfluous error handling

Most error handling code was was for cases that cannot happen,
or they would be code bugs (and should use ASSERT()). Keep error
handling for just for I/O errors, like in rest of BIRD.
This commit is contained in:
Ondrej Zajicek (work) 2021-03-28 04:30:11 +02:00 committed by Ondrej Zajicek
parent a848dad40a
commit ad16e35177
6 changed files with 131 additions and 375 deletions

View File

@ -25,7 +25,6 @@
#include "proto/bmp/bmp.h" #include "proto/bmp/bmp.h"
#include "proto/bmp/buffer.h" #include "proto/bmp/buffer.h"
#include "proto/bmp/map.h" #include "proto/bmp/map.h"
#include "proto/bmp/utils.h"
#include <sys/socket.h> #include <sys/socket.h>
#include <sys/time.h> #include <sys/time.h>
@ -172,11 +171,28 @@ enum bmp_term_reason {
#define IP4_MAX_TTL 255 #define IP4_MAX_TTL 255
#define IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(expr, msg, rv...) \
do { \
if ((expr)) \
{ \
log(L_WARN "[BMP] " msg); \
return rv; \
} \
} while (0)
#define IF_PTR_IS_NULL_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(p, msg, rv...) \
do { \
IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(!(p), msg, rv); \
} while (0)
// Handle BIRD socket error event // Handle BIRD socket error event
static void static void
bmp_sock_err(sock *sk, int err); bmp_sock_err(sock *sk, int err);
static enum bmp_result static void
bmp_send_peer_up_notif_msg(const struct bgp_proto *bgp, bmp_send_peer_up_notif_msg(const struct bgp_proto *bgp,
const byte* tx_data, const size_t tx_data_size, const byte* tx_data, const size_t tx_data_size,
const byte* rx_data, const size_t rx_data_size); const byte* rx_data, const size_t rx_data_size);
@ -204,26 +220,20 @@ bmp_common_hdr_serialize(buffer *stream, const enum bmp_message_type type, const
bmp_put_u8(stream, type); bmp_put_u8(stream, type);
} }
static enum bmp_result static void
bmp_info_tlv_hdr_serialize(buffer *stream, const enum bmp_info_tlv_type type, bmp_info_tlv_hdr_serialize(buffer *stream, const enum bmp_info_tlv_type type,
const char *ascii) const char *str)
{ {
const size_t ascii_len = strlen(ascii); size_t str_len = strlen(str);
if (ascii_len > MIB_II_STR_LEN) str_len = MIN(str_len, 65535);
{
return BMP_E_INVALID_ARG;
}
bmp_put_u16(stream, type); bmp_put_u16(stream, type);
// It doesn't contain null terminating character '\0' bmp_put_u16(stream, str_len);
bmp_put_u16(stream, ascii_len); bmp_put_data(stream, str, str_len);
bmp_put_data(stream, ascii, ascii_len);
return BMP_E_NONE;
} }
// Serializes BMP Initiation message header [RFC 7854 - Section 4.3] // Serializes BMP Initiation message header [RFC 7854 - Section 4.3]
static enum bmp_result static void
bmp_init_msg_serialize(buffer *stream, const char *sys_descr, const char *sys_name) bmp_init_msg_serialize(buffer *stream, const char *sys_descr, const char *sys_name)
{ {
const size_t sys_descr_len = strlen(sys_descr); const size_t sys_descr_len = strlen(sys_descr);
@ -233,70 +243,42 @@ bmp_init_msg_serialize(buffer *stream, const char *sys_descr, const char *sys_na
const size_t data_size = (2 * BMP_INFO_TLV_FIX_SIZE) + sys_descr_len + sys_name_len; const size_t data_size = (2 * BMP_INFO_TLV_FIX_SIZE) + sys_descr_len + sys_name_len;
bmp_buffer_need(stream, BMP_COMMON_HDR_SIZE + data_size); bmp_buffer_need(stream, BMP_COMMON_HDR_SIZE + data_size);
bmp_common_hdr_serialize(stream, BMP_INIT_MSG, data_size); bmp_common_hdr_serialize(stream, BMP_INIT_MSG, data_size);
IF_BMP_FAILED_RETURN_RC( bmp_info_tlv_hdr_serialize(stream, BMP_INFO_TLV_TYPE_SYS_DESCR, sys_descr);
bmp_info_tlv_hdr_serialize(stream, BMP_INFO_TLV_TYPE_SYS_DESCR, sys_descr) bmp_info_tlv_hdr_serialize(stream, BMP_INFO_TLV_TYPE_SYS_NAME, sys_name);
);
IF_BMP_FAILED_RETURN_RC(
bmp_info_tlv_hdr_serialize(stream, BMP_INFO_TLV_TYPE_SYS_NAME, sys_name)
);
return BMP_E_NONE;
} }
static enum bmp_result static void
bmp_schedule_tx_packet(const byte *payload, const size_t size) bmp_schedule_tx_packet(const byte *payload, const size_t size)
{ {
IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( ASSERT(g_bmp->station_connected);
!g_bmp->station_connected,
"Not BMP station connected yet",
BMP_E_CONNECT_TO_SRV
);
struct bmp_data_node *tx_data = mb_alloc(g_bmp->tx_mem_pool, sizeof (struct bmp_data_node)); struct bmp_data_node *tx_data = mb_alloc(g_bmp->tx_mem_pool, sizeof (struct bmp_data_node));
tx_data->data = mb_alloc(g_bmp->tx_mem_pool, size); tx_data->data = mb_alloc(g_bmp->tx_mem_pool, size);
memcpy(tx_data->data, payload, size); memcpy(tx_data->data, payload, size);
tx_data->data_size = size; tx_data->data_size = size;
add_tail(&g_bmp->tx_queue, &tx_data->n); add_tail(&g_bmp->tx_queue, &tx_data->n);
if (sk_tx_buffer_empty(g_bmp->conn->sk) if (sk_tx_buffer_empty(g_bmp->conn->sk)
&& !ev_active(g_bmp->conn->tx_ev)) && !ev_active(g_bmp->conn->tx_ev))
{ {
ev_schedule(g_bmp->conn->tx_ev); ev_schedule(g_bmp->conn->tx_ev);
} }
return BMP_E_NONE;
} }
/** /**
* bmp_startup - connect to the BMP collector. * bmp_startup - connect to the BMP collector.
* NOTE: Send Initiation Message to the BMP collector. * NOTE: Send Initiation Message to the BMP collector.
*/ */
static enum bmp_result static void
bmp_startup(void) bmp_startup(void)
{ {
if (g_bmp->started || !g_bmp->station_connected) ASSERT(g_bmp->station_connected && !g_bmp->started);
{
return BMP_E_NONE;
}
enum bmp_result rc = BMP_E_NONE; buffer payload = bmp_buffer_alloc(g_bmp->buffer_mpool, DEFAULT_MEM_BLOCK_SIZE);
buffer payload bmp_init_msg_serialize(&payload, g_bmp->sys_descr, g_bmp->sys_name);
= bmp_buffer_alloc(g_bmp->buffer_mpool, DEFAULT_MEM_BLOCK_SIZE); bmp_schedule_tx_packet(bmp_buffer_data(&payload), bmp_buffer_pos(&payload));
IF_BMP_FAILED_GOTO_LABEL(
bmp_init_msg_serialize(&payload, g_bmp->sys_descr, g_bmp->sys_name),
err
);
IF_BMP_FAILED_GOTO_LABEL(
bmp_schedule_tx_packet(bmp_buffer_data(&payload), bmp_buffer_pos(&payload)),
err
);
g_bmp->started = true; g_bmp->started = true;
err:
bmp_buffer_free(&payload);
return rc;
} }
void void
@ -331,8 +313,8 @@ bmp_fire_tx(void *vconn)
rem_node((node *) tx_data); rem_node((node *) tx_data);
mb_free(tx_data); mb_free(tx_data);
IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(
(sk_send(conn->sk, data_size) == 0), (sk_send(conn->sk, data_size) <= 0),
"Failed to send BMP packet", "Failed to send BMP packet"
); );
// BMP packets should be treat with lowest priority when scheduling sending // BMP packets should be treat with lowest priority when scheduling sending
@ -353,37 +335,35 @@ bmp_fire_tx(void *vconn)
static void static void
bmp_tx(struct birdsock *sk) bmp_tx(struct birdsock *sk)
{ {
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(
bmp_startup(),
"Failed to send Initation message to BMP collector"
);
bmp_fire_tx(sk->data); bmp_fire_tx(sk->data);
} }
static inline enum bmp_result static inline int
bmp_open_socket(struct bmp_proto *bmp) bmp_open_socket(struct bmp_proto *bmp)
{ {
bmp->conn->sk->daddr = bmp->station_ip; sock *s = bmp->conn->sk;
bmp->conn->sk->dport = bmp->station_port; s->daddr = bmp->station_ip;
bmp->conn->sk->err_hook = bmp_sock_err; s->dport = bmp->station_port;
s->err_hook = bmp_sock_err;
if (sk_open(bmp->conn->sk) < 0) int rc = sk_open(s);
{
return BMP_E_OPEN_SOCKET;
}
return BMP_E_NONE; if (rc < 0)
sk_log_error(s, bmp->p.name);
return rc;
} }
static void static void
bmp_connection_retry(timer *t) bmp_connection_retry(timer *t)
{ {
struct bmp_proto *bmp = (struct bmp_proto *) t->data; struct bmp_proto *bmp = (struct bmp_proto *) t->data;
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(
bmp_open_socket(bmp), if (bmp_open_socket(bmp) < 0)
"Failed to connect to BMP station" {
); log(L_DEBUG "Failed to connect to BMP station");
return;
}
log(L_DEBUG "Connected to BMP station after connection retry"); log(L_DEBUG "Connected to BMP station after connection retry");
tm_stop(t); tm_stop(t);
@ -534,49 +514,30 @@ bmp_peer_down_notif_msg_serialize(buffer *stream, const bool is_peer_global,
* bmp_open - initialize internal resources of BMP implementation. * bmp_open - initialize internal resources of BMP implementation.
* NOTE: It does not connect to BMP collector yet. * NOTE: It does not connect to BMP collector yet.
*/ */
enum bmp_result void
bmp_open(const struct proto *P) bmp_open(const struct proto *P)
{ {
struct bmp_proto *bmp = (struct bmp_proto *) P; struct bmp_proto *bmp = (struct bmp_proto *) P;
g_bmp = bmp; g_bmp = bmp;
enum bmp_result rc = BMP_E_NONE;
log(L_DEBUG "Init BMP"); log(L_DEBUG "Init BMP");
if (!P)
{
return BMP_E_NULL_REF;
}
g_bmp->buffer_mpool = rp_new(P->pool, "BMP Buffer"); g_bmp->buffer_mpool = rp_new(P->pool, "BMP Buffer");
g_bmp->map_mem_pool = rp_new(P->pool, "BMP Map"); g_bmp->map_mem_pool = rp_new(P->pool, "BMP Map");
g_bmp->tx_mem_pool = rp_new(P->pool, "BMP Tx"); g_bmp->tx_mem_pool = rp_new(P->pool, "BMP Tx");
g_bmp->update_msg_mem_pool = rp_new(P->pool, "BMP Update"); g_bmp->update_msg_mem_pool = rp_new(P->pool, "BMP Update");
bmp->conn->tx_ev = ev_new_init(g_bmp->tx_mem_pool, bmp_fire_tx, bmp->conn); bmp->conn->tx_ev = ev_new_init(g_bmp->tx_mem_pool, bmp_fire_tx, bmp->conn);
if (!bmp->conn->tx_ev)
{
return BMP_E_NEW_TX_EVENT;
}
IF_BMP_FAILED_RETURN_RC( bmp_peer_map_init(&g_bmp->peer_open_msg.tx_msg, g_bmp->map_mem_pool);
bmp_peer_map_init(&g_bmp->peer_open_msg.tx_msg, g_bmp->map_mem_pool) bmp_peer_map_init(&g_bmp->peer_open_msg.rx_msg, g_bmp->map_mem_pool);
); bmp_peer_map_init(&g_bmp->bgp_peers, g_bmp->map_mem_pool);
IF_BMP_FAILED_GOTO_LABEL(
bmp_peer_map_init(&g_bmp->peer_open_msg.rx_msg, g_bmp->map_mem_pool),
err_free_ip_map_tx_msg
);
IF_BMP_FAILED_GOTO_LABEL(
bmp_peer_map_init(&g_bmp->bgp_peers, g_bmp->map_mem_pool),
err_free_ip_map_rx_msg
);
init_list(&g_bmp->tx_queue); init_list(&g_bmp->tx_queue);
init_list(&g_bmp->rt_table_in_pre_policy.update_msg_queue); init_list(&g_bmp->rt_table_in_pre_policy.update_msg_queue);
g_bmp->station_connected = false; g_bmp->station_connected = false;
g_bmp->started = false; g_bmp->started = false;
g_bmp->connect_retry_timer = NULL; g_bmp->connect_retry_timer = NULL;
if (BMP_FAILED(bmp_open_socket(bmp))) if (bmp_open_socket(bmp) < 0)
{ {
log(L_DEBUG "Failed to connect to BMP station"); log(L_DEBUG "Failed to connect to BMP station");
g_bmp->connect_retry_timer = tm_new_init(P->pool, bmp_connection_retry, bmp, g_bmp->connect_retry_timer = tm_new_init(P->pool, bmp_connection_retry, bmp,
@ -591,14 +552,6 @@ bmp_open(const struct proto *P)
strncpy(g_bmp->sys_name, bmp->sys_name, sizeof (g_bmp->sys_name) - 1); strncpy(g_bmp->sys_name, bmp->sys_name, sizeof (g_bmp->sys_name) - 1);
strncpy(g_bmp->sys_descr, bmp->sys_descr, sizeof (g_bmp->sys_descr) - 1); strncpy(g_bmp->sys_descr, bmp->sys_descr, sizeof (g_bmp->sys_descr) - 1);
return BMP_E_NONE;
err_free_ip_map_rx_msg:
bmp_peer_map_free(&g_bmp->peer_open_msg.rx_msg);
err_free_ip_map_tx_msg:
bmp_peer_map_free(&g_bmp->peer_open_msg.tx_msg);
return rc;
} }
void void
@ -606,6 +559,7 @@ bmp_peer_map_walk_tx_open_msg_and_send_peer_up_notif(
const struct bmp_peer_map_key key, const byte *tx_msg, const struct bmp_peer_map_key key, const byte *tx_msg,
const size_t tx_msg_size) const size_t tx_msg_size)
{ {
ASSERT(g_bmp->station_connected);
const struct bmp_peer_map_entry *map_rx_msg = bmp_peer_map_get(&g_bmp->peer_open_msg.rx_msg, key); const struct bmp_peer_map_entry *map_rx_msg = bmp_peer_map_get(&g_bmp->peer_open_msg.rx_msg, key);
IF_PTR_IS_NULL_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( IF_PTR_IS_NULL_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(
map_rx_msg, map_rx_msg,
@ -622,11 +576,8 @@ bmp_peer_map_walk_tx_open_msg_and_send_peer_up_notif(
memcpy(&bgp, map_bgp_proto->data.buf, sizeof (bgp)); memcpy(&bgp, map_bgp_proto->data.buf, sizeof (bgp));
if (bgp->p.proto_state == PS_UP) if (bgp->p.proto_state == PS_UP)
{ {
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( bmp_send_peer_up_notif_msg(bgp, tx_msg, tx_msg_size,
bmp_send_peer_up_notif_msg(bgp, tx_msg, tx_msg_size, map_rx_msg->data.buf, map_rx_msg->data.buf, map_rx_msg->data.buf_size);
map_rx_msg->data.buf_size),
"Failed to send BMP Peer Up Notification MSG"
);
} }
} }
@ -655,6 +606,7 @@ static const struct birdsock *
bmp_get_birdsock_ext(const struct bgp_proto *bgp) bmp_get_birdsock_ext(const struct bgp_proto *bgp)
{ {
const struct birdsock *sk = bmp_get_birdsock(bgp); const struct birdsock *sk = bmp_get_birdsock(bgp);
if (sk != NULL) if (sk != NULL)
{ {
return sk; return sk;
@ -713,47 +665,35 @@ bmp_is_peer_global_instance(const struct bgp_proto *bgp)
: (bgp->cf->peer_type == BGP_PT_EXTERNAL); : (bgp->cf->peer_type == BGP_PT_EXTERNAL);
} }
enum bmp_result static void
bmp_send_peer_up_notif_msg(const struct bgp_proto *bgp, bmp_send_peer_up_notif_msg(const struct bgp_proto *bgp,
const byte* tx_data, const size_t tx_data_size, const byte* tx_data, const size_t tx_data_size,
const byte* rx_data, const size_t rx_data_size) const byte* rx_data, const size_t rx_data_size)
{ {
ASSERT(g_bmp->station_connected);
const struct birdsock *sk = bmp_get_birdsock_ext(bgp); const struct birdsock *sk = bmp_get_birdsock_ext(bgp);
if (IS_NULL(sk)) IF_PTR_IS_NULL_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(
{ sk,
return BMP_E_BIRDSOCK_NULL_REF; "[BMP] No BGP socket"
} );
const struct bgp_caps *remote_caps = bmp_get_bgp_remote_caps_ext(bgp);
if (IS_NULL(remote_caps))
{
return BMP_E_REMOTE_CAPS_NULL_REF;
}
enum bmp_result rc = BMP_E_NONE;
const bool is_global_instance_peer = bmp_is_peer_global_instance(bgp); const bool is_global_instance_peer = bmp_is_peer_global_instance(bgp);
buffer payload = bmp_buffer_alloc(g_bmp->buffer_mpool, DEFAULT_MEM_BLOCK_SIZE); buffer payload = bmp_buffer_alloc(g_bmp->buffer_mpool, DEFAULT_MEM_BLOCK_SIZE);
bmp_peer_up_notif_msg_serialize(&payload, is_global_instance_peer, bmp_peer_up_notif_msg_serialize(&payload, is_global_instance_peer,
bgp->remote_as, bgp->remote_id, remote_caps->as4_support, bgp->remote_as, bgp->remote_id, 1,
sk->saddr, sk->daddr, sk->sport, sk->dport, tx_data, tx_data_size, sk->saddr, sk->daddr, sk->sport, sk->dport, tx_data, tx_data_size,
rx_data, rx_data_size); rx_data, rx_data_size);
IF_BMP_FAILED_GOTO_LABEL( bmp_schedule_tx_packet(bmp_buffer_data(&payload), bmp_buffer_pos(&payload));
bmp_schedule_tx_packet(bmp_buffer_data(&payload), bmp_buffer_pos(&payload)),
err
);
bmp_peer_up(bgp); bmp_peer_up(bgp);
err:
bmp_buffer_free(&payload);
return rc;
} }
void void
bmp_put_sent_bgp_open_msg(const struct bgp_proto *bgp, const byte* pkt, bmp_put_sent_bgp_open_msg(const struct bgp_proto *bgp, const byte* pkt,
const size_t pkt_size) const size_t pkt_size)
{ {
if (IS_NULL(g_bmp)) if (!g_bmp)
{ {
return; return;
} }
@ -764,35 +704,25 @@ bmp_put_sent_bgp_open_msg(const struct bgp_proto *bgp, const byte* pkt,
= bmp_peer_map_get(&g_bmp->peer_open_msg.rx_msg, key); = bmp_peer_map_get(&g_bmp->peer_open_msg.rx_msg, key);
if (!map_entry || !g_bmp->started) if (!map_entry || !g_bmp->started)
{ {
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( bmp_peer_map_insert(&g_bmp->peer_open_msg.tx_msg, key, pkt, pkt_size);
bmp_peer_map_insert(&g_bmp->peer_open_msg.tx_msg, key, pkt, pkt_size),
"Failed to put data into map"
);
if (!map_entry) if (!map_entry)
{ {
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( bmp_peer_map_insert(&g_bmp->bgp_peers, key, (const byte *) &bgp, sizeof (bgp));
bmp_peer_map_insert(&g_bmp->bgp_peers, key, (const byte *) &bgp,
sizeof (bgp)),
"Failed to put data into map"
);
} }
return; return;
} }
IF_BMP_FAILED_PRINT_ERR_MSG(
bmp_send_peer_up_notif_msg(bgp, pkt, pkt_size, map_entry->data.buf, bmp_send_peer_up_notif_msg(bgp, pkt, pkt_size, map_entry->data.buf,
map_entry->data.buf_size), map_entry->data.buf_size);
"Failed to send PEER UP notification"
);
} }
void void
bmp_put_recv_bgp_open_msg(const struct bgp_proto *bgp, const byte* pkt, bmp_put_recv_bgp_open_msg(const struct bgp_proto *bgp, const byte* pkt,
const size_t pkt_size) const size_t pkt_size)
{ {
if (IS_NULL(g_bmp)) if (!g_bmp)
{ {
return; return;
} }
@ -803,34 +733,24 @@ bmp_put_recv_bgp_open_msg(const struct bgp_proto *bgp, const byte* pkt,
= bmp_peer_map_get(&g_bmp->peer_open_msg.tx_msg, key); = bmp_peer_map_get(&g_bmp->peer_open_msg.tx_msg, key);
if (!map_data || !g_bmp->started) if (!map_data || !g_bmp->started)
{ {
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( bmp_peer_map_insert(&g_bmp->peer_open_msg.rx_msg, key, pkt, pkt_size);
bmp_peer_map_insert(&g_bmp->peer_open_msg.rx_msg, key, pkt, pkt_size),
"Failed to insert item into peer map"
);
if (!map_data) if (!map_data)
{ {
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( bmp_peer_map_insert(&g_bmp->bgp_peers, key, (const byte *) &bgp, sizeof (bgp));
bmp_peer_map_insert(&g_bmp->bgp_peers, key, (const byte *) &bgp,
sizeof (bgp)),
"Failed to insert item into peer map"
);
} }
return; return;
} }
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(
bmp_send_peer_up_notif_msg(bgp, map_data->data.buf, map_data->data.buf_size, bmp_send_peer_up_notif_msg(bgp, map_data->data.buf, map_data->data.buf_size,
pkt, pkt_size), pkt, pkt_size);
"Failed to send peer up notification"
);
} }
void void
bmp_route_monitor_update_in_pre_begin() bmp_route_monitor_update_in_pre_begin()
{ {
if (IS_NULL(g_bmp)) if (!g_bmp)
{ {
return; return;
} }
@ -859,7 +779,7 @@ bmp_route_monitor_update_in_pre_begin()
void void
bmp_route_monitor_put_update_in_pre_msg(const byte *data, const size_t data_size) bmp_route_monitor_put_update_in_pre_msg(const byte *data, const size_t data_size)
{ {
if (IS_NULL(g_bmp)) if (!g_bmp)
{ {
return; return;
} }
@ -891,7 +811,7 @@ bmp_route_monitor_put_update_in_pre_msg(const byte *data, const size_t data_size
void void
bmp_route_monitor_update_in_pre_commit(const struct bgp_proto *bgp) bmp_route_monitor_update_in_pre_commit(const struct bgp_proto *bgp)
{ {
if (IS_NULL(g_bmp)) if (!g_bmp)
{ {
return; return;
} }
@ -938,25 +858,17 @@ bmp_route_monitor_update_in_pre_commit(const struct bgp_proto *bgp)
g_bmp->rt_table_in_pre_policy.update_begin_time.tv_sec, g_bmp->rt_table_in_pre_policy.update_begin_time.tv_sec,
g_bmp->rt_table_in_pre_policy.update_begin_time.tv_usec); g_bmp->rt_table_in_pre_policy.update_begin_time.tv_usec);
IF_BMP_FAILED_PRINT_ERR_MSG_AND_GOTO_LABEL( bmp_schedule_tx_packet(bmp_buffer_data(&payload), bmp_buffer_pos(&payload));
bmp_schedule_tx_packet(bmp_buffer_data(&payload), bmp_buffer_pos(&payload)),
"Failed to schedule packet for sent",
err
);
bmp_buffer_flush(&payload); bmp_buffer_flush(&payload);
bmp_buffer_flush(&update_msgs); bmp_buffer_flush(&update_msgs);
} }
err:
bmp_buffer_free(&update_msgs);
bmp_buffer_free(&payload);
} }
void void
bmp_route_monitor_update_in_pre_end() bmp_route_monitor_update_in_pre_end()
{ {
if (IS_NULL(g_bmp)) if (!g_bmp)
{ {
return; return;
} }
@ -990,7 +902,7 @@ bmp_route_monitor_pre_policy_table_in_snapshot(const struct channel *C)
} }
struct rtable *tab = C->in_table; struct rtable *tab = C->in_table;
if (IS_NULL(tab)) if (!tab)
{ {
return; return;
} }
@ -1039,33 +951,30 @@ bmp_route_monitor_pre_policy_table_in_snapshot(const struct channel *C)
} }
} }
static enum bmp_result static void
bmp_send_peer_down_notif_msg(const struct bgp_proto *bgp, bmp_send_peer_down_notif_msg(const struct bgp_proto *bgp,
const byte* data, const size_t data_size) const byte* data, const size_t data_size)
{ {
ASSERT(g_bmp->station_connected);
const struct bgp_caps *remote_caps = bmp_get_bgp_remote_caps_ext(bgp); const struct bgp_caps *remote_caps = bmp_get_bgp_remote_caps_ext(bgp);
bool is_global_instance_peer = bmp_is_peer_global_instance(bgp); bool is_global_instance_peer = bmp_is_peer_global_instance(bgp);
enum bmp_result rc = BMP_E_NONE;
buffer payload buffer payload
= bmp_buffer_alloc(g_bmp->buffer_mpool, DEFAULT_MEM_BLOCK_SIZE); = bmp_buffer_alloc(g_bmp->buffer_mpool, DEFAULT_MEM_BLOCK_SIZE);
bmp_peer_down_notif_msg_serialize(&payload, is_global_instance_peer, bmp_peer_down_notif_msg_serialize(&payload, is_global_instance_peer,
bgp->remote_as, bgp->remote_id, bgp->remote_as, bgp->remote_id,
remote_caps ? remote_caps->as4_support : bgp->as4_session, remote_caps ? remote_caps->as4_support : bgp->as4_session,
bgp->remote_ip, data, data_size); bgp->remote_ip, data, data_size);
IF_BMP_FAILED_PRINT_ERR_MSG( bmp_schedule_tx_packet(bmp_buffer_data(&payload), bmp_buffer_pos(&payload));
bmp_schedule_tx_packet(bmp_buffer_data(&payload), bmp_buffer_pos(&payload)),
"Failed to schedule packet for transmit"
);
bmp_buffer_free(&payload); bmp_buffer_free(&payload);
return rc;
} }
void void
bmp_peer_down(const struct bgp_proto *bgp, const int err_class, const byte *pkt, bmp_peer_down(const struct bgp_proto *bgp, const int err_class, const byte *pkt,
size_t pkt_size) size_t pkt_size)
{ {
if (IS_NULL(g_bmp) || (!g_bmp->started)) if (!g_bmp || !g_bmp->started)
{ {
return; return;
} }
@ -1119,20 +1028,15 @@ bmp_peer_down(const struct bgp_proto *bgp, const int err_class, const byte *pkt,
} }
} }
IF_BMP_FAILED_PRINT_ERR_MSG( bmp_send_peer_down_notif_msg(bgp, bmp_buffer_data(&payload), bmp_buffer_pos(&payload));
bmp_send_peer_down_notif_msg(bgp, bmp_buffer_data(&payload),
bmp_buffer_pos(&payload)),
"Failed to send PEER DOWN notification"
);
bmp_buffer_free(&payload); bmp_buffer_free(&payload);
} }
static enum bmp_result static void
bmp_send_termination_msg(struct bmp_proto *bmp, bmp_send_termination_msg(struct bmp_proto *bmp,
const enum bmp_term_reason reason) const enum bmp_term_reason reason)
{ {
enum bmp_result rc = BMP_E_NONE;
const size_t term_msg_hdr_size = BMP_TERM_INFO_TYPE_SIZE const size_t term_msg_hdr_size = BMP_TERM_INFO_TYPE_SIZE
+ BMP_TERM_INFO_LEN_FIELD_SIZE + BMP_TERM_INFO_LEN_FIELD_SIZE
+ BMP_TERM_REASON_CODE_SIZE; + BMP_TERM_REASON_CODE_SIZE;
@ -1144,13 +1048,12 @@ bmp_send_termination_msg(struct bmp_proto *bmp,
bmp_put_u16(&stream, BMP_TERM_REASON_CODE_SIZE); // 2-byte code indication the reason bmp_put_u16(&stream, BMP_TERM_REASON_CODE_SIZE); // 2-byte code indication the reason
bmp_put_u16(&stream, reason); bmp_put_u16(&stream, reason);
memcpy(bmp->conn->sk->tbuf, bmp_buffer_data(&stream), bmp_buffer_pos(&stream)); memcpy(bmp->conn->sk->tbuf, bmp_buffer_data(&stream), bmp_buffer_pos(&stream));
if (sk_send(bmp->conn->sk, bmp_buffer_pos(&stream)) < 0) IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(
{ sk_send(bmp->conn->sk, bmp_buffer_pos(&stream)) < 0,
rc = BMP_E_SEND_DATA; "Failed to send BMP termination message"
} );
bmp_buffer_free(&stream); bmp_buffer_free(&stream);
return rc;
} }
static void static void
@ -1159,10 +1062,8 @@ bmp_station_connected(struct birdsock *sk)
struct bmp_conn *conn = (struct bmp_conn *)sk->data; struct bmp_conn *conn = (struct bmp_conn *)sk->data;
conn->sk->tx_hook = bmp_tx; conn->sk->tx_hook = bmp_tx;
conn->bmp->station_connected = true; conn->bmp->station_connected = true;
IF_BMP_FAILED_PRINT_ERR_MSG(
bmp_startup(), bmp_startup();
"Failed to start BMP protocol"
);
bmp_peer_map_walk(&conn->bmp->peer_open_msg.tx_msg, bmp_peer_map_walk(&conn->bmp->peer_open_msg.tx_msg,
bmp_peer_map_walk_tx_open_msg_and_send_peer_up_notif); bmp_peer_map_walk_tx_open_msg_and_send_peer_up_notif);
@ -1190,11 +1091,6 @@ bmp_init(struct proto_config *CF)
{ {
struct proto *P = proto_new(CF); struct proto *P = proto_new(CF);
struct bmp_proto *bmp = (struct bmp_proto *) P; struct bmp_proto *bmp = (struct bmp_proto *) P;
IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(
(g_bmp != NULL),
"Single BMP protocol instance has been already run",
PS_DOWN
);
struct bmp_config *cf = (struct bmp_config *) CF; struct bmp_config *cf = (struct bmp_config *) CF;
bmp->cf = cf; bmp->cf = cf;
@ -1229,11 +1125,7 @@ bmp_start(struct proto *P)
bmp->conn = mb_allocz(P->pool, sizeof (struct bmp_conn)); bmp->conn = mb_allocz(P->pool, sizeof (struct bmp_conn));
bmp->conn->bmp = bmp; bmp->conn->bmp = bmp;
bmp_setup_socket(bmp->conn); bmp_setup_socket(bmp->conn);
IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL( bmp_open(P);
bmp_open(P),
"Failed to connect to BMP collector",
PS_DOWN
);
return PS_UP; return PS_UP;
} }
@ -1242,10 +1134,7 @@ static int
bmp_shutdown(struct proto *P) bmp_shutdown(struct proto *P)
{ {
struct bmp_proto *bmp = (struct bmp_proto *) P; struct bmp_proto *bmp = (struct bmp_proto *) P;
IF_BMP_FAILED_PRINT_ERR_MSG( bmp_send_termination_msg(bmp, BMP_TERM_REASON_ADM);
bmp_send_termination_msg(bmp, BMP_TERM_REASON_ADM),
"Failed to send BMP termination message"
);
g_bmp->station_connected = false; g_bmp->station_connected = false;
g_bmp->started = false; g_bmp->started = false;

View File

@ -39,8 +39,8 @@ struct monitoring_rib {
struct bmp_config { struct bmp_config {
struct proto_config c; struct proto_config c;
char *sys_descr; // sysDescr MIB-II [RFC1213] object const char *sys_descr; // sysDescr MIB-II [RFC1213] object
char *sys_name; // sysName MIB-II [RFC1213] object const char *sys_name; // sysName MIB-II [RFC1213] object
ip_addr station_ip; // Monitoring station address ip_addr station_ip; // Monitoring station address
u16 station_port; // Monitoring station TCP port u16 station_port; // Monitoring station TCP port
bool disabled; // Manually disabled bool disabled; // Manually disabled

View File

@ -10,7 +10,6 @@
#define _BIRD_BMP_BUFFER_H_ #define _BIRD_BMP_BUFFER_H_
#include "proto/bmp/bmp.h" #include "proto/bmp/bmp.h"
#include "proto/bmp/utils.h"
#include <stdlib.h> #include <stdlib.h>

View File

@ -7,7 +7,6 @@
*/ */
#include "proto/bmp/map.h" #include "proto/bmp/map.h"
#include "proto/bmp/utils.h"
/* Peer Index Table */ /* Peer Index Table */
#define PEER_KEY(n) (n)->peer_as, (n)->peer_ip #define PEER_KEY(n) (n)->peer_as, (n)->peer_ip
@ -23,18 +22,11 @@ HASH_DEFINE_REHASH_FN(PEER, struct bmp_peer_map_key)
#define PEER_INIT_ORDER 6 #define PEER_INIT_ORDER 6
enum bmp_result void
bmp_peer_map_init(struct bmp_peer_map *map, pool *mpool) bmp_peer_map_init(struct bmp_peer_map *map, pool *mpool)
{ {
if (IS_NULL(map) || IS_NULL(mpool))
{
return BMP_E_NULL_REF;
}
map->mpool = mpool; map->mpool = mpool;
HASH_INIT(map->peer_hash, map->mpool, PEER_INIT_ORDER); HASH_INIT(map->peer_hash, map->mpool, PEER_INIT_ORDER);
return BMP_E_NONE;
} }
struct bmp_peer_map_key struct bmp_peer_map_key
@ -48,14 +40,9 @@ bmp_peer_map_key_create(const ip_addr peer_ip, const u32 peer_as)
return key; return key;
} }
enum bmp_result void
bmp_peer_map_flush(struct bmp_peer_map *map) bmp_peer_map_flush(struct bmp_peer_map *map)
{ {
if (IS_NULL(map))
{
return BMP_E_NULL_REF;
}
struct bmp_peer_map_entry *entry; struct bmp_peer_map_entry *entry;
HASH_WALK_DELSAFE(map->peer_hash, next, e) HASH_WALK_DELSAFE(map->peer_hash, next, e)
{ {
@ -67,78 +54,55 @@ bmp_peer_map_flush(struct bmp_peer_map *map)
HASH_WALK_DELSAFE_END; HASH_WALK_DELSAFE_END;
HASH_MAY_RESIZE_DOWN(map->peer_hash, PEER, map->mpool); HASH_MAY_RESIZE_DOWN(map->peer_hash, PEER, map->mpool);
return BMP_E_NONE;
} }
enum bmp_result void
bmp_peer_map_free(struct bmp_peer_map *map) bmp_peer_map_free(struct bmp_peer_map *map)
{ {
if (IS_NULL(map)) bmp_peer_map_flush(map);
{
return BMP_E_NULL_REF;
}
IF_BMP_FAILED_RETURN_RC(bmp_peer_map_flush(map));
HASH_FREE(map->peer_hash); HASH_FREE(map->peer_hash);
return BMP_E_NONE;
} }
enum bmp_result void
bmp_peer_map_insert(struct bmp_peer_map *map, const struct bmp_peer_map_key key, bmp_peer_map_insert(struct bmp_peer_map *map, const struct bmp_peer_map_key key,
const byte *data, const size_t data_size) const byte *data, const size_t data_size)
{ {
if (IS_NULL(map)) struct bmp_peer_map_entry *entry
= (void *) HASH_FIND(map->peer_hash, PEER, PEER_KEY(&key));
if (entry)
{ {
return BMP_E_NULL_REF; mb_free(entry->data.buf);
entry->data.buf = mb_alloc(map->mpool, data_size);
memcpy(entry->data.buf, data, data_size);
entry->data.buf_size = data_size;
return;
} }
if (HASH_FIND(map->peer_hash, PEER, PEER_KEY(&key))) entry = mb_alloc(map->mpool, sizeof (struct bmp_peer_map_entry));
{
return BMP_E_EXISTS;
}
struct bmp_peer_map_entry *entry = mb_alloc(map->mpool,
sizeof (struct bmp_peer_map_entry));
entry->data.buf = mb_alloc(map->mpool, data_size); entry->data.buf = mb_alloc(map->mpool, data_size);
memcpy(entry->data.buf, data, data_size); memcpy(entry->data.buf, data, data_size);
entry->data.buf_size = data_size; entry->data.buf_size = data_size;
entry->key = key; entry->key = key;
HASH_INSERT2(map->peer_hash, PEER, map->mpool, &entry->key); HASH_INSERT2(map->peer_hash, PEER, map->mpool, &entry->key);
return BMP_E_NONE;
} }
enum bmp_result void
bmp_peer_map_remove(struct bmp_peer_map *map, const struct bmp_peer_map_key key) bmp_peer_map_remove(struct bmp_peer_map *map, const struct bmp_peer_map_key key)
{ {
if (IS_NULL(map))
{
return BMP_E_NULL_REF;
}
struct bmp_peer_map_entry *entry struct bmp_peer_map_entry *entry
= (struct bmp_peer_map_entry *) HASH_FIND(map->peer_hash, PEER, PEER_KEY(&key)); = (void *) HASH_DELETE(map->peer_hash, PEER, PEER_KEY(&key));
if (IS_NULL(entry))
{ if (!entry)
return BMP_E_NOT_EXISTS; return;
}
mb_free(entry->data.buf); mb_free(entry->data.buf);
HASH_DELETE(map->peer_hash, PEER, PEER_KEY(&entry->key));
mb_free(entry); mb_free(entry);
return BMP_E_NONE;
} }
const struct bmp_peer_map_entry * const struct bmp_peer_map_entry *
bmp_peer_map_get(struct bmp_peer_map *map, const struct bmp_peer_map_key key) bmp_peer_map_get(struct bmp_peer_map *map, const struct bmp_peer_map_key key)
{ {
if (IS_NULL(map))
{
return NULL;
}
return (struct bmp_peer_map_entry *) HASH_FIND(map->peer_hash, PEER, PEER_KEY(&key)); return (struct bmp_peer_map_entry *) HASH_FIND(map->peer_hash, PEER, PEER_KEY(&key));
} }

View File

@ -16,8 +16,6 @@
#include "lib/hash.h" #include "lib/hash.h"
#include "lib/resource.h" #include "lib/resource.h"
#include "proto/bmp/utils.h"
struct bmp_peer_map_key { struct bmp_peer_map_key {
struct bmp_peer_map_key *next; struct bmp_peer_map_key *next;
ip_addr peer_ip; ip_addr peer_ip;
@ -39,23 +37,23 @@ struct bmp_peer_map {
HASH(struct bmp_peer_map_key) peer_hash; // Hash for peers to find the index HASH(struct bmp_peer_map_key) peer_hash; // Hash for peers to find the index
}; };
enum bmp_result void
bmp_peer_map_init(struct bmp_peer_map *map, pool *mpool); bmp_peer_map_init(struct bmp_peer_map *map, pool *mpool);
struct bmp_peer_map_key struct bmp_peer_map_key
bmp_peer_map_key_create(const ip_addr peer_ip, const u32 peer_as); bmp_peer_map_key_create(const ip_addr peer_ip, const u32 peer_as);
enum bmp_result void
bmp_peer_map_free(struct bmp_peer_map *map); bmp_peer_map_free(struct bmp_peer_map *map);
enum bmp_result void
bmp_peer_map_flush(struct bmp_peer_map *map); bmp_peer_map_flush(struct bmp_peer_map *map);
enum bmp_result void
bmp_peer_map_insert(struct bmp_peer_map *map, const struct bmp_peer_map_key key, bmp_peer_map_insert(struct bmp_peer_map *map, const struct bmp_peer_map_key key,
const byte *data, const size_t data_size); const byte *data, const size_t data_size);
enum bmp_result void
bmp_peer_map_remove(struct bmp_peer_map *map, const struct bmp_peer_map_key key); bmp_peer_map_remove(struct bmp_peer_map *map, const struct bmp_peer_map_key key);
const struct bmp_peer_map_entry * const struct bmp_peer_map_entry *

View File

@ -1,94 +0,0 @@
/*
* BIRD -- The BGP Monitoring Protocol (BMP)
*
* (c) 2020 Akamai Technologies, Inc. (Pawel Maslanka, pmaslank@akamai.com)
*
* Can be freely distributed and used under the terms of the GNU GPL.
*/
#ifndef _BIRD_BMP_UTILS_H_
#define _BIRD_BMP_UTILS_H_
/**
* TODO:
* - Provide descriptive explanation for the follwing enums
*/
enum bmp_result {
BMP_E_NONE = 0,
BMP_E_INVALID_ARG,
BMP_E_NULL_REF,
BMP_E_EXISTS,
BMP_E_NOT_EXISTS,
BMP_E_OPEN_SOCKET,
BMP_E_CONNECT_TO_SRV,
BMP_E_SEND_DATA,
BMP_E_BIRDSOCK_NULL_REF,
BMP_E_REMOTE_CAPS_NULL_REF,
BMP_E_NEW_TX_EVENT
};
#define BMP_FAILED(v) \
(v != BMP_E_NONE)
#define IF_BMP_FAILED_RETURN_RC(func) \
do { \
enum bmp_result rc = func; \
if (BMP_FAILED(rc)) \
{ \
return rc; \
} \
} while (0)
// The following macro requires to define locally enum bmp_result rc;
#define IF_BMP_FAILED_GOTO_LABEL(func, label) \
do { \
rc = func; \
if (BMP_FAILED(rc)) \
{ \
goto label; \
} \
} while (0)
#define IF_BMP_FAILED_PRINT_ERR_MSG(func, msg) \
do { \
enum bmp_result rc = func; \
if (BMP_FAILED(rc)) \
{ \
log(L_WARN "[BMP] " msg " (rc = %d)", rc); \
} \
} while (0)
#define IF_BMP_FAILED_PRINT_ERR_MSG_AND_GOTO_LABEL(func, msg, label) \
do { \
enum bmp_result rc = func; \
if (BMP_FAILED(rc)) \
{ \
log(L_WARN "[BMP] " msg " (rc = %d)", rc); \
goto label; \
} \
} while (0)
#define IS_NULL(ptr) \
((ptr) == NULL)
#define IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(expr, msg, rv...) \
do { \
if ((expr)) \
{ \
log(L_WARN "[BMP] " msg); \
return rv; \
} \
} while (0)
#define IF_PTR_IS_NULL_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(p, msg, rv...) \
do { \
IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(IS_NULL(p), msg, rv); \
} while (0)
#define IF_BMP_FAILED_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(func, msg, rv...) \
do { \
enum bmp_result rc = func; \
IF_COND_TRUE_PRINT_ERR_MSG_AND_RETURN_OPT_VAL(BMP_FAILED(rc), msg, rv); \
} while (0)
#endif /* _BIRD_BMP_UTILS_H_ */