Annotation of embedaddon/bird2/nest/cli.c, revision 1.1.1.1

1.1       misho       1: /*
                      2:  *     BIRD Internet Routing Daemon -- Command-Line Interface
                      3:  *
                      4:  *     (c) 1999--2000 Martin Mares <mj@ucw.cz>
                      5:  *
                      6:  *     Can be freely distributed and used under the terms of the GNU GPL.
                      7:  */
                      8: 
                      9: /**
                     10:  * DOC: Command line interface
                     11:  *
                     12:  * This module takes care of the BIRD's command-line interface (CLI).
                     13:  * The CLI exists to provide a way to control BIRD remotely and to inspect
                     14:  * its status. It uses a very simple textual protocol over a stream
                     15:  * connection provided by the platform dependent code (on UNIX systems,
                     16:  * it's a UNIX domain socket).
                     17:  *
                     18:  * Each session of the CLI consists of a sequence of request and replies,
                     19:  * slightly resembling the FTP and SMTP protocols.
                     20:  * Requests are commands encoded as a single line of text, replies are
                     21:  * sequences of lines starting with a four-digit code followed by either
                     22:  * a space (if it's the last line of the reply) or a minus sign (when the
                     23:  * reply is going to continue with the next line), the rest of the line
                     24:  * contains a textual message semantics of which depends on the numeric
                     25:  * code. If a reply line has the same code as the previous one and it's
                     26:  * a continuation line, the whole prefix can be replaced by a single
                     27:  * white space character.
                     28:  *
                     29:  * Reply codes starting with 0 stand for `action successfully completed' messages,
                     30:  * 1 means `table entry', 8 `runtime error' and 9 `syntax error'.
                     31:  *
                     32:  * Each CLI session is internally represented by a &cli structure and a
                     33:  * resource pool containing all resources associated with the connection,
                     34:  * so that it can be easily freed whenever the connection gets closed, not depending
                     35:  * on the current state of command processing.
                     36:  *
                     37:  * The CLI commands are declared as a part of the configuration grammar
                     38:  * by using the |CF_CLI| macro. When a command is received, it is processed
                     39:  * by the same lexical analyzer and parser as used for the configuration, but
                     40:  * it's switched to a special mode by prepending a fake token to the text,
                     41:  * so that it uses only the CLI command rules. Then the parser invokes
                     42:  * an execution routine corresponding to the command, which either constructs
                     43:  * the whole reply and returns it back or (in case it expects the reply will be long)
                     44:  * it prints a partial reply and asks the CLI module (using the @cont hook)
                     45:  * to call it again when the output is transferred to the user.
                     46:  *
                     47:  * The @this_cli variable points to a &cli structure of the session being
                     48:  * currently parsed, but it's of course available only in command handlers
                     49:  * not entered using the @cont hook.
                     50:  *
                     51:  * TX buffer management works as follows: At cli.tx_buf there is a
                     52:  * list of TX buffers (struct cli_out), cli.tx_write is the buffer
                     53:  * currently used by the producer (cli_printf(), cli_alloc_out()) and
                     54:  * cli.tx_pos is the buffer currently used by the consumer
                     55:  * (cli_write(), in system dependent code). The producer uses
                     56:  * cli_out.wpos ptr as the current write position and the consumer
                     57:  * uses cli_out.outpos ptr as the current read position. When the
                     58:  * producer produces something, it calls cli_write_trigger(). If there
                     59:  * is not enough space in the current buffer, the producer allocates
                     60:  * the new one. When the consumer processes everything in the buffer
                     61:  * queue, it calls cli_written(), tha frees all buffers (except the
                     62:  * first one) and schedules cli.event .
                     63:  *
                     64:  */
                     65: 
                     66: #include "nest/bird.h"
                     67: #include "nest/cli.h"
                     68: #include "conf/conf.h"
                     69: #include "lib/string.h"
                     70: 
                     71: pool *cli_pool;
                     72: 
                     73: static byte *
                     74: cli_alloc_out(cli *c, int size)
                     75: {
                     76:   struct cli_out *o;
                     77: 
                     78:   if (!(o = c->tx_write) || o->wpos + size > o->end)
                     79:     {
                     80:       if (!o && c->tx_buf)
                     81:        o = c->tx_buf;
                     82:       else
                     83:        {
                     84:          o = mb_alloc(c->pool, sizeof(struct cli_out) + CLI_TX_BUF_SIZE);
                     85:          if (c->tx_write)
                     86:            c->tx_write->next = o;
                     87:          else
                     88:            c->tx_buf = o;
                     89:          o->wpos = o->outpos = o->buf;
                     90:          o->end = o->buf + CLI_TX_BUF_SIZE;
                     91:        }
                     92:       c->tx_write = o;
                     93:       if (!c->tx_pos)
                     94:        c->tx_pos = o;
                     95:       o->next = NULL;
                     96:     }
                     97:   o->wpos += size;
                     98:   return o->wpos - size;
                     99: }
                    100: 
                    101: /**
                    102:  * cli_printf - send reply to a CLI connection
                    103:  * @c: CLI connection
                    104:  * @code: numeric code of the reply, negative for continuation lines
                    105:  * @msg: a printf()-like formatting string.
                    106:  *
                    107:  * This function send a single line of reply to a given CLI connection.
                    108:  * In works in all aspects like bsprintf() except that it automatically
                    109:  * prepends the reply line prefix.
                    110:  *
                    111:  * Please note that if the connection can be already busy sending some
                    112:  * data in which case cli_printf() stores the output to a temporary buffer,
                    113:  * so please avoid sending a large batch of replies without waiting
                    114:  * for the buffers to be flushed.
                    115:  *
                    116:  * If you want to write to the current CLI output, you can use the cli_msg()
                    117:  * macro instead.
                    118:  */
                    119: void
                    120: cli_printf(cli *c, int code, char *msg, ...)
                    121: {
                    122:   va_list args;
                    123:   byte buf[CLI_LINE_SIZE];
                    124:   int cd = code;
                    125:   int errcode;
                    126:   int size, cnt;
                    127: 
                    128:   if (cd < 0)
                    129:     {
                    130:       cd = -cd;
                    131:       if (cd == c->last_reply)
                    132:        size = bsprintf(buf, " ");
                    133:       else
                    134:        size = bsprintf(buf, "%04d-", cd);
                    135:       errcode = -8000;
                    136:     }
                    137:   else if (cd == CLI_ASYNC_CODE)
                    138:     {
                    139:       size = 1; buf[0] = '+';
                    140:       errcode = cd;
                    141:     }
                    142:   else
                    143:     {
                    144:       size = bsprintf(buf, "%04d ", cd);
                    145:       errcode = 8000;
                    146:     }
                    147: 
                    148:   c->last_reply = cd;
                    149:   va_start(args, msg);
                    150:   cnt = bvsnprintf(buf+size, sizeof(buf)-size-1, msg, args);
                    151:   va_end(args);
                    152:   if (cnt < 0)
                    153:     {
                    154:       cli_printf(c, errcode, "<line overflow>");
                    155:       return;
                    156:     }
                    157:   size += cnt;
                    158:   buf[size++] = '\n';
                    159:   memcpy(cli_alloc_out(c, size), buf, size);
                    160: }
                    161: 
                    162: static void
                    163: cli_copy_message(cli *c)
                    164: {
                    165:   byte *p, *q;
                    166:   uint cnt = 2;
                    167: 
                    168:   if (c->ring_overflow)
                    169:     {
                    170:       byte buf[64];
                    171:       int n = bsprintf(buf, "<%d messages lost>\n", c->ring_overflow);
                    172:       c->ring_overflow = 0;
                    173:       memcpy(cli_alloc_out(c, n), buf, n);
                    174:     }
                    175:   p = c->ring_read;
                    176:   while (*p)
                    177:     {
                    178:       cnt++;
                    179:       p++;
                    180:       if (p == c->ring_end)
                    181:        p = c->ring_buf;
                    182:       ASSERT(p != c->ring_write);
                    183:     }
                    184:   c->async_msg_size += cnt;
                    185:   q = cli_alloc_out(c, cnt);
                    186:   *q++ = '+';
                    187:   p = c->ring_read;
                    188:   do
                    189:     {
                    190:       *q = *p++;
                    191:       if (p == c->ring_end)
                    192:        p = c->ring_buf;
                    193:     }
                    194:   while (*q++);
                    195:   c->ring_read = p;
                    196:   q[-1] = '\n';
                    197: }
                    198: 
                    199: static void
                    200: cli_hello(cli *c)
                    201: {
                    202:   cli_printf(c, 1, "BIRD " BIRD_VERSION " ready.");
                    203:   c->cont = NULL;
                    204: }
                    205: 
                    206: static void
                    207: cli_free_out(cli *c)
                    208: {
                    209:   struct cli_out *o, *p;
                    210: 
                    211:   if (o = c->tx_buf)
                    212:     {
                    213:       o->wpos = o->outpos = o->buf;
                    214:       while (p = o->next)
                    215:        {
                    216:          o->next = p->next;
                    217:          mb_free(p);
                    218:        }
                    219:     }
                    220:   c->tx_write = c->tx_pos = NULL;
                    221:   c->async_msg_size = 0;
                    222: }
                    223: 
                    224: void
                    225: cli_written(cli *c)
                    226: {
                    227:   cli_free_out(c);
                    228:   ev_schedule(c->event);
                    229: }
                    230: 
                    231: 
                    232: static byte *cli_rh_pos;
                    233: static uint cli_rh_len;
                    234: static int cli_rh_trick_flag;
                    235: struct cli *this_cli;
                    236: 
                    237: static int
                    238: cli_cmd_read_hook(byte *buf, uint max, UNUSED int fd)
                    239: {
                    240:   if (!cli_rh_trick_flag)
                    241:     {
                    242:       cli_rh_trick_flag = 1;
                    243:       buf[0] = '!';
                    244:       return 1;
                    245:     }
                    246:   if (max > cli_rh_len)
                    247:     max = cli_rh_len;
                    248:   memcpy(buf, cli_rh_pos, max);
                    249:   cli_rh_pos += max;
                    250:   cli_rh_len -= max;
                    251:   return max;
                    252: }
                    253: 
                    254: static void
                    255: cli_command(struct cli *c)
                    256: {
                    257:   struct config f;
                    258:   int res;
                    259: 
                    260:   if (config->cli_debug > 1)
                    261:     log(L_TRACE "CLI: %s", c->rx_buf);
                    262:   bzero(&f, sizeof(f));
                    263:   f.mem = c->parser_pool;
                    264:   f.pool = rp_new(c->pool, "Config");
                    265:   init_list(&f.symbols);
                    266:   cf_read_hook = cli_cmd_read_hook;
                    267:   cli_rh_pos = c->rx_buf;
                    268:   cli_rh_len = strlen(c->rx_buf);
                    269:   cli_rh_trick_flag = 0;
                    270:   this_cli = c;
                    271:   lp_flush(c->parser_pool);
                    272:   res = cli_parse(&f);
                    273:   if (!res)
                    274:     cli_printf(c, 9001, f.err_msg);
                    275: 
                    276:   config_free(&f);
                    277: }
                    278: 
                    279: static void
                    280: cli_event(void *data)
                    281: {
                    282:   cli *c = data;
                    283:   int err;
                    284: 
                    285:   while (c->ring_read != c->ring_write &&
                    286:       c->async_msg_size < CLI_MAX_ASYNC_QUEUE)
                    287:     cli_copy_message(c);
                    288: 
                    289:   if (c->tx_pos)
                    290:     ;
                    291:   else if (c->cont)
                    292:     c->cont(c);
                    293:   else
                    294:     {
                    295:       err = cli_get_command(c);
                    296:       if (!err)
                    297:        return;
                    298:       if (err < 0)
                    299:        cli_printf(c, 9000, "Command too long");
                    300:       else
                    301:        cli_command(c);
                    302:     }
                    303: 
                    304:   cli_write_trigger(c);
                    305: }
                    306: 
                    307: cli *
                    308: cli_new(void *priv)
                    309: {
                    310:   pool *p = rp_new(cli_pool, "CLI");
                    311:   cli *c = mb_alloc(p, sizeof(cli));
                    312: 
                    313:   bzero(c, sizeof(cli));
                    314:   c->pool = p;
                    315:   c->priv = priv;
                    316:   c->event = ev_new(p);
                    317:   c->event->hook = cli_event;
                    318:   c->event->data = c;
                    319:   c->cont = cli_hello;
                    320:   c->parser_pool = lp_new_default(c->pool);
                    321:   c->show_pool = lp_new_default(c->pool);
                    322:   c->rx_buf = mb_alloc(c->pool, CLI_RX_BUF_SIZE);
                    323:   ev_schedule(c->event);
                    324:   return c;
                    325: }
                    326: 
                    327: void
                    328: cli_kick(cli *c)
                    329: {
                    330:   if (!c->cont && !c->tx_pos)
                    331:     ev_schedule(c->event);
                    332: }
                    333: 
                    334: static list cli_log_hooks;
                    335: static int cli_log_inited;
                    336: 
                    337: void
                    338: cli_set_log_echo(cli *c, uint mask, uint size)
                    339: {
                    340:   if (c->ring_buf)
                    341:     {
                    342:       mb_free(c->ring_buf);
                    343:       c->ring_buf = c->ring_end = c->ring_read = c->ring_write = NULL;
                    344:       rem_node(&c->n);
                    345:     }
                    346:   c->log_mask = mask;
                    347:   if (mask && size)
                    348:     {
                    349:       c->ring_buf = mb_alloc(c->pool, size);
                    350:       c->ring_end = c->ring_buf + size;
                    351:       c->ring_read = c->ring_write = c->ring_buf;
                    352:       add_tail(&cli_log_hooks, &c->n);
                    353:       c->log_threshold = size / 8;
                    354:     }
                    355:   c->ring_overflow = 0;
                    356: }
                    357: 
                    358: void
                    359: cli_echo(uint class, byte *msg)
                    360: {
                    361:   unsigned len, free, i, l;
                    362:   cli *c;
                    363:   byte *m;
                    364: 
                    365:   if (!cli_log_inited || EMPTY_LIST(cli_log_hooks))
                    366:     return;
                    367:   len = strlen(msg) + 1;
                    368:   WALK_LIST(c, cli_log_hooks)
                    369:     {
                    370:       if (!(c->log_mask & (1 << class)))
                    371:        continue;
                    372:       if (c->ring_read <= c->ring_write)
                    373:        free = (c->ring_end - c->ring_buf) - (c->ring_write - c->ring_read + 1);
                    374:       else
                    375:        free = c->ring_read - c->ring_write - 1;
                    376:       if ((len > free) ||
                    377:          (free < c->log_threshold && class < (unsigned) L_INFO[0]))
                    378:        {
                    379:          c->ring_overflow++;
                    380:          continue;
                    381:        }
                    382:       if (c->ring_read == c->ring_write)
                    383:        ev_schedule(c->event);
                    384:       m = msg;
                    385:       l = len;
                    386:       while (l)
                    387:        {
                    388:          if (c->ring_read <= c->ring_write)
                    389:            i = c->ring_end - c->ring_write;
                    390:          else
                    391:            i = c->ring_read - c->ring_write;
                    392:          if (i > l)
                    393:            i = l;
                    394:          memcpy(c->ring_write, m, i);
                    395:          m += i;
                    396:          l -= i;
                    397:          c->ring_write += i;
                    398:          if (c->ring_write == c->ring_end)
                    399:            c->ring_write = c->ring_buf;
                    400:        }
                    401:     }
                    402: }
                    403: 
                    404: /* Hack for scheduled undo notification */
                    405: extern cli *cmd_reconfig_stored_cli;
                    406: 
                    407: void
                    408: cli_free(cli *c)
                    409: {
                    410:   cli_set_log_echo(c, 0, 0);
                    411:   if (c->cleanup)
                    412:     c->cleanup(c);
                    413:   if (c == cmd_reconfig_stored_cli)
                    414:     cmd_reconfig_stored_cli = NULL;
                    415:   rfree(c->pool);
                    416: }
                    417: 
                    418: /**
                    419:  * cli_init - initialize the CLI module
                    420:  *
                    421:  * This function is called during BIRD startup to initialize
                    422:  * the internal data structures of the CLI module.
                    423:  */
                    424: void
                    425: cli_init(void)
                    426: {
                    427:   cli_pool = rp_new(&root_pool, "CLI");
                    428:   init_list(&cli_log_hooks);
                    429:   cli_log_inited = 1;
                    430: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>