blob: ed9f439c361f2579f7c4559633d5db650ac16f2b [file] [log] [blame]
Sylvain Munaut12ba7782014-06-16 10:13:40 +02001#pragma once
Harald Welte3fb0b6f2010-05-19 19:02:52 +02002
3#include <stdio.h>
4#include <stdarg.h>
Neels Hofmeyr63053002019-04-10 02:41:53 +02005#include <stdbool.h>
Alexander Couzens06929162021-09-05 23:08:59 +02006#include <time.h>
Harald Welte3fb0b6f2010-05-19 19:02:52 +02007
Neels Hofmeyr4a31ffa2017-09-07 03:08:06 +02008#include <osmocom/core/linuxlist.h>
Neels Hofmeyr70ce8712019-11-24 19:52:44 +01009#include <osmocom/core/defs.h>
Neels Hofmeyr4a31ffa2017-09-07 03:08:06 +020010
Harald Welte7acb30c2011-08-17 17:13:48 +020011/*! \defgroup vty VTY (Virtual TTY) interface
12 * @{
Neels Hofmeyr17518fe2017-06-20 04:35:06 +020013 * \file vty.h */
Harald Welte7acb30c2011-08-17 17:13:48 +020014
Harald Welte3fb0b6f2010-05-19 19:02:52 +020015/* GCC have printf type attribute check. */
16#ifdef __GNUC__
17#define VTY_PRINTF_ATTRIBUTE(a,b) __attribute__ ((__format__ (__printf__, a, b)))
18#else
19#define VTY_PRINTF_ATTRIBUTE(a,b)
20#endif /* __GNUC__ */
21
22/* Does the I/O error indicate that the operation should be retried later? */
23#define ERRNO_IO_RETRY(EN) \
24 (((EN) == EAGAIN) || ((EN) == EWOULDBLOCK) || ((EN) == EINTR))
25
26/* Vty read buffer size. */
27#define VTY_READ_BUFSIZ 512
28
29#define VTY_BUFSIZ 512
30#define VTY_MAXHIST 20
31
Vadim Yanitskiye566bdd2020-10-06 17:41:22 +070032/* Number of application / library specific VTY attributes */
33#define VTY_CMD_USR_ATTR_NUM 32
Vadim Yanitskiyef4c5972020-10-07 13:44:31 +070034/* Flag characters reserved for global VTY attributes */
35#define VTY_CMD_ATTR_FLAGS_RESERVED \
Vadim Yanitskiy72b90882020-10-21 05:07:34 +070036 { '.', '!', '@', '^' }
Vadim Yanitskiye566bdd2020-10-06 17:41:22 +070037
Neels Hofmeyr87e45502017-06-20 00:17:59 +020038/*! VTY events */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020039enum event {
40 VTY_SERV,
41 VTY_READ,
42 VTY_WRITE,
43 VTY_CLOSED,
44 VTY_TIMEOUT_RESET,
45#ifdef VTYSH
46 VTYSH_SERV,
47 VTYSH_READ,
48 VTYSH_WRITE
49#endif /* VTYSH */
50};
51
Andreas Eversberg3c6a2ce2012-07-12 09:22:56 +020052enum vty_type {
53 VTY_TERM,
54 VTY_FILE,
55 VTY_SHELL,
56 VTY_SHELL_SERV
57};
58
Neels Hofmeyr4a31ffa2017-09-07 03:08:06 +020059struct vty_parent_node {
60 struct llist_head entry;
61
62 /*! private data, specified by creator */
63 void *priv;
64
65 /*! Node status of this vty */
66 int node;
67
68 /*! When reading from a config file, these are the indenting characters expected for children of
69 * this VTY node. */
70 char *indent;
71};
72
Harald Welte7acb30c2011-08-17 17:13:48 +020073/*! Internal representation of a single VTY */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020074struct vty {
Neels Hofmeyr87e45502017-06-20 00:17:59 +020075 /*! underlying file (if any) */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020076 FILE *file;
77
Neels Hofmeyr87e45502017-06-20 00:17:59 +020078 /*! private data, specified by creator */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020079 void *priv;
80
Neels Hofmeyr87e45502017-06-20 00:17:59 +020081 /*! File descripter of this vty. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020082 int fd;
83
Neels Hofmeyr87e45502017-06-20 00:17:59 +020084 /*! Is this vty connect to file or not */
Andreas Eversberg3c6a2ce2012-07-12 09:22:56 +020085 enum vty_type type;
Harald Welte3fb0b6f2010-05-19 19:02:52 +020086
Neels Hofmeyr87e45502017-06-20 00:17:59 +020087 /*! Node status of this vty */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020088 int node;
89
Neels Hofmeyr87e45502017-06-20 00:17:59 +020090 /*! Failure count */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020091 int fail;
92
Neels Hofmeyr87e45502017-06-20 00:17:59 +020093 /*! Output buffer. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020094 struct buffer *obuf;
95
Neels Hofmeyr87e45502017-06-20 00:17:59 +020096 /*! Command input buffer */
Harald Welte3fb0b6f2010-05-19 19:02:52 +020097 char *buf;
98
Neels Hofmeyr87e45502017-06-20 00:17:59 +020099 /*! Command cursor point */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200100 int cp;
101
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200102 /*! Command length */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200103 int length;
104
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200105 /*! Command max length. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200106 int max;
107
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200108 /*! Histry of command */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200109 char *hist[VTY_MAXHIST];
110
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200111 /*! History lookup current point */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200112 int hp;
113
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200114 /*! History insert end point */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200115 int hindex;
116
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200117 /*! For current referencing point of interface, route-map,
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200118 access-list etc... */
119 void *index;
120
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200121 /*! For multiple level index treatment such as key chain and key. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200122 void *index_sub;
123
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200124 /*! For escape character. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200125 unsigned char escape;
126
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200127 /*! Current vty status. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200128 enum { VTY_NORMAL, VTY_CLOSE, VTY_MORE, VTY_MORELINE } status;
129
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200130 /*! IAC handling
Harald Welte7acb30c2011-08-17 17:13:48 +0200131 *
132 * IAC handling: was the last character received the IAC
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200133 * (interpret-as-command) escape character (and therefore the next
134 * character will be the command code)? Refer to Telnet RFC 854. */
135 unsigned char iac;
136
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200137 /*! IAC SB (option subnegotiation) handling */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200138 unsigned char iac_sb_in_progress;
139 /* At the moment, we care only about the NAWS (window size) negotiation,
140 * and that requires just a 5-character buffer (RFC 1073):
141 * <NAWS char> <16-bit width> <16-bit height> */
142#define TELNET_NAWS_SB_LEN 5
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200143 /*! sub-negotiation buffer */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200144 unsigned char sb_buf[TELNET_NAWS_SB_LEN];
Pau Espin Pedrol645aec82021-05-18 14:46:29 +0200145 /*! How many subnegotiation characters have we received?
Harald Welte7acb30c2011-08-17 17:13:48 +0200146 *
147 * We just drop those that do not fit in the buffer. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200148 size_t sb_len;
149
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200150 /*! Window width */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200151 int width;
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200152 /*! Widnow height */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200153 int height;
154
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200155 /*! Configure lines. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200156 int lines;
157
158 int monitor;
159
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200160 /*! In configure mode. */
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200161 int config;
Neels Hofmeyr4a31ffa2017-09-07 03:08:06 +0200162
163 /*! List of parent nodes, last item is the outermost parent. */
164 struct llist_head parent_nodes;
165
166 /*! When reading from a config file, these are the indenting characters expected for children of
167 * the current VTY node. */
168 char *indent;
Vadim Yanitskiy0a2d9bd2020-10-25 16:34:57 +0700169
170 /*! Whether the expert mode is enabled. */
171 bool expert_mode;
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200172};
173
174/* Small macro to determine newline is newline only or linefeed needed. */
175#define VTY_NEWLINE ((vty->type == VTY_TERM) ? "\r\n" : "\n")
176
Andreas Eversberg3c6a2ce2012-07-12 09:22:56 +0200177static inline const char *vty_newline(struct vty *vty)
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200178{
179 return VTY_NEWLINE;
180}
181
Harald Welte7acb30c2011-08-17 17:13:48 +0200182/*! Information an application registers with the VTY */
Harald Welte237f6242010-05-25 23:00:45 +0200183struct vty_app_info {
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200184 /*! name of the application */
Holger Hans Peter Freytherdd195272010-06-08 16:12:58 +0800185 const char *name;
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200186 /*! version string of the application */
Holger Hans Peter Freytherdd195272010-06-08 16:12:58 +0800187 const char *version;
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200188 /*! copyright string of the application */
Holger Hans Peter Freytherdd195272010-06-08 16:12:58 +0800189 const char *copyright;
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200190 /*! \ref talloc context */
Harald Welte237f6242010-05-25 23:00:45 +0200191 void *tall_ctx;
Neels Hofmeyrd31de232019-10-31 16:09:23 +0100192 /*! Call-back for taking actions upon exiting a node.
193 * The return value is ignored, and changes to vty->node and vty->index made in this callback are ignored.
194 * Implicit parent node tracking always sets the correct parent node and vty->index after this callback exits,
195 * so this callback can handle only those nodes that should take specific actions upon node exit, or can be left
196 * NULL entirely. */
Holger Hans Peter Freythera9e52522015-08-02 02:14:07 +0000197 int (*go_parent_cb)(struct vty *vty);
Neels Hofmeyrd31de232019-10-31 16:09:23 +0100198 /*! OBSOLETED: Implicit parent node tracking has replaced the use of this callback. This callback is no longer
199 * called, ever, and can be left NULL. */
Neels Hofmeyr70ce8712019-11-24 19:52:44 +0100200 int (*is_config_node)(struct vty *vty, int node)
201 OSMO_DEPRECATED("Implicit parent node tracking has replaced the use of this callback. This callback is"
202 " no longer called, ever, and can be left NULL.");
Neels Hofmeyr87e45502017-06-20 00:17:59 +0200203 /*! Check if the config is consistent before write */
Holger Hans Peter Freyther9f0f9782014-11-21 10:40:07 +0100204 int (*config_is_consistent)(struct vty *vty);
Vadim Yanitskiy7f1ecd92020-08-15 22:16:30 +0700205 /*! Description of the application specific VTY attributes (optional). */
Vadim Yanitskiye566bdd2020-10-06 17:41:22 +0700206 const char * usr_attr_desc[VTY_CMD_USR_ATTR_NUM];
Vadim Yanitskiy7f1ecd92020-08-15 22:16:30 +0700207 /*! Flag letters of the application specific VTY attributes (optional). */
Vadim Yanitskiye566bdd2020-10-06 17:41:22 +0700208 char usr_attr_letters[VTY_CMD_USR_ATTR_NUM];
Harald Welte237f6242010-05-25 23:00:45 +0200209};
210
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200211/* Prototypes. */
Harald Welte237f6242010-05-25 23:00:45 +0200212void vty_init(struct vty_app_info *app_info);
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200213int vty_read_config_file(const char *file_name, void *priv);
Pau Espin Pedrol645aec82021-05-18 14:46:29 +0200214int vty_read_config_filep(FILE *confp, void *priv);
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200215void vty_init_vtysh (void);
216void vty_reset (void);
217struct vty *vty_new (void);
218struct vty *vty_create (int vty_sock, void *priv);
Neels Hofmeyr63053002019-04-10 02:41:53 +0200219bool vty_is_active(struct vty *vty);
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200220int vty_out (struct vty *, const char *, ...) VTY_PRINTF_ATTRIBUTE(2, 3);
Neels Hofmeyrc1aa1782019-02-01 05:38:44 +0100221int vty_out_va(struct vty *vty, const char *format, va_list ap);
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200222int vty_out_newline(struct vty *);
Alexander Couzens06929162021-09-05 23:08:59 +0200223int vty_out_uptime(struct vty *vty, const struct timespec *starttime);
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200224int vty_read(struct vty *vty);
225//void vty_time_print (struct vty *, int);
226void vty_close (struct vty *);
Eric11a58a12021-09-09 15:42:32 +0200227void vty_flush(struct vty *vty);
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200228char *vty_get_cwd (void);
229void vty_log (const char *level, const char *proto, const char *fmt, va_list);
230int vty_config_lock (struct vty *);
231int vty_config_unlock (struct vty *);
232int vty_shell (struct vty *);
233int vty_shell_serv (struct vty *);
234void vty_hello (struct vty *);
Holger Hans Peter Freyther08aaded2010-09-14 02:24:03 +0800235void *vty_current_index(struct vty *);
236int vty_current_node(struct vty *vty);
Holger Hans Peter Freythera9e52522015-08-02 02:14:07 +0000237int vty_go_parent(struct vty *vty);
Harald Welte3fb0b6f2010-05-19 19:02:52 +0200238
Neels Hofmeyr96172f02016-02-23 14:01:41 +0100239/* Return IP address passed to the 'line vty'/'bind' command, or "127.0.0.1" */
240const char *vty_get_bind_addr(void);
Holger Hans Peter Freyther99ae4012018-12-15 17:36:41 +0000241/** Returns configured port passed to the 'line vty'/'bind' command or default_port. */
242int vty_get_bind_port(int default_port);
Neels Hofmeyr96172f02016-02-23 14:01:41 +0100243
Holger Hans Peter Freyther4aaccd72010-08-31 17:09:44 +0800244extern void *tall_vty_ctx;
Harald Welte7acb30c2011-08-17 17:13:48 +0200245
Harald Welted61d5172011-09-04 22:56:10 +0200246extern struct cmd_element cfg_description_cmd;
247extern struct cmd_element cfg_no_description_cmd;
248
Holger Hans Peter Freyther2c9168c2013-10-10 20:21:33 +0200249
250/**
251 * signal handling
252 */
253enum signal_vty {
254 S_VTY_EVENT,
255};
256
257struct vty_signal_data {
258 enum event event;
259 int sock;
260 struct vty *vty;
261};
262
Sylvain Munautdca7d2c2012-04-18 21:53:23 +0200263/*! @} */