mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-10 19:11:54 +00:00
59a5bf18f9
Some CLI actions, notably "show route", are run by queuing an event somewhere else. If the user closes the socket, in case such an action is being executed, the CLI must free the socket immediately from the error hook but the pool must remain until the asynchronous event finishes and cleans everything up.
484 lines
12 KiB
C
484 lines
12 KiB
C
/*
|
|
* BIRD -- Route Display Routines
|
|
*
|
|
* (c) 1998--2000 Martin Mares <mj@ucw.cz>
|
|
* (c) 2017 Jan Moskyto Matejka <mq@jmq.cz>
|
|
*
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
|
*/
|
|
|
|
#undef LOCAL_DEBUG
|
|
|
|
#include "nest/bird.h"
|
|
#include "nest/rt.h"
|
|
#include "nest/protocol.h"
|
|
#include "nest/cli.h"
|
|
#include "nest/iface.h"
|
|
#include "filter/filter.h"
|
|
#include "filter/data.h"
|
|
#include "sysdep/unix/krt.h"
|
|
|
|
static void
|
|
rt_show_table(struct rt_show_data *d)
|
|
{
|
|
struct cli *c = d->cli;
|
|
|
|
/* No table blocks in 'show route count' */
|
|
if (d->stats == 2)
|
|
return;
|
|
|
|
if (d->last_table) cli_printf(c, -1007, "");
|
|
cli_printf(c, -1007, "Table %s:",
|
|
d->tab->name);
|
|
d->last_table = d->tab;
|
|
}
|
|
|
|
static void
|
|
rt_show_rte(struct cli *c, byte *ia, rte *e, struct rt_show_data *d, int primary)
|
|
{
|
|
byte from[IPA_MAX_TEXT_LENGTH+8];
|
|
byte tm[TM_DATETIME_BUFFER_SIZE], info[256];
|
|
ea_list *a = e->attrs;
|
|
int sync_error = d->tab->kernel ? krt_get_sync_error(d->tab->kernel, e) : 0;
|
|
void (*get_route_info)(struct rte *, byte *buf);
|
|
eattr *nhea = net_type_match(e->net, NB_DEST) ?
|
|
ea_find(a, &ea_gen_nexthop) : NULL;
|
|
struct nexthop_adata *nhad = nhea ? (struct nexthop_adata *) nhea->u.ptr : NULL;
|
|
int dest = nhad ? (NEXTHOP_IS_REACHABLE(nhad) ? RTD_UNICAST : nhad->dest) : RTD_NONE;
|
|
int flowspec_valid = net_is_flow(e->net) ? rt_get_flowspec_valid(e) : FLOWSPEC_UNKNOWN;
|
|
|
|
tm_format_time(tm, &config->tf_route, e->lastmod);
|
|
ip_addr a_from = ea_get_ip(a, &ea_gen_from, IPA_NONE);
|
|
if (ipa_nonzero(a_from) && (!nhad || !ipa_equal(a_from, nhad->nh.gw)))
|
|
bsprintf(from, " from %I", a_from);
|
|
else
|
|
from[0] = 0;
|
|
|
|
/* Need to normalize the extended attributes */
|
|
if (d->verbose && !rta_is_cached(a) && a)
|
|
a = ea_normalize(a, 0);
|
|
|
|
get_route_info = e->src->owner->class ? e->src->owner->class->get_route_info : NULL;
|
|
if (get_route_info)
|
|
get_route_info(e, info);
|
|
else
|
|
bsprintf(info, " (%d)", rt_get_preference(e));
|
|
|
|
if (d->last_table != d->tab)
|
|
rt_show_table(d);
|
|
|
|
eattr *heea;
|
|
struct hostentry_adata *had = NULL;
|
|
if (!net_is_flow(e->net) && (dest == RTD_NONE) && (heea = ea_find(a, &ea_gen_hostentry)))
|
|
had = (struct hostentry_adata *) heea->u.ptr;
|
|
|
|
cli_printf(c, -1007, "%-20s %s [%s %s%s]%s%s", ia,
|
|
net_is_flow(e->net) ? flowspec_valid_name(flowspec_valid) : had ? "recursive" : rta_dest_name(dest),
|
|
e->src->owner->name, tm, from, primary ? (sync_error ? " !" : " *") : "", info);
|
|
|
|
if (d->verbose)
|
|
{
|
|
ea_show_list(c, a);
|
|
cli_printf(c, -1008, "\tInternal route handling values: %uL %uG %uS id %u",
|
|
e->src->private_id, e->src->global_id, e->stale_cycle, e->id);
|
|
}
|
|
else if (dest == RTD_UNICAST)
|
|
ea_show_nexthop_list(c, nhad);
|
|
else if (had)
|
|
{
|
|
char hetext[256];
|
|
ea_show_hostentry(&had->ad, hetext, sizeof hetext);
|
|
cli_printf(c, -1007, "\t%s", hetext);
|
|
}
|
|
}
|
|
|
|
static void
|
|
rt_show_net(struct rt_show_data *d, const net_addr *n, rte **feed, uint count)
|
|
{
|
|
struct cli *c = d->cli;
|
|
byte ia[NET_MAX_TEXT_LENGTH+1];
|
|
struct channel *ec = d->tab->export_channel;
|
|
|
|
/* The Clang static analyzer complains that ec may be NULL.
|
|
* It should be ensured to be not NULL by rt_show_prepare_tables() */
|
|
ASSUME(!d->export_mode || ec);
|
|
|
|
int first = 1;
|
|
int first_show = 1;
|
|
int pass = 0;
|
|
|
|
for (uint i = 0; i < count; i++)
|
|
{
|
|
if (!d->tab->prefilter && (rte_is_filtered(feed[i]) != d->filtered))
|
|
continue;
|
|
|
|
d->rt_counter++;
|
|
d->net_counter += first;
|
|
first = 0;
|
|
|
|
if (pass)
|
|
continue;
|
|
|
|
struct rte e = *feed[i];
|
|
if (d->tab->prefilter)
|
|
if (e.sender != d->tab->prefilter->in_req.hook)
|
|
continue;
|
|
else while (e.attrs->next)
|
|
e.attrs = e.attrs->next;
|
|
|
|
/* Export channel is down, do not try to export routes to it */
|
|
if (ec && !ec->out_req.hook)
|
|
goto skip;
|
|
|
|
if (d->export_mode == RSEM_EXPORTED)
|
|
{
|
|
if (!bmap_test(&ec->export_map, e.id))
|
|
goto skip;
|
|
|
|
// if (ec->ra_mode != RA_ANY)
|
|
// pass = 1;
|
|
}
|
|
else if ((d->export_mode == RSEM_EXPORT) && (ec->ra_mode == RA_MERGED))
|
|
{
|
|
/* Special case for merged export */
|
|
pass = 1;
|
|
rte *em = rt_export_merged(ec, feed, count, tmp_linpool, 1);
|
|
|
|
if (em)
|
|
e = *em;
|
|
else
|
|
goto skip;
|
|
}
|
|
else if (d->export_mode)
|
|
{
|
|
struct proto *ep = ec->proto;
|
|
int ic = ep->preexport ? ep->preexport(ec, &e) : 0;
|
|
|
|
if (ec->ra_mode == RA_OPTIMAL || ec->ra_mode == RA_MERGED)
|
|
pass = 1;
|
|
|
|
if (ic < 0)
|
|
goto skip;
|
|
|
|
if (d->export_mode > RSEM_PREEXPORT)
|
|
{
|
|
/*
|
|
* FIXME - This shows what should be exported according to current
|
|
* filters, but not what was really exported. 'configure soft'
|
|
* command may change the export filter and do not update routes.
|
|
*/
|
|
int do_export = (ic > 0) ||
|
|
(f_run(ec->out_filter, &e, FF_SILENT) <= F_ACCEPT);
|
|
|
|
if (do_export != (d->export_mode == RSEM_EXPORT))
|
|
goto skip;
|
|
|
|
if ((d->export_mode == RSEM_EXPORT) && (ec->ra_mode == RA_ACCEPTED))
|
|
pass = 1;
|
|
}
|
|
}
|
|
|
|
if (d->show_protocol && (&d->show_protocol->sources != e.src->owner))
|
|
goto skip;
|
|
|
|
if (f_run(d->filter, &e, 0) > F_ACCEPT)
|
|
goto skip;
|
|
|
|
if (d->stats < 2)
|
|
{
|
|
if (first_show)
|
|
net_format(n, ia, sizeof(ia));
|
|
else
|
|
ia[0] = 0;
|
|
|
|
rt_show_rte(c, ia, &e, d, !d->tab->prefilter && !i);
|
|
first_show = 0;
|
|
}
|
|
|
|
d->show_counter++;
|
|
|
|
skip:
|
|
if (d->primary_only)
|
|
break;
|
|
}
|
|
|
|
if ((d->show_counter - d->show_counter_last_flush) > 64)
|
|
{
|
|
d->show_counter_last_flush = d->show_counter;
|
|
cli_write_trigger(d->cli);
|
|
}
|
|
}
|
|
|
|
static void
|
|
rt_show_net_export_bulk(struct rt_export_request *req, const net_addr *n,
|
|
struct rt_pending_export *rpe UNUSED, rte **feed, uint count)
|
|
{
|
|
struct rt_show_data *d = SKIP_BACK(struct rt_show_data, req, req);
|
|
return rt_show_net(d, n, feed, count);
|
|
}
|
|
|
|
static void
|
|
rt_show_export_stopped_cleanup(struct rt_export_request *req)
|
|
{
|
|
struct rt_show_data *d = SKIP_BACK(struct rt_show_data, req, req);
|
|
|
|
/* The hook is now invalid */
|
|
req->hook = NULL;
|
|
|
|
/* And free the CLI (deferred) */
|
|
rfree(d->cli->pool);
|
|
}
|
|
|
|
static int
|
|
rt_show_cleanup(struct cli *c)
|
|
{
|
|
struct rt_show_data *d = c->rover;
|
|
c->cleanup = NULL;
|
|
|
|
/* Cancel the feed */
|
|
if (d->req.hook)
|
|
{
|
|
rt_stop_export(&d->req, rt_show_export_stopped_cleanup);
|
|
return 1;
|
|
}
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
static void rt_show_export_stopped(struct rt_export_request *req);
|
|
|
|
static void
|
|
rt_show_log_state_change(struct rt_export_request *req, u8 state)
|
|
{
|
|
if (state == TES_READY)
|
|
rt_stop_export(req, rt_show_export_stopped);
|
|
}
|
|
|
|
static void
|
|
rt_show_dump_req(struct rt_export_request *req)
|
|
{
|
|
debug(" CLI Show Route Feed %p\n", req);
|
|
}
|
|
|
|
static void
|
|
rt_show_done(struct rt_show_data *d)
|
|
{
|
|
/* No more action */
|
|
d->cli->cleanup = NULL;
|
|
d->cli->cont = NULL;
|
|
d->cli->rover = NULL;
|
|
|
|
/* Write pending messages */
|
|
cli_write_trigger(d->cli);
|
|
}
|
|
|
|
static void
|
|
rt_show_cont(struct rt_show_data *d)
|
|
{
|
|
struct cli *c = d->cli;
|
|
|
|
if (d->running_on_config && (d->running_on_config != config))
|
|
{
|
|
cli_printf(c, 8004, "Stopped due to reconfiguration");
|
|
return rt_show_done(d);
|
|
}
|
|
|
|
d->req = (struct rt_export_request) {
|
|
.addr = d->addr,
|
|
.name = "CLI Show Route",
|
|
.list = &global_work_list,
|
|
.export_bulk = rt_show_net_export_bulk,
|
|
.dump_req = rt_show_dump_req,
|
|
.log_state_change = rt_show_log_state_change,
|
|
.addr_mode = d->addr_mode,
|
|
};
|
|
|
|
d->table_counter++;
|
|
|
|
d->show_counter_last = d->show_counter;
|
|
d->rt_counter_last = d->rt_counter;
|
|
d->net_counter_last = d->net_counter;
|
|
|
|
if (d->tables_defined_by & RSD_TDB_SET)
|
|
rt_show_table(d);
|
|
|
|
rt_request_export_other(d->tab->table, &d->req);
|
|
}
|
|
|
|
static void
|
|
rt_show_export_stopped(struct rt_export_request *req)
|
|
{
|
|
struct rt_show_data *d = SKIP_BACK(struct rt_show_data, req, req);
|
|
|
|
/* The hook is now invalid */
|
|
req->hook = NULL;
|
|
|
|
if (d->stats)
|
|
{
|
|
if (d->last_table != d->tab)
|
|
rt_show_table(d);
|
|
|
|
cli_printf(d->cli, -1007, "%d of %d routes for %d networks in table %s",
|
|
d->show_counter - d->show_counter_last, d->rt_counter - d->rt_counter_last,
|
|
d->net_counter - d->net_counter_last, d->tab->name);
|
|
}
|
|
|
|
d->tab = NODE_NEXT(d->tab);
|
|
|
|
if (NODE_VALID(d->tab))
|
|
return rt_show_cont(d);
|
|
|
|
/* Printout total stats */
|
|
if (d->stats && (d->table_counter > 1))
|
|
{
|
|
if (d->last_table) cli_printf(d->cli, -1007, "");
|
|
cli_printf(d->cli, 14, "Total: %d of %d routes for %d networks in %d tables",
|
|
d->show_counter, d->rt_counter, d->net_counter, d->table_counter);
|
|
}
|
|
else if (!d->rt_counter && ((d->addr_mode == TE_ADDR_EQUAL) || (d->addr_mode == TE_ADDR_FOR)))
|
|
cli_printf(d->cli, 8001, "Network not found");
|
|
else
|
|
cli_printf(d->cli, 0, "");
|
|
|
|
/* No more route showing */
|
|
rt_show_done(d);
|
|
}
|
|
|
|
struct rt_show_data_rtable *
|
|
rt_show_add_exporter(struct rt_show_data *d, struct rt_exporter *t, const char *name)
|
|
{
|
|
struct rt_show_data_rtable *tab = cfg_allocz(sizeof(struct rt_show_data_rtable));
|
|
tab->table = t;
|
|
tab->name = name;
|
|
add_tail(&(d->tables), &(tab->n));
|
|
return tab;
|
|
}
|
|
|
|
struct rt_show_data_rtable *
|
|
rt_show_add_table(struct rt_show_data *d, rtable *t)
|
|
{
|
|
struct rt_show_data_rtable *rsdr;
|
|
RT_LOCKED(t, tp)
|
|
rsdr = rt_show_add_exporter(d, &tp->exporter.e, t->name);
|
|
|
|
struct proto_config *krt = t->config->krt_attached;
|
|
if (krt)
|
|
rsdr->kernel = (struct krt_proto *) krt->proto;
|
|
|
|
return rsdr;
|
|
}
|
|
|
|
static inline void
|
|
rt_show_get_default_tables(struct rt_show_data *d)
|
|
{
|
|
struct channel *c;
|
|
struct rt_show_data_rtable *tab;
|
|
|
|
if (d->export_channel)
|
|
{
|
|
c = d->export_channel;
|
|
tab = rt_show_add_table(d, c->table);
|
|
tab->export_channel = c;
|
|
return;
|
|
}
|
|
|
|
if (d->export_protocol)
|
|
{
|
|
WALK_LIST(c, d->export_protocol->channels)
|
|
{
|
|
if (!c->out_req.hook)
|
|
continue;
|
|
|
|
tab = rt_show_add_table(d, c->table);
|
|
tab->export_channel = c;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (d->show_protocol)
|
|
{
|
|
WALK_LIST(c, d->show_protocol->channels)
|
|
rt_show_add_table(d, c->table);
|
|
return;
|
|
}
|
|
|
|
for (int i=1; i<NET_MAX; i++)
|
|
if (config->def_tables[i] && config->def_tables[i]->table && config->def_tables[i]->table->table)
|
|
rt_show_add_table(d, config->def_tables[i]->table->table);
|
|
}
|
|
|
|
static inline void
|
|
rt_show_prepare_tables(struct rt_show_data *d)
|
|
{
|
|
struct rt_show_data_rtable *tab, *tabx;
|
|
|
|
/* Add implicit tables if no table is specified */
|
|
if (EMPTY_LIST(d->tables))
|
|
rt_show_get_default_tables(d);
|
|
|
|
WALK_LIST_DELSAFE(tab, tabx, d->tables)
|
|
{
|
|
/* Ensure there is defined export_channel for each table */
|
|
if (d->export_mode)
|
|
{
|
|
rtable *rt = SKIP_BACK(rtable, priv.exporter.e, tab->table);
|
|
if (!tab->export_channel && d->export_channel &&
|
|
(rt == d->export_channel->table))
|
|
tab->export_channel = d->export_channel;
|
|
|
|
if (!tab->export_channel && d->export_protocol)
|
|
tab->export_channel = proto_find_channel_by_table(d->export_protocol, rt);
|
|
|
|
if (!tab->export_channel)
|
|
{
|
|
if (d->tables_defined_by & RSD_TDB_NMN)
|
|
cf_error("No export channel for table %s", tab->name);
|
|
|
|
rem_node(&(tab->n));
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* Ensure specified network is compatible with each table */
|
|
if (d->addr && (tab->table->addr_type != d->addr->type))
|
|
{
|
|
if (d->tables_defined_by & RSD_TDB_NMN)
|
|
cf_error("Incompatible type of prefix/ip for table %s", tab->name);
|
|
|
|
rem_node(&(tab->n));
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* Ensure there is at least one table */
|
|
if (EMPTY_LIST(d->tables))
|
|
cf_error("No valid tables");
|
|
}
|
|
|
|
static void
|
|
rt_show_dummy_cont(struct cli *c UNUSED)
|
|
{
|
|
/* Explicitly do nothing to prevent CLI from trying to parse another command. */
|
|
}
|
|
|
|
void
|
|
rt_show(struct rt_show_data *d)
|
|
{
|
|
/* Filtered routes are neither exported nor have sensible ordering */
|
|
if (d->filtered && (d->export_mode || d->primary_only))
|
|
cf_error("Incompatible show route options");
|
|
|
|
rt_show_prepare_tables(d);
|
|
|
|
if (EMPTY_LIST(d->tables))
|
|
cf_error("No suitable tables found");
|
|
|
|
d->tab = HEAD(d->tables);
|
|
|
|
this_cli->cleanup = rt_show_cleanup;
|
|
this_cli->rover = d;
|
|
this_cli->cont = rt_show_dummy_cont;
|
|
|
|
rt_show_cont(d);
|
|
}
|