1 /*
2 * Copyright (c) 2018 Nordic Semiconductor ASA
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6 #include <ctype.h>
7 #include <zephyr/device.h>
8 #include <zephyr/sys/iterable_sections.h>
9 #include <stdlib.h>
10 #include "shell_utils.h"
11 #include "shell_wildcard.h"
12
13 TYPE_SECTION_START_EXTERN(union shell_cmd_entry, shell_dynamic_subcmds);
14 TYPE_SECTION_END_EXTERN(union shell_cmd_entry, shell_dynamic_subcmds);
15
16 TYPE_SECTION_START_EXTERN(union shell_cmd_entry, shell_subcmds);
17 TYPE_SECTION_END_EXTERN(union shell_cmd_entry, shell_subcmds);
18
19 /* Macro creates empty entry at the bottom of the memory section with subcommands
20 * it is used to detect end of subcommand set that is located before this marker.
21 */
22 #define Z_SHELL_SUBCMD_END_MARKER_CREATE() \
23 static const TYPE_SECTION_ITERABLE(struct shell_static_entry, \
24 z_shell_subcmd_end_marker, shell_subcmds, Z_SHELL_UNDERSCORE(999))
25
26 Z_SHELL_SUBCMD_END_MARKER_CREATE();
27
shell_root_cmd_get(uint32_t id)28 static inline const union shell_cmd_entry *shell_root_cmd_get(uint32_t id)
29 {
30 const union shell_cmd_entry *cmd;
31
32 TYPE_SECTION_GET(union shell_cmd_entry, shell_root_cmds, id, &cmd);
33
34 return cmd;
35 }
36
37 /* Determine if entry is a dynamic command by checking if address is within
38 * dynamic commands memory section.
39 */
is_dynamic_cmd(const union shell_cmd_entry * entry)40 static inline bool is_dynamic_cmd(const union shell_cmd_entry *entry)
41 {
42 return (entry >= TYPE_SECTION_START(shell_dynamic_subcmds)) &&
43 (entry < TYPE_SECTION_END(shell_dynamic_subcmds));
44 }
45
46 /* Determine if entry is a section command by checking if address is within
47 * subcommands memory section.
48 */
is_section_cmd(const union shell_cmd_entry * entry)49 static inline bool is_section_cmd(const union shell_cmd_entry *entry)
50 {
51 return (entry >= TYPE_SECTION_START(shell_subcmds)) &&
52 (entry < TYPE_SECTION_END(shell_subcmds));
53 }
54
55 /* Calculates relative line number of given position in buffer */
line_num_with_buffer_offset_get(struct shell_multiline_cons * cons,uint16_t buffer_pos)56 static uint32_t line_num_with_buffer_offset_get(struct shell_multiline_cons *cons,
57 uint16_t buffer_pos)
58 {
59 return ((buffer_pos + cons->name_len) / cons->terminal_wid);
60 }
61
62 /* Calculates column number of given position in buffer */
col_num_with_buffer_offset_get(struct shell_multiline_cons * cons,uint16_t buffer_pos)63 static uint32_t col_num_with_buffer_offset_get(struct shell_multiline_cons *cons,
64 uint16_t buffer_pos)
65 {
66 /* columns are counted from 1 */
67 return (1 + ((buffer_pos + cons->name_len) % cons->terminal_wid));
68 }
69
z_column_span_with_buffer_offsets_get(struct shell_multiline_cons * cons,uint16_t offset1,uint16_t offset2)70 int32_t z_column_span_with_buffer_offsets_get(struct shell_multiline_cons *cons,
71 uint16_t offset1,
72 uint16_t offset2)
73 {
74 return col_num_with_buffer_offset_get(cons, offset2)
75 - col_num_with_buffer_offset_get(cons, offset1);
76 }
77
z_row_span_with_buffer_offsets_get(struct shell_multiline_cons * cons,uint16_t offset1,uint16_t offset2)78 int32_t z_row_span_with_buffer_offsets_get(struct shell_multiline_cons *cons,
79 uint16_t offset1,
80 uint16_t offset2)
81 {
82 return line_num_with_buffer_offset_get(cons, offset2)
83 - line_num_with_buffer_offset_get(cons, offset1);
84 }
85
z_shell_multiline_data_calc(struct shell_multiline_cons * cons,uint16_t buff_pos,uint16_t buff_len)86 void z_shell_multiline_data_calc(struct shell_multiline_cons *cons,
87 uint16_t buff_pos, uint16_t buff_len)
88 {
89 /* Current cursor position in command.
90 * +1 -> because home position is (1, 1)
91 */
92 cons->cur_x = (buff_pos + cons->name_len) % cons->terminal_wid + 1;
93 cons->cur_y = (buff_pos + cons->name_len) / cons->terminal_wid + 1;
94
95 /* Extreme position when cursor is at the end of command. */
96 cons->cur_y_end = (buff_len + cons->name_len) / cons->terminal_wid + 1;
97 cons->cur_x_end = (buff_len + cons->name_len) % cons->terminal_wid + 1;
98 }
99
make_argv(char ** ppcmd,uint8_t c)100 static char make_argv(char **ppcmd, uint8_t c)
101 {
102 char *cmd = *ppcmd;
103 char quote = 0;
104
105 while (1) {
106 c = *cmd;
107
108 if (c == '\0') {
109 break;
110 }
111
112 if (!quote) {
113 switch (c) {
114 case '\\':
115 memmove(cmd, cmd + 1,
116 z_shell_strlen(cmd));
117 cmd += 1;
118 continue;
119
120 case '\'':
121 case '\"':
122 memmove(cmd, cmd + 1,
123 z_shell_strlen(cmd));
124 quote = c;
125 continue;
126 default:
127 break;
128 }
129 }
130
131 if (quote == c) {
132 memmove(cmd, cmd + 1, z_shell_strlen(cmd));
133 quote = 0;
134 continue;
135 }
136
137 if (quote && c == '\\') {
138 char t = *(cmd + 1);
139
140 if (t == quote) {
141 memmove(cmd, cmd + 1,
142 z_shell_strlen(cmd));
143 cmd += 1;
144 continue;
145 }
146
147 if (t == '0') {
148 uint8_t i;
149 uint8_t v = 0U;
150
151 for (i = 2U; i < (2 + 3); i++) {
152 t = *(cmd + i);
153
154 if (t >= '0' && t <= '7') {
155 v = (v << 3) | (t - '0');
156 } else {
157 break;
158 }
159 }
160
161 if (i > 2) {
162 memmove(cmd, cmd + (i - 1),
163 z_shell_strlen(cmd) - (i - 2));
164 *cmd++ = v;
165 continue;
166 }
167 }
168
169 if (t == 'x') {
170 uint8_t i;
171 uint8_t v = 0U;
172
173 for (i = 2U; i < (2 + 2); i++) {
174 t = *(cmd + i);
175
176 if (t >= '0' && t <= '9') {
177 v = (v << 4) | (t - '0');
178 } else if ((t >= 'a') &&
179 (t <= 'f')) {
180 v = (v << 4) | (t - 'a' + 10);
181 } else if ((t >= 'A') && (t <= 'F')) {
182 v = (v << 4) | (t - 'A' + 10);
183 } else {
184 break;
185 }
186 }
187
188 if (i > 2) {
189 memmove(cmd, cmd + (i - 1),
190 z_shell_strlen(cmd) - (i - 2));
191 *cmd++ = v;
192 continue;
193 }
194 }
195 }
196
197 if (!quote && isspace((int) c) != 0) {
198 break;
199 }
200
201 cmd += 1;
202 }
203 *ppcmd = cmd;
204
205 return quote;
206 }
207
208
z_shell_make_argv(size_t * argc,const char ** argv,char * cmd,uint8_t max_argc)209 char z_shell_make_argv(size_t *argc, const char **argv, char *cmd,
210 uint8_t max_argc)
211 {
212 char quote = 0;
213 char c;
214
215 *argc = 0;
216 do {
217 c = *cmd;
218 if (c == '\0') {
219 break;
220 }
221
222 if (isspace((int) c) != 0) {
223 *cmd++ = '\0';
224 continue;
225 }
226
227 argv[(*argc)++] = cmd;
228 if (*argc == max_argc) {
229 break;
230 }
231 quote = make_argv(&cmd, c);
232 } while (true);
233
234 return quote;
235 }
236
z_shell_pattern_remove(char * buff,uint16_t * buff_len,const char * pattern)237 void z_shell_pattern_remove(char *buff, uint16_t *buff_len, const char *pattern)
238 {
239 char *pattern_addr = strstr(buff, pattern);
240 uint16_t shift;
241 uint16_t pattern_len = z_shell_strlen(pattern);
242
243 if (!pattern_addr) {
244 return;
245 }
246
247 if (pattern_addr > buff) {
248 if (*(pattern_addr - 1) == ' ') {
249 pattern_len++; /* space needs to be removed as well */
250 pattern_addr--; /* set pointer to space */
251 }
252 }
253
254 shift = z_shell_strlen(pattern_addr) - pattern_len + 1; /* +1 for EOS */
255 *buff_len -= pattern_len;
256
257 memmove(pattern_addr, pattern_addr + pattern_len, shift);
258 }
259
shell_root_cmd_count(void)260 static inline uint32_t shell_root_cmd_count(void)
261 {
262 size_t len;
263
264 TYPE_SECTION_COUNT(union shell_cmd_entry, shell_root_cmds, &len);
265
266 return len;
267 }
268
269 /* Function returning pointer to parent command matching requested syntax. */
root_cmd_find(const char * syntax)270 const struct shell_static_entry *root_cmd_find(const char *syntax)
271 {
272 const size_t cmd_count = shell_root_cmd_count();
273 const union shell_cmd_entry *cmd;
274
275 for (size_t cmd_idx = 0; cmd_idx < cmd_count; ++cmd_idx) {
276 cmd = shell_root_cmd_get(cmd_idx);
277 if (strcmp(syntax, cmd->entry->syntax) == 0) {
278 return cmd->entry;
279 }
280 }
281
282 return NULL;
283 }
284
z_shell_cmd_get(const struct shell_static_entry * parent,size_t idx,struct shell_static_entry * dloc)285 const struct shell_static_entry *z_shell_cmd_get(
286 const struct shell_static_entry *parent,
287 size_t idx,
288 struct shell_static_entry *dloc)
289 {
290 const struct shell_static_entry *res = NULL;
291
292 if (parent == NULL) {
293 return (idx < shell_root_cmd_count()) ?
294 shell_root_cmd_get(idx)->entry : NULL;
295 }
296
297 __ASSERT_NO_MSG(dloc != NULL);
298
299 if (parent->subcmd) {
300 if (is_dynamic_cmd(parent->subcmd)) {
301 parent->subcmd->dynamic_get(idx, dloc);
302 if (dloc->syntax != NULL) {
303 res = dloc;
304 }
305 } else {
306 const struct shell_static_entry *entry_list;
307
308 if (is_section_cmd(parent->subcmd)) {
309 /* First element is null */
310 entry_list =
311 (const struct shell_static_entry *)parent->subcmd;
312 idx++;
313 } else {
314 entry_list = parent->subcmd->entry;
315 }
316
317
318 if (entry_list[idx].syntax != NULL) {
319 res = &entry_list[idx];
320 }
321 }
322 }
323
324 return res;
325 }
326
327 /* Function returns pointer to a command matching given pattern.
328 *
329 * @param cmd Pointer to commands array that will be searched.
330 * @param lvl Root command indicator. If set to 0 shell will search
331 * for pattern in parent commands.
332 * @param cmd_str Command pattern to be found.
333 * @param dloc Shell static command descriptor.
334 *
335 * @return Pointer to found command.
336 */
z_shell_find_cmd(const struct shell_static_entry * parent,const char * cmd_str,struct shell_static_entry * dloc)337 const struct shell_static_entry *z_shell_find_cmd(
338 const struct shell_static_entry *parent,
339 const char *cmd_str,
340 struct shell_static_entry *dloc)
341 {
342 const struct shell_static_entry *entry;
343 struct shell_static_entry parent_cpy;
344 size_t idx = 0;
345
346 /* Dynamic command operates on shared memory. If we are processing two
347 * dynamic commands at the same time (current and subcommand) they
348 * will operate on the same memory region what can cause undefined
349 * behaviour.
350 * Hence we need a separate memory for each of them.
351 */
352 if (parent) {
353 memcpy(&parent_cpy, parent, sizeof(struct shell_static_entry));
354 parent = &parent_cpy;
355 }
356
357 while ((entry = z_shell_cmd_get(parent, idx++, dloc)) != NULL) {
358 if (strcmp(cmd_str, entry->syntax) == 0) {
359 return entry;
360 }
361 }
362
363 return NULL;
364 }
365
z_shell_get_last_command(const struct shell_static_entry * entry,size_t argc,const char * argv[],size_t * match_arg,struct shell_static_entry * dloc,bool only_static)366 const struct shell_static_entry *z_shell_get_last_command(
367 const struct shell_static_entry *entry,
368 size_t argc,
369 const char *argv[],
370 size_t *match_arg,
371 struct shell_static_entry *dloc,
372 bool only_static)
373 {
374 const struct shell_static_entry *prev_entry = NULL;
375
376 *match_arg = Z_SHELL_CMD_ROOT_LVL;
377
378 while (*match_arg < argc) {
379
380 if (IS_ENABLED(CONFIG_SHELL_WILDCARD)) {
381 /* ignore wildcard argument */
382 if (z_shell_has_wildcard(argv[*match_arg])) {
383 (*match_arg)++;
384 continue;
385 }
386 }
387
388 prev_entry = entry;
389 entry = z_shell_find_cmd(entry, argv[*match_arg], dloc);
390 if (entry) {
391 (*match_arg)++;
392 } else {
393 entry = prev_entry;
394 break;
395 }
396
397 if (only_static && (entry == dloc)) {
398 (*match_arg)--;
399 return NULL;
400 }
401 }
402
403 return entry;
404 }
405
shell_set_root_cmd(const char * cmd)406 int shell_set_root_cmd(const char *cmd)
407 {
408 const struct shell_static_entry *entry;
409
410 entry = cmd ? root_cmd_find(cmd) : NULL;
411
412 if (cmd && (entry == NULL)) {
413 return -EINVAL;
414 }
415
416 STRUCT_SECTION_FOREACH(shell, sh) {
417 sh->ctx->selected_cmd = entry;
418 }
419
420 return 0;
421 }
422
423
424
425
z_shell_spaces_trim(char * str)426 void z_shell_spaces_trim(char *str)
427 {
428 uint16_t len = z_shell_strlen(str);
429 uint16_t shift = 0U;
430
431 if (len == 0U) {
432 return;
433 }
434
435 for (uint16_t i = 0; i < len - 1; i++) {
436 if (isspace((int)str[i]) != 0) {
437 for (uint16_t j = i + 1; j < len; j++) {
438 if (isspace((int)str[j]) != 0) {
439 shift++;
440 continue;
441 }
442
443 if (shift > 0) {
444 /* +1 for EOS */
445 memmove(&str[i + 1],
446 &str[j],
447 len - j + 1);
448 len -= shift;
449 shift = 0U;
450 }
451
452 break;
453 }
454 }
455 }
456 }
457
458 /** @brief Remove white chars from beginning and end of command buffer.
459 *
460 */
buffer_trim(char * buff,uint16_t * buff_len)461 static void buffer_trim(char *buff, uint16_t *buff_len)
462 {
463 uint16_t i = 0U;
464
465 /* no command in the buffer */
466 if (buff[0] == '\0') {
467 return;
468 }
469
470 while (isspace((int) buff[*buff_len - 1U]) != 0) {
471 *buff_len -= 1U;
472 if (*buff_len == 0U) {
473 buff[0] = '\0';
474 return;
475 }
476 }
477 buff[*buff_len] = '\0';
478
479 /* Counting whitespace characters starting from beginning of the
480 * command.
481 */
482 while (isspace((int) buff[i++]) != 0) {
483 }
484
485
486 /* Removing counted whitespace characters. */
487 if (--i > 0) {
488 memmove(buff, buff + i, (*buff_len + 1U) - i); /* +1 for '\0' */
489 *buff_len = *buff_len - i;
490 }
491 }
492
z_shell_cmd_trim(const struct shell * sh)493 void z_shell_cmd_trim(const struct shell *sh)
494 {
495 buffer_trim(sh->ctx->cmd_buff, &sh->ctx->cmd_buff_len);
496 sh->ctx->cmd_buff_pos = sh->ctx->cmd_buff_len;
497 }
498
shell_device_internal(size_t idx,const char * prefix,shell_device_filter_t filter)499 static const struct device *shell_device_internal(size_t idx,
500 const char *prefix,
501 shell_device_filter_t filter)
502 {
503 size_t match_idx = 0;
504 const struct device *dev;
505 size_t len = z_device_get_all_static(&dev);
506 const struct device *dev_end = dev + len;
507
508 while (dev < dev_end) {
509 if (device_is_ready(dev)
510 && (dev->name != NULL)
511 && (strlen(dev->name) != 0)
512 && ((prefix == NULL)
513 || (strncmp(prefix, dev->name,
514 strlen(prefix)) == 0))
515 && (filter == NULL || filter(dev))) {
516 if (match_idx == idx) {
517 return dev;
518 }
519 ++match_idx;
520 }
521 ++dev;
522 }
523
524 return NULL;
525 }
526
shell_device_filter(size_t idx,shell_device_filter_t filter)527 const struct device *shell_device_filter(size_t idx,
528 shell_device_filter_t filter)
529 {
530 return shell_device_internal(idx, NULL, filter);
531 }
532
shell_device_lookup(size_t idx,const char * prefix)533 const struct device *shell_device_lookup(size_t idx,
534 const char *prefix)
535 {
536 return shell_device_internal(idx, prefix, NULL);
537 }
538
shell_device_get_binding(const char * name)539 const struct device *shell_device_get_binding(const char *name)
540 {
541 const struct device *dev = device_get_binding(name);
542
543 if (IS_ENABLED(CONFIG_DEVICE_DT_METADATA) && dev == NULL) {
544 dev = device_get_by_dt_nodelabel(name);
545 }
546
547 return dev;
548 }
549
shell_strtol(const char * str,int base,int * err)550 long shell_strtol(const char *str, int base, int *err)
551 {
552 long val;
553 char *endptr;
554
555 errno = 0;
556 val = strtol(str, &endptr, base);
557 if (errno == ERANGE) {
558 *err = -ERANGE;
559 return 0;
560 } else if (errno || endptr == str || *endptr) {
561 *err = -EINVAL;
562 return 0;
563 }
564
565 return val;
566 }
567
shell_strtoul(const char * str,int base,int * err)568 unsigned long shell_strtoul(const char *str, int base, int *err)
569 {
570 unsigned long val;
571 char *endptr;
572
573 if (*str == '-') {
574 *err = -EINVAL;
575 return 0;
576 }
577
578 errno = 0;
579 val = strtoul(str, &endptr, base);
580 if (errno == ERANGE) {
581 *err = -ERANGE;
582 return 0;
583 } else if (errno || endptr == str || *endptr) {
584 *err = -EINVAL;
585 return 0;
586 }
587
588 return val;
589 }
590
shell_strtoull(const char * str,int base,int * err)591 unsigned long long shell_strtoull(const char *str, int base, int *err)
592 {
593 unsigned long long val;
594 char *endptr;
595
596 if (*str == '-') {
597 *err = -EINVAL;
598 return 0;
599 }
600
601 errno = 0;
602 val = strtoull(str, &endptr, base);
603 if (errno == ERANGE) {
604 *err = -ERANGE;
605 return 0;
606 } else if (errno || endptr == str || *endptr) {
607 *err = -EINVAL;
608 return 0;
609 }
610
611 return val;
612 }
613
shell_strtobool(const char * str,int base,int * err)614 bool shell_strtobool(const char *str, int base, int *err)
615 {
616 if (!strcmp(str, "on") || !strcmp(str, "enable") || !strcmp(str, "true")) {
617 return true;
618 }
619
620 if (!strcmp(str, "off") || !strcmp(str, "disable") || !strcmp(str, "false")) {
621 return false;
622 }
623
624 return shell_strtoul(str, base, err);
625 }
626