1 /*
2 * Copyright (c) 2018 Nordic Semiconductor ASA
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6 #ifndef SHELL_OPS_H__
7 #define SHELL_OPS_H__
8
9 #include <stdbool.h>
10 #include <zephyr/shell/shell.h>
11 #include "shell_vt100.h"
12 #include "shell_utils.h"
13
14 #ifdef __cplusplus
15 extern "C" {
16 #endif
17
z_shell_raw_fprintf(const struct shell_fprintf * const ctx,const char * fmt,...)18 static inline void z_shell_raw_fprintf(const struct shell_fprintf *const ctx,
19 const char *fmt, ...)
20 {
21 va_list args;
22
23 va_start(args, fmt);
24 z_shell_fprintf_fmt(ctx, fmt, args);
25 va_end(args);
26 }
27
28 /* Macro to send VT100 command. */
29 #define Z_SHELL_VT100_CMD(_shell_, ...) \
30 do { \
31 if (!IS_ENABLED(CONFIG_SHELL_VT100_COMMANDS) || \
32 !z_flag_use_vt100_get(_shell_)) \
33 break; \
34 z_shell_raw_fprintf(_shell_->fprintf_ctx, __VA_ARGS__); \
35 } while (0)
36
37 #define Z_SHELL_SET_FLAG_ATOMIC(_shell_, _type_, _flag_, _val_, _ret_) \
38 do { \
39 union shell_backend_##_type_ _internal_; \
40 atomic_t *_dst_ = (atomic_t *)&(_shell_)->ctx->_type_.value; \
41 _internal_.value = 0U; \
42 _internal_.flags._flag_ = 1U; \
43 if (_val_) { \
44 _internal_.value = atomic_or(_dst_, \
45 _internal_.value); \
46 } else { \
47 _internal_.value = atomic_and(_dst_, \
48 ~_internal_.value); \
49 } \
50 _ret_ = (_internal_.flags._flag_ != 0); \
51 } while (false)
52
z_flag_insert_mode_get(const struct shell * sh)53 static inline bool z_flag_insert_mode_get(const struct shell *sh)
54 {
55 return sh->ctx->cfg.flags.insert_mode == 1;
56 }
57
z_flag_insert_mode_set(const struct shell * sh,bool val)58 static inline bool z_flag_insert_mode_set(const struct shell *sh, bool val)
59 {
60 bool ret;
61
62 Z_SHELL_SET_FLAG_ATOMIC(sh, cfg, insert_mode, val, ret);
63 return ret;
64 }
65
z_flag_use_colors_get(const struct shell * sh)66 static inline bool z_flag_use_colors_get(const struct shell *sh)
67 {
68 return sh->ctx->cfg.flags.use_colors == 1;
69 }
70
z_flag_use_colors_set(const struct shell * sh,bool val)71 static inline bool z_flag_use_colors_set(const struct shell *sh, bool val)
72 {
73 bool ret;
74
75 Z_SHELL_SET_FLAG_ATOMIC(sh, cfg, use_colors, val, ret);
76 return ret;
77 }
78
z_flag_use_vt100_get(const struct shell * sh)79 static inline bool z_flag_use_vt100_get(const struct shell *sh)
80 {
81 return sh->ctx->cfg.flags.use_vt100 == 1;
82 }
83
z_flag_use_vt100_set(const struct shell * sh,bool val)84 static inline bool z_flag_use_vt100_set(const struct shell *sh, bool val)
85 {
86 bool ret;
87
88 Z_SHELL_SET_FLAG_ATOMIC(sh, cfg, use_vt100, val, ret);
89 return ret;
90 }
91
z_flag_echo_get(const struct shell * sh)92 static inline bool z_flag_echo_get(const struct shell *sh)
93 {
94 return sh->ctx->cfg.flags.echo == 1;
95 }
96
z_flag_echo_set(const struct shell * sh,bool val)97 static inline bool z_flag_echo_set(const struct shell *sh, bool val)
98 {
99 bool ret;
100
101 Z_SHELL_SET_FLAG_ATOMIC(sh, cfg, echo, val, ret);
102 return ret;
103 }
104
z_flag_obscure_get(const struct shell * sh)105 static inline bool z_flag_obscure_get(const struct shell *sh)
106 {
107 return sh->ctx->cfg.flags.obscure == 1;
108 }
109
z_flag_obscure_set(const struct shell * sh,bool val)110 static inline bool z_flag_obscure_set(const struct shell *sh, bool val)
111 {
112 bool ret;
113
114 Z_SHELL_SET_FLAG_ATOMIC(sh, cfg, obscure, val, ret);
115 return ret;
116 }
117
z_flag_processing_get(const struct shell * sh)118 static inline bool z_flag_processing_get(const struct shell *sh)
119 {
120 return sh->ctx->ctx.flags.processing == 1;
121 }
122
z_flag_processing_set(const struct shell * sh,bool val)123 static inline bool z_flag_processing_set(const struct shell *sh, bool val)
124 {
125 bool ret;
126
127 Z_SHELL_SET_FLAG_ATOMIC(sh, ctx, processing, val, ret);
128 return ret;
129 }
130
z_flag_tx_rdy_get(const struct shell * sh)131 static inline bool z_flag_tx_rdy_get(const struct shell *sh)
132 {
133 return sh->ctx->ctx.flags.tx_rdy == 1;
134 }
135
z_flag_tx_rdy_set(const struct shell * sh,bool val)136 static inline bool z_flag_tx_rdy_set(const struct shell *sh, bool val)
137 {
138 bool ret;
139
140 Z_SHELL_SET_FLAG_ATOMIC(sh, ctx, tx_rdy, val, ret);
141 return ret;
142 }
143
z_flag_mode_delete_get(const struct shell * sh)144 static inline bool z_flag_mode_delete_get(const struct shell *sh)
145 {
146 return sh->ctx->cfg.flags.mode_delete == 1;
147 }
148
z_flag_mode_delete_set(const struct shell * sh,bool val)149 static inline bool z_flag_mode_delete_set(const struct shell *sh, bool val)
150 {
151 bool ret;
152
153 Z_SHELL_SET_FLAG_ATOMIC(sh, cfg, mode_delete, val, ret);
154 return ret;
155 }
156
z_flag_history_exit_get(const struct shell * sh)157 static inline bool z_flag_history_exit_get(const struct shell *sh)
158 {
159 return sh->ctx->ctx.flags.history_exit == 1;
160 }
161
z_flag_history_exit_set(const struct shell * sh,bool val)162 static inline bool z_flag_history_exit_set(const struct shell *sh, bool val)
163 {
164 bool ret;
165
166 Z_SHELL_SET_FLAG_ATOMIC(sh, ctx, history_exit, val, ret);
167 return ret;
168 }
169
z_flag_cmd_ctx_get(const struct shell * sh)170 static inline bool z_flag_cmd_ctx_get(const struct shell *sh)
171 {
172 return sh->ctx->ctx.flags.cmd_ctx == 1;
173 }
174
z_flag_cmd_ctx_set(const struct shell * sh,bool val)175 static inline bool z_flag_cmd_ctx_set(const struct shell *sh, bool val)
176 {
177 bool ret;
178
179 Z_SHELL_SET_FLAG_ATOMIC(sh, ctx, cmd_ctx, val, ret);
180 return ret;
181 }
182
z_flag_last_nl_get(const struct shell * sh)183 static inline uint8_t z_flag_last_nl_get(const struct shell *sh)
184 {
185 return sh->ctx->ctx.flags.last_nl;
186 }
187
z_shell_get_return_value(const struct shell * sh)188 static inline int z_shell_get_return_value(const struct shell *sh)
189 {
190 return sh->ctx->ret_val;
191 }
192
z_flag_last_nl_set(const struct shell * sh,uint8_t val)193 static inline void z_flag_last_nl_set(const struct shell *sh, uint8_t val)
194 {
195 sh->ctx->ctx.flags.last_nl = val;
196 }
197
z_flag_print_noinit_get(const struct shell * sh)198 static inline bool z_flag_print_noinit_get(const struct shell *sh)
199 {
200 return sh->ctx->ctx.flags.print_noinit == 1;
201 }
202
z_flag_print_noinit_set(const struct shell * sh,bool val)203 static inline bool z_flag_print_noinit_set(const struct shell *sh, bool val)
204 {
205 bool ret;
206
207 Z_SHELL_SET_FLAG_ATOMIC(sh, ctx, print_noinit, val, ret);
208 return ret;
209 }
210
z_flag_sync_mode_get(const struct shell * sh)211 static inline bool z_flag_sync_mode_get(const struct shell *sh)
212 {
213 return sh->ctx->ctx.flags.sync_mode == 1;
214 }
215
z_flag_sync_mode_set(const struct shell * sh,bool val)216 static inline bool z_flag_sync_mode_set(const struct shell *sh, bool val)
217 {
218 bool ret;
219
220 Z_SHELL_SET_FLAG_ATOMIC(sh, ctx, sync_mode, val, ret);
221 return ret;
222 }
223
z_flag_handle_log_get(const struct shell * sh)224 static inline bool z_flag_handle_log_get(const struct shell *sh)
225 {
226 return sh->ctx->ctx.flags.handle_log == 1;
227 }
228
z_flag_handle_log_set(const struct shell * sh,bool val)229 static inline bool z_flag_handle_log_set(const struct shell *sh, bool val)
230 {
231 bool ret;
232
233 Z_SHELL_SET_FLAG_ATOMIC(sh, ctx, handle_log, val, ret);
234 return ret;
235 }
236
237 /* Function sends VT100 command to clear the screen from cursor position to
238 * end of the screen.
239 */
z_clear_eos(const struct shell * sh)240 static inline void z_clear_eos(const struct shell *sh)
241 {
242 Z_SHELL_VT100_CMD(sh, SHELL_VT100_CLEAREOS);
243 }
244
245 /* Function sends VT100 command to save cursor position. */
z_cursor_save(const struct shell * sh)246 static inline void z_cursor_save(const struct shell *sh)
247 {
248 Z_SHELL_VT100_CMD(sh, SHELL_VT100_SAVECURSOR);
249 }
250
251 /* Function sends VT100 command to restore saved cursor position. */
z_cursor_restore(const struct shell * sh)252 static inline void z_cursor_restore(const struct shell *sh)
253 {
254 Z_SHELL_VT100_CMD(sh, SHELL_VT100_RESTORECURSOR);
255 }
256
257 /* Function forcing new line - cannot be replaced with function
258 * cursor_down_move.
259 */
z_cursor_next_line_move(const struct shell * sh)260 static inline void z_cursor_next_line_move(const struct shell *sh)
261 {
262 z_shell_raw_fprintf(sh->fprintf_ctx, "\n");
263 }
264
265 void z_shell_op_cursor_vert_move(const struct shell *sh, int32_t delta);
266
267 void z_shell_op_cursor_horiz_move(const struct shell *sh, int32_t delta);
268
269 void z_shell_op_cond_next_line(const struct shell *sh);
270
271 /* Function will move cursor back to position == cmd_buff_pos. Example usage is
272 * when cursor needs to be moved back after printing some text. This function
273 * cannot be used to move cursor to new location by manual change of
274 * cmd_buff_pos.
275 */
276 void z_shell_op_cursor_position_synchronize(const struct shell *sh);
277
278 void z_shell_op_cursor_move(const struct shell *sh, int16_t val);
279
280 void z_shell_op_left_arrow(const struct shell *sh);
281
282 void z_shell_op_right_arrow(const struct shell *sh);
283
284 /* Moves cursor by defined number of words left (val negative) or right. */
285 void z_shell_op_cursor_word_move(const struct shell *sh, int16_t val);
286
287 /*
288 * Removes the "word" to the left of the cursor:
289 * - if there are spaces at the cursor position, remove all spaces to the left
290 * - remove the non-spaces (word) until a space is found or a beginning of
291 * buffer
292 */
293 void z_shell_op_word_remove(const struct shell *sh);
294
295 /* Function moves cursor to begin of command position, just after console
296 * name.
297 */
298 void z_shell_op_cursor_home_move(const struct shell *sh);
299
300 /* Function moves cursor to end of command. */
301 void z_shell_op_cursor_end_move(const struct shell *sh);
302
303 void z_shell_op_char_insert(const struct shell *sh, char data);
304
305 void z_shell_op_char_backspace(const struct shell *sh);
306
307 void z_shell_op_char_delete(const struct shell *sh);
308
309 void z_shell_op_delete_from_cursor(const struct shell *sh);
310
311 void z_shell_op_completion_insert(const struct shell *sh,
312 const char *compl,
313 uint16_t compl_len);
314
315 bool z_shell_cursor_in_empty_line(const struct shell *sh);
316
317 void z_shell_cmd_line_erase(const struct shell *sh);
318
319 /**
320 * @brief Print command buffer.
321 *
322 * @param sh Shell instance.
323 */
324 void z_shell_print_cmd(const struct shell *sh);
325
326 /**
327 * @brief Print prompt followed by command buffer.
328 *
329 * @param sh Shell instance.
330 */
331 void z_shell_print_prompt_and_cmd(const struct shell *sh);
332
333 /* Function sends data stream to the shell instance. Each time before the
334 * shell_write function is called, it must be ensured that IO buffer of fprintf
335 * is flushed to avoid synchronization issues.
336 * For that purpose, use function transport_buffer_flush(shell)
337 *
338 * This function can be only used by shell module, it shall not be called
339 * directly.
340 */
341 void z_shell_write(const struct shell *sh, const void *data, size_t length);
342
343 /**
344 * @internal @brief This function shall not be used directly, it is required by
345 * the fprintf module.
346 *
347 * @param[in] p_user_ctx Pointer to the context for the shell instance.
348 * @param[in] p_data Pointer to the data buffer.
349 * @param[in] len Data buffer size.
350 */
351 void z_shell_print_stream(const void *user_ctx, const char *data, size_t len);
352
353 /** @internal @brief Function for setting font color */
354 void z_shell_vt100_color_set(const struct shell *sh,
355 enum shell_vt100_color color);
356
z_shell_vt100_colors_store(const struct shell * sh,struct shell_vt100_colors * color)357 static inline void z_shell_vt100_colors_store(const struct shell *sh,
358 struct shell_vt100_colors *color)
359 {
360 memcpy(color, &sh->ctx->vt100_ctx.col, sizeof(*color));
361 }
362
363 void z_shell_vt100_colors_restore(const struct shell *sh,
364 const struct shell_vt100_colors *color);
365
366 /* This function can be called only within shell thread but not from command
367 * handlers.
368 */
369 void z_shell_fprintf(const struct shell *sh, enum shell_vt100_color color,
370 const char *fmt, ...);
371
372 void z_shell_vfprintf(const struct shell *sh, enum shell_vt100_color color,
373 const char *fmt, va_list args);
374
375 /**
376 * @brief Flushes the shell backend receive buffer.
377 *
378 * This function repeatedly reads from the shell interface's receive buffer
379 * until it is empty or a maximum number of iterations is reached.
380 * It ensures that no additional data is left in the buffer.
381 */
382 void z_shell_backend_rx_buffer_flush(const struct shell *sh);
383
384 #ifdef __cplusplus
385 }
386 #endif
387
388 #endif /* SHELL_OPS_H__ */
389