help: re-implement 'help' independent from tree of struct command 70/5670/7
authorAntonio Borneo <borneo.antonio@gmail.com>
Sat, 27 Mar 2021 15:12:11 +0000 (16:12 +0100)
committerAntonio Borneo <borneo.antonio@gmail.com>
Sun, 18 Apr 2021 14:33:54 +0000 (15:33 +0100)
The current implementation of "help" related commands is tightly
connected to the tree of struct command.
The TCL commands 'add_usage_text' and 'add_help_text' have to add
fake commands in the tree of struct command to handle the help of
TCL procs.

Move all the help texts in a list accessible from the struct
command_context and register the commands through their full name.
Keep the list sorted alphabetically by the command name, so the
result of commands 'help' and 'usage' will be sorted too.

Remove the associated help and usage during commands un-register,
but call help_del_all_commands() for the text added through TCL
commands 'add_usage_text' and 'add_help_text'.

The resulting help and usage output is not changed by this patch
(tested on all the help and usage strings in current master
branch).

Change-Id: Ifd37bb5bd374cba1a22cd7aac208505b4ae1e6fc
Signed-off-by: Antonio Borneo <borneo.antonio@gmail.com>
Reviewed-on: http://openocd.zylin.com/5670
Tested-by: jenkins
Reviewed-by: Oleksij Rempel <linux@rempel-privat.de>
src/helper/command.c
src/helper/command.h
src/openocd.c

index 288ba99aa874ba53518e40eb64ab8dfb98757d6f..3b531807f3dd1aba1d0364aaf6278c84d7982905 100644 (file)
@@ -52,6 +52,9 @@ struct log_capture_state {
 static int unregister_command(struct command_context *context,
        struct command *parent, const char *name);
 static char *command_name(struct command *c, char delim);
+static int help_add_command(struct command_context *cmd_ctx,
+       const char *cmd_name, const char *help_text, const char *usage_text);
+static int help_del_command(struct command_context *cmd_ctx, const char *cmd_name);
 
 /* wrap jimtcl internal data */
 static inline bool jimcmd_is_proc(Jim_Cmd *cmd)
@@ -293,8 +296,6 @@ static void command_free(struct command *c)
        }
 
        free(c->name);
-       free(c->help);
-       free(c->usage);
        free(c);
 }
 
@@ -323,12 +324,7 @@ static struct command *command_new(struct command_context *cmd_ctx,
                return NULL;
 
        c->name = strdup(cr->name);
-       if (cr->help)
-               c->help = strdup(cr->help);
-       if (cr->usage)
-               c->usage = strdup(cr->usage);
-
-       if (!c->name || (cr->help && !c->help) || (cr->usage && !c->usage))
+       if (!c->name)
                goto command_new_error;
 
        c->parent = parent;
@@ -338,6 +334,12 @@ static struct command *command_new(struct command_context *cmd_ctx,
 
        command_add_child(command_list_for_parent(cmd_ctx, parent), c);
 
+       if (cr->help || cr->usage) {
+               char *full_name = command_name(c, ' ');
+               help_add_command(cmd_ctx, full_name, cr->help, cr->usage);
+               free(full_name);
+       }
+
        return c;
 
 command_new_error:
@@ -464,6 +466,10 @@ static int unregister_command(struct command_context *context,
                if (strcmp(name, c->name) != 0)
                        continue;
 
+               char *full_name = command_name(c, ' ');
+               help_del_command(context, full_name);
+               free(full_name);
+
                if (p)
                        p->next = c->next;
                else
@@ -785,28 +791,22 @@ static int jim_capture(Jim_Interp *interp, int argc, Jim_Obj *const *argv)
        return retcode;
 }
 
-static COMMAND_HELPER(command_help_find, struct command *head,
-       struct command **out)
-{
-       if (0 == CMD_ARGC)
-               return ERROR_COMMAND_SYNTAX_ERROR;
-       *out = command_find(head, CMD_ARGV[0]);
-       if (NULL == *out)
-               return ERROR_COMMAND_SYNTAX_ERROR;
-       if (--CMD_ARGC == 0)
-               return ERROR_OK;
-       CMD_ARGV++;
-       return CALL_COMMAND_HANDLER(command_help_find, (*out)->children, out);
-}
+struct help_entry {
+       struct list_head lh;
+       char *cmd_name;
+       char *help;
+       char *usage;
+};
 
-static COMMAND_HELPER(command_help_show, struct command *c, unsigned n,
+static COMMAND_HELPER(command_help_show, struct help_entry *c,
        bool show_help, const char *cmd_match);
 
-static COMMAND_HELPER(command_help_show_list, struct command *head, unsigned n,
-       bool show_help, const char *cmd_match)
+static COMMAND_HELPER(command_help_show_list, bool show_help, const char *cmd_match)
 {
-       for (struct command *c = head; NULL != c; c = c->next)
-               CALL_COMMAND_HANDLER(command_help_show, c, n, show_help, cmd_match);
+       struct help_entry *entry;
+
+       list_for_each_entry(entry, CMD_CTX->help_list, lh)
+               CALL_COMMAND_HANDLER(command_help_show, entry, show_help, cmd_match);
        return ERROR_OK;
 }
 
@@ -837,26 +837,23 @@ static void command_help_show_wrap(const char *str, unsigned n, unsigned n2)
        }
 }
 
-static COMMAND_HELPER(command_help_show, struct command *c, unsigned n,
+static COMMAND_HELPER(command_help_show, struct help_entry *c,
        bool show_help, const char *cmd_match)
 {
-       char *cmd_name = command_name(c, ' ');
-       if (NULL == cmd_name)
-               return ERROR_FAIL;
+       unsigned int n = 0;
+       for (const char *s = strchr(c->cmd_name, ' '); s; s = strchr(s + 1, ' '))
+               n++;
 
        /* If the match string occurs anywhere, we print out
         * stuff for this command. */
-       bool is_match = (strstr(cmd_name, cmd_match) != NULL) ||
+       bool is_match = (strstr(c->cmd_name, cmd_match) != NULL) ||
                ((c->usage != NULL) && (strstr(c->usage, cmd_match) != NULL)) ||
                ((c->help != NULL) && (strstr(c->help, cmd_match) != NULL));
 
        if (is_match) {
                command_help_show_indent(n);
-               LOG_USER_N("%s", cmd_name);
-       }
-       free(cmd_name);
+               LOG_USER_N("%s", c->cmd_name);
 
-       if (is_match) {
                if (c->usage && strlen(c->usage) > 0) {
                        LOG_USER_N(" ");
                        command_help_show_wrap(c->usage, 0, n + 5);
@@ -867,11 +864,30 @@ static COMMAND_HELPER(command_help_show, struct command *c, unsigned n,
        if (is_match && show_help) {
                char *msg;
 
+               /* TODO: factorize jim_command_mode() to avoid running jim command here */
+               char *request = alloc_printf("command mode %s", c->cmd_name);
+               if (!request) {
+                       LOG_ERROR("Out of memory");
+                       return ERROR_FAIL;
+               }
+               int retval = Jim_Eval(CMD_CTX->interp, request);
+               free(request);
+               enum command_mode mode = COMMAND_UNKNOWN;
+               if (retval != JIM_ERR) {
+                       const char *result = Jim_GetString(Jim_GetResult(CMD_CTX->interp), NULL);
+                       if (!strcmp(result, "any"))
+                               mode = COMMAND_ANY;
+                       else if (!strcmp(result, "config"))
+                               mode = COMMAND_CONFIG;
+                       else if (!strcmp(result, "exec"))
+                               mode = COMMAND_EXEC;
+               }
+
                /* Normal commands are runtime-only; highlight exceptions */
-               if (c->mode != COMMAND_EXEC) {
+               if (mode != COMMAND_EXEC) {
                        const char *stage_msg = "";
 
-                       switch (c->mode) {
+                       switch (mode) {
                                case COMMAND_CONFIG:
                                        stage_msg = " (configuration command)";
                                        break;
@@ -893,20 +909,13 @@ static COMMAND_HELPER(command_help_show, struct command *c, unsigned n,
                        return -ENOMEM;
        }
 
-       if (++n > 5) {
-               LOG_ERROR("command recursion exceeded");
-               return ERROR_FAIL;
-       }
-
-       return CALL_COMMAND_HANDLER(command_help_show_list,
-               c->children, n, show_help, cmd_match);
+       return ERROR_OK;
 }
 
 COMMAND_HANDLER(handle_help_command)
 {
        bool full = strcmp(CMD_NAME, "help") == 0;
        int retval;
-       struct command *c = CMD_CTX->commands;
        char *cmd_match;
 
        if (CMD_ARGC <= 0)
@@ -926,8 +935,7 @@ COMMAND_HANDLER(handle_help_command)
                LOG_ERROR("unable to build search string");
                return -ENOMEM;
        }
-       retval = CALL_COMMAND_HANDLER(command_help_show_list,
-                       c, 0, full, cmd_match);
+       retval = CALL_COMMAND_HANDLER(command_help_show_list, full, cmd_match);
 
        free(cmd_match);
        return retval;
@@ -1124,81 +1132,104 @@ static int jim_command_mode(Jim_Interp *interp, int argc, Jim_Obj *const *argv)
        return JIM_OK;
 }
 
-int help_add_command(struct command_context *cmd_ctx, struct command *parent,
-       const char *cmd_name, const char *help_text, const char *usage)
+int help_del_all_commands(struct command_context *cmd_ctx)
+{
+       struct help_entry *curr, *n;
+
+       list_for_each_entry_safe(curr, n, cmd_ctx->help_list, lh) {
+               list_del(&curr->lh);
+               free(curr->cmd_name);
+               free(curr->help);
+               free(curr->usage);
+               free(curr);
+       }
+       return ERROR_OK;
+}
+
+static int help_del_command(struct command_context *cmd_ctx, const char *cmd_name)
 {
-       struct command **head = command_list_for_parent(cmd_ctx, parent);
-       struct command *nc = command_find(*head, cmd_name);
-       if (NULL == nc) {
-               /* add a new command with help text */
-               struct command_registration cr = {
-                       .name = cmd_name,
-                       .mode = COMMAND_ANY,
-                       .help = help_text,
-                       .usage = usage ? : "",
-               };
-               nc = register_command(cmd_ctx, parent, &cr);
-               if (NULL == nc) {
-                       LOG_ERROR("failed to add '%s' help text", cmd_name);
+       struct help_entry *curr;
+
+       list_for_each_entry(curr, cmd_ctx->help_list, lh) {
+               if (!strcmp(cmd_name, curr->cmd_name)) {
+                       list_del(&curr->lh);
+                       free(curr->cmd_name);
+                       free(curr->help);
+                       free(curr->usage);
+                       free(curr);
+                       break;
+               }
+       }
+
+       return ERROR_OK;
+}
+
+static int help_add_command(struct command_context *cmd_ctx,
+       const char *cmd_name, const char *help_text, const char *usage_text)
+{
+       int cmp = -1; /* add after curr */
+       struct help_entry *curr;
+
+       list_for_each_entry_reverse(curr, cmd_ctx->help_list, lh) {
+               cmp = strcmp(cmd_name, curr->cmd_name);
+               if (cmp >= 0)
+                       break;
+       }
+
+       struct help_entry *entry;
+       if (cmp) {
+               entry = calloc(1, sizeof(*entry));
+               if (!entry) {
+                       LOG_ERROR("Out of memory");
                        return ERROR_FAIL;
                }
-               LOG_DEBUG("added '%s' help text", cmd_name);
-               return ERROR_OK;
+               entry->cmd_name = strdup(cmd_name);
+               if (!entry->cmd_name) {
+                       LOG_ERROR("Out of memory");
+                       free(entry);
+                       return ERROR_FAIL;
+               }
+               list_add(&entry->lh, &curr->lh);
+       } else {
+               entry = curr;
        }
+
        if (help_text) {
-               bool replaced = false;
-               if (nc->help) {
-                       free(nc->help);
-                       replaced = true;
+               char *text = strdup(help_text);
+               if (!text) {
+                       LOG_ERROR("Out of memory");
+                       return ERROR_FAIL;
                }
-               nc->help = strdup(help_text);
-               if (replaced)
-                       LOG_INFO("replaced existing '%s' help", cmd_name);
-               else
-                       LOG_DEBUG("added '%s' help text", cmd_name);
+               free(entry->help);
+               entry->help = text;
        }
-       if (usage) {
-               bool replaced = false;
-               if (nc->usage) {
-                       if (*nc->usage)
-                               replaced = true;
-                       free(nc->usage);
+
+       if (usage_text) {
+               char *text = strdup(usage_text);
+               if (!text) {
+                       LOG_ERROR("Out of memory");
+                       return ERROR_FAIL;
                }
-               nc->usage = strdup(usage);
-               if (replaced)
-                       LOG_INFO("replaced existing '%s' usage", cmd_name);
-               else
-                       LOG_DEBUG("added '%s' usage text", cmd_name);
+               free(entry->usage);
+               entry->usage = text;
        }
+
        return ERROR_OK;
 }
 
 COMMAND_HANDLER(handle_help_add_command)
 {
-       if (CMD_ARGC < 2) {
-               LOG_ERROR("%s: insufficient arguments", CMD_NAME);
+       if (CMD_ARGC != 2)
                return ERROR_COMMAND_SYNTAX_ERROR;
-       }
 
-       /* save help text and remove it from argument list */
-       const char *str = CMD_ARGV[--CMD_ARGC];
-       const char *help = !strcmp(CMD_NAME, "add_help_text") ? str : NULL;
-       const char *usage = !strcmp(CMD_NAME, "add_usage_text") ? str : NULL;
+       const char *help = !strcmp(CMD_NAME, "add_help_text") ? CMD_ARGV[1] : NULL;
+       const char *usage = !strcmp(CMD_NAME, "add_usage_text") ? CMD_ARGV[1] : NULL;
        if (!help && !usage) {
                LOG_ERROR("command name '%s' is unknown", CMD_NAME);
                return ERROR_COMMAND_SYNTAX_ERROR;
        }
-       /* likewise for the leaf command name */
-       const char *cmd_name = CMD_ARGV[--CMD_ARGC];
-
-       struct command *c = NULL;
-       if (CMD_ARGC > 0) {
-               c = CMD_CTX->commands;
-               int retval = CALL_COMMAND_HANDLER(command_help_find, c, &c);
-               if (ERROR_OK != retval)
-                       return retval;
-       }
-       return help_add_command(CMD_CTX, c, cmd_name, help, usage);
+       const char *cmd_name = CMD_ARGV[0];
+       return help_add_command(CMD_CTX, cmd_name, help, usage);
 }
 
 /* sleep command sleeps for <n> milliseconds
@@ -1329,6 +1360,10 @@ struct command_context *command_init(const char *startup_tcl, Jim_Interp *interp
 
        context->mode = COMMAND_EXEC;
 
+       /* context can be duplicated. Put list head on separate mem-chunk to keep list consistent */
+       context->help_list = malloc(sizeof(*context->help_list));
+       INIT_LIST_HEAD(context->help_list);
+
        /* Create a jim interpreter if we were not handed one */
        if (interp == NULL) {
                /* Create an interpreter */
@@ -1393,6 +1428,7 @@ void command_exit(struct command_context *context)
                return;
 
        Jim_FreeInterp(context->interp);
+       free(context->help_list);
        command_done(context);
 }
 
index 9a04e9fa100b1ceb10f3bb489164f8119128a3bf..db095972af3725fa0921ad90799408cfa4572ce2 100644 (file)
@@ -26,6 +26,7 @@
 #include <stdbool.h>
 #include <jim-nvp.h>
 
+#include <helper/list.h>
 #include <helper/types.h>
 
 /* To achieve C99 printf compatibility in MinGW, gnu_printf should be
@@ -41,6 +42,7 @@ enum command_mode {
        COMMAND_EXEC,
        COMMAND_CONFIG,
        COMMAND_ANY,
+       COMMAND_UNKNOWN = -1, /* error condition */
 };
 
 struct command_context;
@@ -64,6 +66,7 @@ struct command_context {
                 */
        command_output_handler_t output_handler;
        void *output_handler_priv;
+       struct list_head *help_list;
 };
 
 struct command;
@@ -179,8 +182,6 @@ typedef __COMMAND_HANDLER((*command_handler_t));
 
 struct command {
        char *name;
-       char *help;
-       char *usage;
        struct command *parent;
        struct command *children;
        command_handler_t handler;
@@ -316,6 +317,14 @@ static inline int register_commands_with_data(struct command_context *cmd_ctx,
 int unregister_all_commands(struct command_context *cmd_ctx,
                struct command *parent);
 
+/**
+ * Unregisters the help for all commands. Used at exit to remove the help
+ * added through the commands 'add_help_text' and 'add_usage_text'.
+ * @param cmd_ctx The context that will be cleared of registered helps.
+ * @returns ERROR_OK on success, or an error code.
+ */
+int help_del_all_commands(struct command_context *cmd_ctx);
+
 void command_set_output_handler(struct command_context *context,
                command_output_handler_t output_handler, void *priv);
 
index fcefdbe70fae5eef42879c666e5749deeab77668..32b68b6fc2cabfcf90e6abac29d90afa708d0098 100644 (file)
@@ -361,6 +361,7 @@ int openocd_main(int argc, char *argv[])
        server_free();
 
        unregister_all_commands(cmd_ctx, NULL);
+       help_del_all_commands(cmd_ctx);
 
        /* free all DAP and CTI objects */
        dap_cleanup_all();

Linking to existing account procedure

If you already have an account and want to add another login method you MUST first sign in with your existing account and then change URL to read https://review.openocd.org/login/?link to get to this page again but this time it'll work for linking. Thank you.

SSH host keys fingerprints

1024 SHA256:YKx8b7u5ZWdcbp7/4AeXNaqElP49m6QrwfXaqQGJAOk gerrit-code-review@openocd.zylin.com (DSA)
384 SHA256:jHIbSQa4REvwCFG4cq5LBlBLxmxSqelQPem/EXIrxjk gerrit-code-review@openocd.org (ECDSA)
521 SHA256:UAOPYkU9Fjtcao0Ul/Rrlnj/OsQvt+pgdYSZ4jOYdgs gerrit-code-review@openocd.org (ECDSA)
256 SHA256:A13M5QlnozFOvTllybRZH6vm7iSt0XLxbA48yfc2yfY gerrit-code-review@openocd.org (ECDSA)
256 SHA256:spYMBqEYoAOtK7yZBrcwE8ZpYt6b68Cfh9yEVetvbXg gerrit-code-review@openocd.org (ED25519)
+--[ED25519 256]--+
|=..              |
|+o..   .         |
|*.o   . .        |
|+B . . .         |
|Bo. = o S        |
|Oo.+ + =         |
|oB=.* = . o      |
| =+=.+   + E     |
|. .=o   . o      |
+----[SHA256]-----+
2048 SHA256:0Onrb7/PHjpo6iVZ7xQX2riKN83FJ3KGU0TvI0TaFG4 gerrit-code-review@openocd.zylin.com (RSA)