1
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
25
26
29
30
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
51
52
53
54
55
56
57
58
59
60
61
64
67
68
71
74
75
78
79
80
81
96
97
102
103
104
105
106
109
110
111
112
113
114
115
116
117
120
123
124
127
128
138
139
140
153
154
155
162
163
164
165
166
174
175
176
186
187
188
189
203
204
218
219
220
221
222
223
224
231
232
233
234
235
236
239
240
243
244
245
246
247
251
252
256
259
262
263
264
265
266
271
272
273
274
277
278
279
283
284
285
286
289
290
291
292
297
298
302
303
304
305
306
307
311
312
313
314
315
316
317
318
321
322
325
326
330
331
332
338
339
340
341
342
353
354
355
359
360
361
362
363
364
365
366
367
368
369
370
371
378
379
380
/* ... */
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <sys/param.h>
#include "esp_heap_caps.h"
#include "esp_log.h"
#include "esp_console.h"
#include "esp_system.h"
#include "linenoise/linenoise.h"
#include "argtable3/argtable3.h"
#include "sys/queue.h"11 includes
#define ANSI_COLOR_DEFAULT 39
typedef struct cmd_item_ {
/* ... */
const char *command;
/* ... */
const char *help;
/* ... */
char *hint;
esp_console_cmd_func_t func;
esp_console_cmd_func_with_context_t func_w_context;
void *argtable;
void *context;
SLIST_ENTRY(cmd_item_) next;
}{ ... } cmd_item_t;
typedef void (*const fn_print_arg_t)(cmd_item_t*);
static SLIST_HEAD(cmd_list_, cmd_item_) s_cmd_list;
static esp_console_config_t s_config = {
.heap_alloc_caps = MALLOC_CAP_DEFAULT
}{...};
static char *s_tmp_line_buf;
static const cmd_item_t *find_command_by_name(const char *name);
static esp_console_help_verbose_level_e s_verbose_level = ESP_CONSOLE_HELP_VERBOSE_LEVEL_1;
esp_err_t esp_console_init(const esp_console_config_t *config)
{
if (!config) {
return ESP_ERR_INVALID_ARG;
}{...}
if (s_tmp_line_buf) {
return ESP_ERR_INVALID_STATE;
}{...}
memcpy(&s_config, config, sizeof(s_config));
if (s_config.hint_color == 0) {
s_config.hint_color = ANSI_COLOR_DEFAULT;
}{...}
if (s_config.heap_alloc_caps == 0) {
s_config.heap_alloc_caps = MALLOC_CAP_DEFAULT;
}{...}
s_tmp_line_buf = heap_caps_calloc(1, config->max_cmdline_length, s_config.heap_alloc_caps);
if (s_tmp_line_buf == NULL) {
return ESP_ERR_NO_MEM;
}{...}
return ESP_OK;
}{ ... }
esp_err_t esp_console_deinit(void)
{
if (!s_tmp_line_buf) {
return ESP_ERR_INVALID_STATE;
}{...}
free(s_tmp_line_buf);
s_tmp_line_buf = NULL;
cmd_item_t *it, *tmp;
SLIST_FOREACH_SAFE(it, &s_cmd_list, next, tmp) {
SLIST_REMOVE(&s_cmd_list, it, cmd_item_, next);
free(it->hint);
free(it);
}{...}
return ESP_OK;
}{ ... }
void esp_console_rm_item_free_hint(cmd_item_t *item)
{
SLIST_REMOVE(&s_cmd_list, item, cmd_item_, next);
free(item->hint);
}{ ... }
esp_err_t esp_console_cmd_deregister(const char *cmd_name)
{
cmd_item_t *item = (cmd_item_t *)find_command_by_name(cmd_name);
if (item == NULL) {
return ESP_ERR_INVALID_ARG;
}{...}
esp_console_rm_item_free_hint(item);
heap_caps_free(item);
return ESP_OK;
}{ ... }
esp_err_t esp_console_cmd_register(const esp_console_cmd_t *cmd)
{
cmd_item_t *item = NULL;
if (!cmd || cmd->command == NULL) {
return ESP_ERR_INVALID_ARG;
}{...}
if (strchr(cmd->command, ' ') != NULL) {
return ESP_ERR_INVALID_ARG;
}{...}
if ((cmd->func == NULL && cmd->func_w_context == NULL)
|| (cmd->func != NULL && cmd->func_w_context != NULL)) {
return ESP_ERR_INVALID_ARG;
}{...}
item = (cmd_item_t *)find_command_by_name(cmd->command);
if (!item) {
item = heap_caps_calloc(1, sizeof(*item), s_config.heap_alloc_caps);
if (item == NULL) {
return ESP_ERR_NO_MEM;
}{...}
}{...} else {
esp_console_rm_item_free_hint(item);
}{...}
item->command = cmd->command;
item->help = cmd->help;
if (cmd->hint) {
/* ... */
int unused __attribute__((unused));
unused = asprintf(&item->hint, " %s", cmd->hint);
}{...} else if (cmd->argtable) {
arg_dstr_t ds = arg_dstr_create();
arg_print_syntax_ds(ds, cmd->argtable, NULL);
item->hint = strdup(arg_dstr_cstr(ds));
arg_dstr_destroy(ds);
}{...}
item->argtable = cmd->argtable;
if (cmd->func) {
item->func = cmd->func;
}{...} else {
item->func_w_context = cmd->func_w_context;
item->context = cmd->context;
}{...}
cmd_item_t *last;
cmd_item_t *it;
#if CONFIG_CONSOLE_SORTED_HELP
last = NULL;
SLIST_FOREACH(it, &s_cmd_list, next) {
if (strcmp(it->command, item->command) > 0) {
break;
}{...}
last = it;
}{...}
#else/* ... */
last = SLIST_FIRST(&s_cmd_list);
#endif
if (last == NULL) {
SLIST_INSERT_HEAD(&s_cmd_list, item, next);
}{...} else {
#if !CONFIG_CONSOLE_SORTED_HELP
while ((it = SLIST_NEXT(last, next)) != NULL) {
last = it;
}{...}
#endif/* ... */
SLIST_INSERT_AFTER(last, item, next);
}{...}
return ESP_OK;
}{ ... }
void esp_console_get_completion(const char *buf, linenoiseCompletions *lc)
{
size_t len = strlen(buf);
if (len == 0) {
return;
}{...}
cmd_item_t *it;
SLIST_FOREACH(it, &s_cmd_list, next) {
if (strncmp(buf, it->command, len) == 0) {
linenoiseAddCompletion(lc, it->command);
}{...}
}{...}
}{ ... }
const char *esp_console_get_hint(const char *buf, int *color, int *bold)
{
size_t len = strlen(buf);
cmd_item_t *it;
SLIST_FOREACH(it, &s_cmd_list, next) {
if (strlen(it->command) == len &&
strncmp(buf, it->command, len) == 0) {
*color = s_config.hint_color;
*bold = s_config.hint_bold;
return it->hint;
}{...}
}{...}
return NULL;
}{ ... }
static const cmd_item_t *find_command_by_name(const char *name)
{
const cmd_item_t *cmd = NULL;
cmd_item_t *it;
size_t len = strlen(name);
SLIST_FOREACH(it, &s_cmd_list, next) {
if (strlen(it->command) == len &&
strcmp(name, it->command) == 0) {
cmd = it;
break;
}{...}
}{...}
return cmd;
}{ ... }
esp_err_t esp_console_run(const char *cmdline, int *cmd_ret)
{
if (s_tmp_line_buf == NULL) {
return ESP_ERR_INVALID_STATE;
}{...}
char **argv = (char **) heap_caps_calloc(s_config.max_cmdline_args, sizeof(char *), s_config.heap_alloc_caps);
if (argv == NULL) {
return ESP_ERR_NO_MEM;
}{...}
strlcpy(s_tmp_line_buf, cmdline, s_config.max_cmdline_length);
size_t argc = esp_console_split_argv(s_tmp_line_buf, argv,
s_config.max_cmdline_args);
if (argc == 0) {
free(argv);
return ESP_ERR_INVALID_ARG;
}{...}
const cmd_item_t *cmd = find_command_by_name(argv[0]);
if (cmd == NULL) {
free(argv);
return ESP_ERR_NOT_FOUND;
}{...}
if (cmd->func) {
*cmd_ret = (*cmd->func)(argc, argv);
}{...}
if (cmd->func_w_context) {
*cmd_ret = (*cmd->func_w_context)(cmd->context, argc, argv);
}{...}
free(argv);
return ESP_OK;
}{ ... }
static struct {
struct arg_str *help_cmd;
struct arg_int *verbose_level;
struct arg_end *end;
}{ ... } help_args;
static void print_arg_help(cmd_item_t *it)
{
/* ... */
const char *hint = (it->hint) ? it->hint : "";
printf("%-s %s\n", it->command, hint);
/* ... */
printf(" ");
arg_print_formatted(stdout, 2, 78, it->help);
if (it->argtable) {
arg_print_glossary(stdout, (void **) it->argtable, " %12s %s\n");
}{...}
printf("\n");
}{ ... }
static void print_arg_command(cmd_item_t *it)
{
const char *hint = (it->hint) ? it->hint : "";
printf("%-s %s\n\n", it->command, hint);
}{ ... }
static fn_print_arg_t print_verbose_level_arr[ESP_CONSOLE_HELP_VERBOSE_LEVEL_MAX_NUM] = {
print_arg_command,
print_arg_help,
}{...};
static int help_command(int argc, char **argv)
{
int nerrors = arg_parse(argc, argv, (void **) &help_args);
if (nerrors != 0) {
arg_print_errors(stderr, help_args.end, argv[0]);
return 1;
}{...}
cmd_item_t *it;
int ret_value = 1;
esp_console_help_verbose_level_e verbose_level;
if (help_args.help_cmd->count == 0) {
if (help_args.verbose_level->count == 0) {
verbose_level = s_verbose_level;
}{...}
else {
verbose_level = (esp_console_help_verbose_level_e)help_args.verbose_level->ival[0];
}{...}
if (verbose_level >= ESP_CONSOLE_HELP_VERBOSE_LEVEL_MAX_NUM) {
printf("help: invalid verbose level %d", (int)verbose_level);
return 1;
}{...}
SLIST_FOREACH(it, &s_cmd_list, next) {
if (it->help == NULL) {
continue;
}{...}
print_verbose_level_arr[verbose_level](it);
}{...}
ret_value = 0;
}{...} else {
bool found_command = false;
SLIST_FOREACH(it, &s_cmd_list, next) {
if (it->help == NULL) {
continue;
}{...}
if (strcmp(help_args.help_cmd->sval[0], it->command) == 0) {
print_arg_help(it);
found_command = true;
ret_value = 0;
break;
}{...}
}{...}
if (!found_command) {
printf("help: Unrecognized option '%s'. Please use correct command as argument "
"or type 'help' only to print help for all commands\n", help_args.help_cmd->sval[0]);
}{...}
}{...}
return ret_value;
}{ ... }
esp_err_t esp_console_register_help_command(void)
{
help_args.help_cmd = arg_str0(NULL, NULL, "<string>", "Name of command");
help_args.verbose_level = arg_intn("v", "verbose", "<0|1>", 0, 1,
"If specified, list console commands with given verbose level");
help_args.end = arg_end(2);
esp_console_cmd_t command = {
.command = "help",
.help = "Print the summary of all registered commands if no arguments "
"are given, otherwise print summary of given command.",
.func = &help_command,
.argtable = &help_args
}{...};
return esp_console_cmd_register(&command);
}{ ... }
esp_err_t esp_console_set_help_verbose_level(esp_console_help_verbose_level_e verbose_level)
{
if (verbose_level >= ESP_CONSOLE_HELP_VERBOSE_LEVEL_MAX_NUM) {
return ESP_ERR_INVALID_ARG;
}{...}
s_verbose_level = verbose_level;
return ESP_OK;
}{ ... }