2019-01-21 08:17:54 +00:00
|
|
|
/*
|
|
|
|
* BIRD Internet Routing Daemon -- Filter instructions
|
|
|
|
*
|
2019-02-07 20:25:38 +00:00
|
|
|
* (c) 1999 Pavel Machek <pavel@ucw.cz>
|
2019-01-21 08:17:54 +00:00
|
|
|
* (c) 2018--2019 Maria Matejka <mq@jmq.cz>
|
|
|
|
*
|
|
|
|
* Can be freely distributed and used under the terms of the GNU GPL.
|
2019-03-06 14:01:10 +00:00
|
|
|
*
|
|
|
|
* Filter interpreter data structures and internal API.
|
2019-07-15 11:19:01 +00:00
|
|
|
* See filter/f-inst.c for documentation.
|
2019-01-21 08:17:54 +00:00
|
|
|
*/
|
|
|
|
|
2019-02-07 20:25:38 +00:00
|
|
|
#ifndef _BIRD_F_INST_H_
|
|
|
|
#define _BIRD_F_INST_H_
|
|
|
|
|
2019-02-08 12:38:12 +00:00
|
|
|
#include "nest/bird.h"
|
|
|
|
#include "conf/conf.h"
|
2019-02-07 20:25:38 +00:00
|
|
|
#include "filter/filter.h"
|
2019-02-08 12:38:12 +00:00
|
|
|
#include "filter/data.h"
|
2021-02-07 18:21:42 +00:00
|
|
|
#include "lib/buffer.h"
|
2019-12-09 03:23:01 +00:00
|
|
|
#include "lib/flowspec.h"
|
2019-02-07 20:25:38 +00:00
|
|
|
|
2019-02-15 22:59:44 +00:00
|
|
|
/* Flags for instructions */
|
|
|
|
enum f_instruction_flags {
|
2022-03-06 01:18:01 +00:00
|
|
|
FIF_RECURSIVE = 1, /* FI_CALL: function is directly recursive */
|
2019-02-15 22:59:44 +00:00
|
|
|
} PACKED;
|
|
|
|
|
2019-02-08 12:38:12 +00:00
|
|
|
/* Include generated filter instruction declarations */
|
2019-02-11 15:44:14 +00:00
|
|
|
#include "filter/inst-gen.h"
|
2019-02-07 20:25:38 +00:00
|
|
|
|
2019-02-08 12:38:12 +00:00
|
|
|
#define f_new_inst(...) MACRO_CONCAT_AFTER(f_new_inst_, MACRO_FIRST(__VA_ARGS__))(__VA_ARGS__)
|
2019-01-21 08:17:54 +00:00
|
|
|
|
|
|
|
/* Convert the instruction back to the enum name */
|
2019-11-05 14:13:57 +00:00
|
|
|
const char *f_instruction_name_(enum f_instruction_code fi);
|
|
|
|
static inline const char *f_instruction_name(enum f_instruction_code fi)
|
|
|
|
{ return f_instruction_name_(fi) + 3; }
|
2019-01-21 08:17:54 +00:00
|
|
|
|
2022-03-03 02:38:12 +00:00
|
|
|
struct f_arg {
|
|
|
|
struct symbol *arg;
|
|
|
|
struct f_arg *next;
|
|
|
|
};
|
|
|
|
|
2019-02-07 20:25:38 +00:00
|
|
|
/* Filter structures for execution */
|
|
|
|
/* Line of instructions to be unconditionally executed one after another */
|
|
|
|
struct f_line {
|
|
|
|
uint len; /* Line length */
|
2019-02-26 15:44:24 +00:00
|
|
|
u8 args; /* Function: Args required */
|
2019-05-21 16:33:37 +00:00
|
|
|
u8 vars;
|
2022-03-09 01:32:29 +00:00
|
|
|
u8 results; /* Results left on stack: cmd -> 0, term -> 1 */
|
2022-03-03 02:38:12 +00:00
|
|
|
struct f_arg *arg_list;
|
2019-02-07 20:25:38 +00:00
|
|
|
struct f_line_item items[0]; /* The items themselves */
|
|
|
|
};
|
2019-01-21 08:17:54 +00:00
|
|
|
|
2019-02-07 20:25:38 +00:00
|
|
|
/* Convert the f_inst infix tree to the f_line structures */
|
2022-03-09 01:32:29 +00:00
|
|
|
struct f_line *f_linearize_concat(const struct f_inst * const inst[], uint count, uint results);
|
|
|
|
static inline struct f_line *f_linearize(const struct f_inst *root, uint results)
|
|
|
|
{ return f_linearize_concat(&root, 1, results); }
|
2019-02-07 20:25:38 +00:00
|
|
|
|
2019-02-15 22:59:44 +00:00
|
|
|
void f_dump_line(const struct f_line *, uint indent);
|
|
|
|
|
2021-02-07 18:21:42 +00:00
|
|
|
|
|
|
|
/* Recursive iteration over filter instructions */
|
|
|
|
|
|
|
|
struct filter_iterator {
|
|
|
|
BUFFER_(const struct f_line *) lines;
|
|
|
|
};
|
|
|
|
|
|
|
|
void f_add_lines(const struct f_line_item *what, struct filter_iterator *fit);
|
|
|
|
|
|
|
|
#define FILTER_ITERATE_INIT(fit, filter, pool) \
|
|
|
|
({ \
|
|
|
|
BUFFER_INIT((fit)->lines, (pool), 32); \
|
|
|
|
BUFFER_PUSH((fit)->lines) = (filter)->root; \
|
|
|
|
})
|
|
|
|
|
|
|
|
#define FILTER_ITERATE(fit, fi) ({ \
|
|
|
|
const struct f_line *fl_; \
|
|
|
|
while (!BUFFER_EMPTY((fit)->lines)) \
|
|
|
|
{ \
|
|
|
|
BUFFER_POP((fit)->lines); \
|
|
|
|
fl_ = (fit)->lines.data[(fit)->lines.used]; \
|
|
|
|
for (uint i_ = 0; i_ < fl_->len; i_++) \
|
|
|
|
{ \
|
|
|
|
const struct f_line_item *fi = &fl_->items[i_]; \
|
|
|
|
f_add_lines(fi, (fit));
|
|
|
|
|
|
|
|
#define FILTER_ITERATE_END } } })
|
|
|
|
|
|
|
|
#define FILTER_ITERATE_CLEANUP(fit) \
|
|
|
|
({ \
|
|
|
|
mb_free((fit)->lines.data); \
|
|
|
|
memset((fit), 0, sizeof(struct filter_iterator)); \
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2019-06-28 09:08:48 +00:00
|
|
|
struct filter *f_new_where(struct f_inst *);
|
2022-03-26 10:56:02 +00:00
|
|
|
static inline struct f_static_attr f_new_static_attr(btype type, int code, int readonly)
|
|
|
|
{ return (struct f_static_attr) { .type = type, .sa_code = code, .readonly = readonly }; }
|
2019-02-07 20:25:38 +00:00
|
|
|
struct f_inst *f_generate_roa_check(struct rtable_config *table, struct f_inst *prefix, struct f_inst *asn);
|
|
|
|
|
2022-03-19 15:23:42 +00:00
|
|
|
struct f_attr_bit {
|
|
|
|
const struct ea_class *class;
|
|
|
|
uint bit;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define f_new_dynamic_attr_bit(_bit, _name) ((struct f_attr_bit) { .bit = _bit, .class = ea_class_find(_name) })
|
|
|
|
|
2019-02-07 20:25:38 +00:00
|
|
|
/* Hook for call bt_assert() function in configuration */
|
|
|
|
extern void (*bt_assert_hook)(int result, const struct f_line_item *assert);
|
|
|
|
|
|
|
|
/* Bird Tests */
|
|
|
|
struct f_bt_test_suite {
|
|
|
|
node n; /* Node in config->tests */
|
2019-02-15 12:53:17 +00:00
|
|
|
const struct f_line *fn; /* Root of function */
|
|
|
|
const struct f_line *cmp; /* Compare to this function */
|
2019-02-07 20:25:38 +00:00
|
|
|
const char *fn_name; /* Name of test */
|
|
|
|
const char *dsc; /* Description */
|
2019-02-13 11:25:30 +00:00
|
|
|
int result; /* Desired result */
|
2019-01-21 08:17:54 +00:00
|
|
|
};
|
|
|
|
|
2019-02-07 20:25:38 +00:00
|
|
|
#endif
|