1 /*
2 * Copyright (c) 2018 Nordic Semiconductor ASA
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/shell/shell.h>
8 #include <ctype.h>
9 #include <stdio.h>
10 #include <stdlib.h>
11
12 #define MAX_CMD_CNT (20u)
13 #define MAX_CMD_LEN (33u)
14
15 /* buffer holding dynamically created user commands */
16 static char dynamic_cmd_buffer[MAX_CMD_CNT][MAX_CMD_LEN];
17 /* commands counter */
18 static uint8_t dynamic_cmd_cnt;
19
20 typedef int cmp_t(const void *, const void *);
21 extern void qsort(void *a, size_t n, size_t es, cmp_t *cmp);
22
23 /* function required by qsort */
string_cmp(const void * p_a,const void * p_b)24 static int string_cmp(const void *p_a, const void *p_b)
25 {
26 return strcmp((const char *)p_a, (const char *)p_b);
27 }
28
cmd_dynamic_add(const struct shell * sh,size_t argc,char ** argv)29 static int cmd_dynamic_add(const struct shell *sh,
30 size_t argc, char **argv)
31 {
32 uint16_t cmd_len;
33 uint8_t idx;
34
35 ARG_UNUSED(argc);
36
37 if (dynamic_cmd_cnt >= MAX_CMD_CNT) {
38 shell_error(sh, "command limit reached");
39 return -ENOEXEC;
40 }
41
42 cmd_len = strlen(argv[1]);
43
44 if (cmd_len >= MAX_CMD_LEN) {
45 shell_error(sh, "too long command");
46 return -ENOEXEC;
47 }
48
49 for (idx = 0U; idx < cmd_len; idx++) {
50 if (isalnum((int)(argv[1][idx])) == 0) {
51 shell_error(sh,
52 "bad command name - please use only"
53 " alphanumerical characters");
54 return -ENOEXEC;
55 }
56 }
57
58 for (idx = 0U; idx < MAX_CMD_CNT; idx++) {
59 if (!strcmp(dynamic_cmd_buffer[idx], argv[1])) {
60 shell_error(sh, "duplicated command");
61 return -ENOEXEC;
62 }
63 }
64
65 sprintf(dynamic_cmd_buffer[dynamic_cmd_cnt++], "%s", argv[1]);
66
67 qsort(dynamic_cmd_buffer, dynamic_cmd_cnt,
68 sizeof(dynamic_cmd_buffer[0]), string_cmp);
69
70 shell_print(sh, "command added successfully");
71
72 return 0;
73 }
74
cmd_dynamic_execute(const struct shell * sh,size_t argc,char ** argv)75 static int cmd_dynamic_execute(const struct shell *sh,
76 size_t argc, char **argv)
77 {
78 ARG_UNUSED(argc);
79 ARG_UNUSED(argv);
80
81 for (uint8_t idx = 0; idx < dynamic_cmd_cnt; idx++) {
82 if (!strcmp(dynamic_cmd_buffer[idx], argv[1])) {
83 shell_print(sh, "dynamic command: %s", argv[1]);
84 return 0;
85 }
86 }
87
88 shell_error(sh, "%s: unknown parameter: %s", argv[0], argv[1]);
89
90 return -ENOEXEC;
91 }
92
cmd_dynamic_remove(const struct shell * sh,size_t argc,char ** argv)93 static int cmd_dynamic_remove(const struct shell *sh, size_t argc,
94 char **argv)
95 {
96 ARG_UNUSED(argc);
97 ARG_UNUSED(argv);
98
99 for (uint8_t idx = 0; idx < dynamic_cmd_cnt; idx++) {
100 if (!strcmp(dynamic_cmd_buffer[idx], argv[1])) {
101 if (idx == MAX_CMD_CNT - 1) {
102 dynamic_cmd_buffer[idx][0] = '\0';
103 } else {
104 memmove(dynamic_cmd_buffer[idx],
105 dynamic_cmd_buffer[idx + 1],
106 sizeof(dynamic_cmd_buffer[idx]) *
107 (dynamic_cmd_cnt - idx));
108 }
109
110 --dynamic_cmd_cnt;
111 shell_print(sh, "command removed successfully");
112 return 0;
113 }
114 }
115 shell_error(sh, "did not find command: %s", argv[1]);
116
117 return -ENOEXEC;
118 }
119
cmd_dynamic_show(const struct shell * sh,size_t argc,char ** argv)120 static int cmd_dynamic_show(const struct shell *sh,
121 size_t argc, char **argv)
122 {
123 ARG_UNUSED(argc);
124 ARG_UNUSED(argv);
125
126 if (dynamic_cmd_cnt == 0U) {
127 shell_warn(sh, "Please add some commands first.");
128 return -ENOEXEC;
129 }
130
131 shell_print(sh, "Dynamic command list:");
132
133 for (uint8_t i = 0; i < dynamic_cmd_cnt; i++) {
134 shell_print(sh, "[%3d] %s", i, dynamic_cmd_buffer[i]);
135 }
136
137 return 0;
138 }
139
140 /* dynamic command creation */
dynamic_cmd_get(size_t idx,struct shell_static_entry * entry)141 static void dynamic_cmd_get(size_t idx, struct shell_static_entry *entry)
142 {
143 if (idx < dynamic_cmd_cnt) {
144 /* m_dynamic_cmd_buffer must be sorted alphabetically to ensure
145 * correct CLI completion
146 */
147 entry->syntax = dynamic_cmd_buffer[idx];
148 entry->handler = NULL;
149 entry->subcmd = NULL;
150 entry->help = "Show dynamic command name.";
151 } else {
152 /* if there are no more dynamic commands available syntax
153 * must be set to NULL.
154 */
155 entry->syntax = NULL;
156 }
157 }
158
159 SHELL_DYNAMIC_CMD_CREATE(m_sub_dynamic_set, dynamic_cmd_get);
160 SHELL_STATIC_SUBCMD_SET_CREATE(m_sub_dynamic,
161 SHELL_CMD_ARG(add, NULL,
162 "Add a new dynamic command.\nExample usage: [ dynamic add test "
163 "] will add a dynamic command 'test'.\nIn this example, command"
164 " name length is limited to 32 chars. You can add up to 20"
165 " commands. Commands are automatically sorted to ensure correct"
166 " shell completion.",
167 cmd_dynamic_add, 2, 0),
168 SHELL_CMD_ARG(execute, &m_sub_dynamic_set,
169 "Execute a command.", cmd_dynamic_execute, 2, 0),
170 SHELL_CMD_ARG(remove, &m_sub_dynamic_set,
171 "Remove a command.", cmd_dynamic_remove, 2, 0),
172 SHELL_CMD_ARG(show, NULL,
173 "Show all added dynamic commands.", cmd_dynamic_show, 1, 0),
174 SHELL_SUBCMD_SET_END
175 );
176
177 SHELL_CMD_REGISTER(dynamic, &m_sub_dynamic,
178 "Demonstrate dynamic command usage.", NULL);
179