0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2025-01-12 12:01:54 +00:00

BMP: overflow checker made configurable

Also added documentation for these two options.
This commit is contained in:
Maria Matejka 2024-10-08 12:52:04 +02:00 committed by Maria Matejka
parent c972f54f00
commit 74566d3c85
4 changed files with 111 additions and 19 deletions

View File

@ -3739,6 +3739,22 @@ by default and have to be enabled during installation by the configure option
routes (in <ref id="bgp-import-table" name="BGP import tables">) and post-policy routes (in <ref id="bgp-import-table" name="BGP import tables">) and post-policy
routes (in regular routing tables). All BGP protocols are monitored automatically. routes (in regular routing tables). All BGP protocols are monitored automatically.
<sect1>Configuration (incomplete)
<label id="bmp-config">
<p><descrip>
<tag><label id="bmp-tx-buffer">tx buffer <m/number/</tag>
How big the TX buffer is for holding BMP data. Must be at least 16KB
long to have some slack for routes and open messages, as these may be
(post-policy) longer than the default 4KB. Recommended and default value
is 1 MB and you should not change it unless you know what you are doing.
<tag><label id="bmp-tx-buffer-count">tx buffer count <m/number/</tag>
How many TX buffers are held at most. This is effectively a multiplier
for the <cf/tx buffer/ option, capping the buffer sizes in case BMP
receiver gets clogged. When overflown, BMP restarts. Default: 1024.
</descrip>
<sect1>Example <sect1>Example
<label id="bmp-exam"> <label id="bmp-exam">
@ -3752,6 +3768,9 @@ protocol bmp {
# Monitor accepted routes (passed import filters) # Monitor accepted routes (passed import filters)
monitoring rib in post_policy; monitoring rib in post_policy;
# Allow only 64M of pending data
tx buffer count 64;
} }
</code> </code>

View File

@ -230,7 +230,14 @@ struct bmp_tx_buffer {
resource r; resource r;
node n; node n;
byte *end; byte *end;
byte buf[0xfff00]; byte buf[0];
};
struct bmp_tx_buffer_class {
resource r;
struct resclass class;
uint counter;
bool deprecated;
}; };
// Stores necessary any data in list // Stores necessary any data in list
@ -282,8 +289,33 @@ bmp_init_msg_serialize(buffer *stream, const char *sys_descr, const char *sys_na
bmp_info_tlv_hdr_serialize(stream, BMP_INFO_TLV_TYPE_SYS_NAME, sys_name); bmp_info_tlv_hdr_serialize(stream, BMP_INFO_TLV_TYPE_SYS_NAME, sys_name);
} }
static void btb_free(resource *r UNUSED) { static void
btbm_free(resource *r)
{
struct bmp_tx_buffer_class *btbc = SKIP_BACK(struct bmp_tx_buffer_class, r, r);
ASSERT_DIE(btbc->counter == 0);
}
static void
btbm_dump(resource *r)
{
struct bmp_tx_buffer_class *btbc = SKIP_BACK(struct bmp_tx_buffer_class, r, r);
debug("allocated %u blocks%s\n", btbc->counter, btbc->deprecated ? " (deprecated)" : "");
}
static struct resclass bmp_tx_buffer_metaclass = {
.name = "BMP TX buffer meta",
.size = sizeof(struct bmp_tx_buffer_class),
.free = btbm_free,
.dump = btbm_dump,
};
static void btb_free(resource *r) {
UNUSED struct bmp_tx_buffer *btb = SKIP_BACK(struct bmp_tx_buffer, r, r); UNUSED struct bmp_tx_buffer *btb = SKIP_BACK(struct bmp_tx_buffer, r, r);
struct bmp_tx_buffer_class *class = SKIP_BACK(struct bmp_tx_buffer_class, class, r->class);
ASSERT_DIE(class->counter);
if (!--class->counter && class->deprecated)
rfree(class);
} }
static void static void
@ -293,28 +325,37 @@ btb_dump(resource *r)
debug("used %u bytes\n", btb->end - btb->buf); debug("used %u bytes\n", btb->end - btb->buf);
} }
static uint
btb_bufsize(struct bmp_tx_buffer *btb)
{
return btb->r.class->size - sizeof *btb;
}
static struct resmem static struct resmem
btb_memsize(resource *r) btb_memsize(resource *r)
{ {
struct bmp_tx_buffer *btb = SKIP_BACK(struct bmp_tx_buffer, r, r); struct bmp_tx_buffer *btb = SKIP_BACK(struct bmp_tx_buffer, r, r);
return (struct resmem) { return (struct resmem) {
.effective = btb->end - btb->buf, .effective = btb->end - btb->buf,
.overhead = (sizeof *btb) - (btb->end - btb->buf), .overhead = r->class->size - (btb->end - btb->buf),
}; };
} }
static struct resclass bmp_tx_buffer_class = { static struct bmp_tx_buffer_class *
.name = "BMP TX buffer", btbm_new(struct bmp_proto *p, uint size)
.size = sizeof(struct bmp_tx_buffer),
.free = btb_free,
.dump = btb_dump,
.memsize = btb_memsize,
};
static bool
bmp_check_tx_overflow(u64 cnt)
{ {
return cnt * bmp_tx_buffer_class.size > (1ULL << 30); ASSERT_DIE(size > sizeof (struct bmp_tx_buffer) + 4096);
struct bmp_tx_buffer_class *c = ralloc(p->p.pool, &bmp_tx_buffer_metaclass);
c->class = (struct resclass) {
.name = "BMP TX buffer",
.size = size,
.free = btb_free,
.dump = btb_dump,
.memsize = btb_memsize,
};
c->counter = 0;
c->deprecated = false;
return c;
} }
static void static void
@ -326,15 +367,23 @@ bmp_schedule_tx_packet(struct bmp_proto *p, const byte *payload, const size_t si
struct bmp_tx_buffer *btb = EMPTY_LIST(p->tx_queue) ? NULL : struct bmp_tx_buffer *btb = EMPTY_LIST(p->tx_queue) ? NULL :
SKIP_BACK(struct bmp_tx_buffer, n, TAIL(p->tx_queue)); SKIP_BACK(struct bmp_tx_buffer, n, TAIL(p->tx_queue));
if (btb && (btb->end + size > btb->buf + sizeof btb->buf)) if (btb && (size + btb->end - btb->buf > btb_bufsize(btb)))
btb = NULL; btb = NULL;
if (size + sizeof *btb > p->tx_pcls->class.size)
{
log(L_ERR "%s: configured too small buffer (%u) for message size %lu",
p->tx_pcls->class.size, size + sizeof *btb);
return;
}
if (!btb) if (!btb)
{ {
if (bmp_check_tx_overflow(++p->tx_pending_count)) if (++p->tx_pending_count >= p->tx_pending_limit)
ev_schedule(p->tx_overflow_event); ev_schedule(p->tx_overflow_event);
btb = ralloc(p->tx_mem_pool, &bmp_tx_buffer_class); btb = ralloc(p->tx_mem_pool, &p->tx_pcls->class);
p->tx_pcls->counter++;
btb->end = btb->buf; btb->end = btb->buf;
btb->n = (node) {}; btb->n = (node) {};
add_tail(&p->tx_queue, &btb->n); add_tail(&p->tx_queue, &btb->n);
@ -1215,7 +1264,7 @@ static void
bmp_tx_overflow(void *_p) bmp_tx_overflow(void *_p)
{ {
struct bmp_proto *p = _p; struct bmp_proto *p = _p;
if (!bmp_check_tx_overflow(p->tx_pending_count)) if (p->tx_pending_count < p->tx_pending_limit)
return; return;
p->sock_err = 0; p->sock_err = 0;
@ -1306,6 +1355,8 @@ bmp_init(struct proto_config *CF)
strcpy(p->sys_name, cf->sys_name); strcpy(p->sys_name, cf->sys_name);
p->monitoring_rib.in_pre_policy = cf->monitoring_rib_in_pre_policy; p->monitoring_rib.in_pre_policy = cf->monitoring_rib_in_pre_policy;
p->monitoring_rib.in_post_policy = cf->monitoring_rib_in_post_policy; p->monitoring_rib.in_post_policy = cf->monitoring_rib_in_post_policy;
p->tx_pcls = btbm_new(p, cf->tx_bufsize);
p->tx_pending_limit = cf->tx_pending_limit;
return P; return P;
} }
@ -1381,6 +1432,15 @@ bmp_reconfigure(struct proto *P, struct proto_config *CF)
/* We must update our copy of configuration ptr */ /* We must update our copy of configuration ptr */
p->cf = new; p->cf = new;
/* Reconfigure tx buffer size limits */
if (new->tx_bufsize != p->tx_pcls->class.size)
{
p->tx_pcls->deprecated = true;
p->tx_pcls = btbm_new(p, new->tx_bufsize);
}
p->tx_pending_limit = new->tx_pending_limit;
return 1; return 1;
} }

View File

@ -38,6 +38,8 @@ struct bmp_config {
u16 station_port; // Monitoring station TCP port u16 station_port; // Monitoring station TCP port
bool monitoring_rib_in_pre_policy; // Route monitoring pre-policy Adj-Rib-In bool monitoring_rib_in_pre_policy; // Route monitoring pre-policy Adj-Rib-In
bool monitoring_rib_in_post_policy; // Route monitoring post-policy Adj-Rib-In bool monitoring_rib_in_post_policy; // Route monitoring post-policy Adj-Rib-In
uint tx_bufsize; // Pending TX buffer size
uint tx_pending_limit; // Maximum on pending TX buffer count
}; };
/* Forward declarations */ /* Forward declarations */
@ -68,7 +70,9 @@ struct bmp_proto {
pool *tx_mem_pool; // Memory pool used for packet allocations designated to BMP collector pool *tx_mem_pool; // Memory pool used for packet allocations designated to BMP collector
list tx_queue; // Stores queued packets going to be sent list tx_queue; // Stores queued packets going to be sent
struct bmp_tx_buffer *tx_pending;// This buffer waits for socket to flush struct bmp_tx_buffer *tx_pending;// This buffer waits for socket to flush
struct bmp_tx_buffer_class *tx_pcls; // Class used for tx_pending allocations
uint tx_pending_count; // How many buffers waiting for flush uint tx_pending_count; // How many buffers waiting for flush
uint tx_pending_limit; // Maximum on buffer count
event *tx_overflow_event; // Too many buffers waiting for flush event *tx_overflow_event; // Too many buffers waiting for flush
timer *connect_retry_timer; // Timer for retrying connection to the BMP collector timer *connect_retry_timer; // Timer for retrying connection to the BMP collector
bool started; // Flag that stores running status of BMP instance bool started; // Flag that stores running status of BMP instance
@ -98,7 +102,6 @@ struct bmp_table {
u32 uc; u32 uc;
}; };
#ifdef CONFIG_BMP #ifdef CONFIG_BMP
/** /**

View File

@ -27,6 +27,8 @@ bmp_proto_start: proto_start BMP {
this_proto = proto_config_new(&proto_bmp, $1); this_proto = proto_config_new(&proto_bmp, $1);
BMP_CFG->sys_descr = "Not defined"; BMP_CFG->sys_descr = "Not defined";
BMP_CFG->sys_name = "Not defined"; BMP_CFG->sys_name = "Not defined";
BMP_CFG->tx_bufsize = 1048576;
BMP_CFG->tx_pending_limit = 1024;
} }
; ;
@ -71,6 +73,14 @@ bmp_proto:
| bmp_proto MONITORING RIB IN POST_POLICY bool ';' { | bmp_proto MONITORING RIB IN POST_POLICY bool ';' {
BMP_CFG->monitoring_rib_in_post_policy = $6; BMP_CFG->monitoring_rib_in_post_policy = $6;
} }
| bmp_proto TX BUFFER expr ';' {
BMP_CFG->tx_bufsize = $4;
if ($4 < 16384) cf_error("BMP TX buffer must be at least 16384 bytes long");
}
| bmp_proto TX BUFFER COUNT expr ';' {
BMP_CFG->tx_pending_limit = $5;
if ($5 < 2) cf_error("BMP must allow at least 2 pending TX buffers");
}
; ;
CF_CODE CF_CODE