helper/command: fix memory leak on malloc() fail
[openocd.git] / src / helper / command.c
index 3b531807f3dd1aba1d0364aaf6278c84d7982905..08d14b47fe5f2a803c1703098513035e1724e056 100644 (file)
@@ -50,18 +50,28 @@ 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);
+       const char *cmd_prefix, const char *name);
+static int jim_command_dispatch(Jim_Interp *interp, int argc, Jim_Obj * const *argv);
 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 */
+/* set of functions to wrap jimtcl internal data */
 static inline bool jimcmd_is_proc(Jim_Cmd *cmd)
 {
        return cmd->isproc;
 }
 
+static inline bool jimcmd_is_ocd_command(Jim_Cmd *cmd)
+{
+       return !cmd->isproc && cmd->u.native.cmdProc == jim_command_dispatch;
+}
+
+static inline void *jimcmd_privdata(Jim_Cmd *cmd)
+{
+       return cmd->isproc ? NULL : cmd->u.native.privData;
+}
+
 static void tcl_output(void *privData, const char *file, unsigned line,
        const char *function, const char *string)
 {
@@ -73,17 +83,21 @@ static struct log_capture_state *command_log_capture_start(Jim_Interp *interp)
 {
        /* capture log output and return it. A garbage collect can
         * happen, so we need a reference count to this object */
-       Jim_Obj *tclOutput = Jim_NewStringObj(interp, "", 0);
-       if (NULL == tclOutput)
+       Jim_Obj *jim_output = Jim_NewStringObj(interp, "", 0);
+       if (!jim_output)
                return NULL;
 
+       Jim_IncrRefCount(jim_output);
+
        struct log_capture_state *state = malloc(sizeof(*state));
-       if (NULL == state)
+       if (!state) {
+               LOG_ERROR("Out of memory");
+               Jim_DecrRefCount(interp, jim_output);
                return NULL;
+       }
 
        state->interp = interp;
-       Jim_IncrRefCount(tclOutput);
-       state->output = tclOutput;
+       state->output = jim_output;
 
        log_add_callback(tcl_output, state);
 
@@ -170,7 +184,7 @@ extern struct command_context *global_cmd_ctx;
 
 /* dump a single line to the log for the command.
  * Do nothing in case we are not at debug level 3 */
-void script_debug(Jim_Interp *interp, unsigned int argc, Jim_Obj * const *argv)
+static void script_debug(Jim_Interp *interp, unsigned int argc, Jim_Obj * const *argv)
 {
        if (debug_level < LOG_LVL_DEBUG)
                return;
@@ -232,75 +246,28 @@ struct command_context *current_command_context(Jim_Interp *interp)
        return cmd_ctx;
 }
 
-static struct command *command_root(struct command *c)
-{
-       while (NULL != c->parent)
-               c = c->parent;
-       return c;
-}
-
 /**
- * Find a command by name from a list of commands.
- * @returns Returns the named command if it exists in the list.
+ * Find a openocd command from fullname.
+ * @returns Returns the named command if it is registred in interp.
  * Returns NULL otherwise.
  */
-static struct command *command_find(struct command *head, const char *name)
-{
-       for (struct command *cc = head; cc; cc = cc->next) {
-               if (strcmp(cc->name, name) == 0)
-                       return cc;
-       }
-       return NULL;
-}
-
-/**
- * Add the command into the linked list, sorted by name.
- * @param head Address to head of command list pointer, which may be
- * updated if @c c gets inserted at the beginning of the list.
- * @param c The command to add to the list pointed to by @c head.
- */
-static void command_add_child(struct command **head, struct command *c)
-{
-       assert(head);
-       if (NULL == *head) {
-               *head = c;
-               return;
-       }
-
-       while ((*head)->next && (strcmp(c->name, (*head)->name) > 0))
-               head = &(*head)->next;
-
-       if (strcmp(c->name, (*head)->name) > 0) {
-               c->next = (*head)->next;
-               (*head)->next = c;
-       } else {
-               c->next = *head;
-               *head = c;
-       }
-}
-
-static struct command **command_list_for_parent(
-       struct command_context *cmd_ctx, struct command *parent)
-{
-       return parent ? &parent->children : &cmd_ctx->commands;
-}
-
-static void command_free(struct command *c)
+static struct command *command_find_from_name(Jim_Interp *interp, const char *name)
 {
-       /** @todo if command has a handler, unregister its jim command! */
+       if (!name)
+               return NULL;
 
-       while (NULL != c->children) {
-               struct command *tmp = c->children;
-               c->children = tmp->next;
-               command_free(tmp);
-       }
+       Jim_Obj *jim_name = Jim_NewStringObj(interp, name, -1);
+       Jim_IncrRefCount(jim_name);
+       Jim_Cmd *cmd = Jim_GetCommand(interp, jim_name, JIM_NONE);
+       Jim_DecrRefCount(interp, jim_name);
+       if (!cmd || jimcmd_is_proc(cmd) || !jimcmd_is_ocd_command(cmd))
+               return NULL;
 
-       free(c->name);
-       free(c);
+       return jimcmd_privdata(cmd);
 }
 
 static struct command *command_new(struct command_context *cmd_ctx,
-       struct command *parent, const struct command_registration *cr)
+       const char *full_name, const struct command_registration *cr)
 {
        assert(cr->name);
 
@@ -311,89 +278,86 @@ static struct command *command_new(struct command_context *cmd_ctx,
         * strlen(.usage) == 0 means that the command takes no
         * arguments.
        */
-       if ((cr->jim_handler == NULL) && (cr->usage == NULL)) {
-               LOG_ERROR("BUG: command '%s%s%s' does not have the "
+       if (!cr->jim_handler && !cr->usage)
+               LOG_ERROR("BUG: command '%s' does not have the "
                        "'.usage' field filled out",
-                       parent && parent->name ? parent->name : "",
-                       parent && parent->name ? " " : "",
-                       cr->name);
-       }
+                       full_name);
 
        struct command *c = calloc(1, sizeof(struct command));
        if (NULL == c)
                return NULL;
 
        c->name = strdup(cr->name);
-       if (!c->name)
-               goto command_new_error;
+       if (!c->name) {
+               free(c);
+               return NULL;
+       }
 
-       c->parent = parent;
        c->handler = cr->handler;
        c->jim_handler = cr->jim_handler;
        c->mode = cr->mode;
 
-       command_add_child(command_list_for_parent(cmd_ctx, parent), c);
-
-       if (cr->help || cr->usage) {
-               char *full_name = command_name(c, ' ');
+       if (cr->help || cr->usage)
                help_add_command(cmd_ctx, full_name, cr->help, cr->usage);
-               free(full_name);
-       }
 
        return c;
-
-command_new_error:
-       command_free(c);
-       return NULL;
 }
 
-static int command_unknown(Jim_Interp *interp, int argc, Jim_Obj *const *argv);
-
-static int register_command_handler(struct command_context *cmd_ctx,
-       struct command *c)
+static void command_free(struct Jim_Interp *interp, void *priv)
 {
-       Jim_Interp *interp = cmd_ctx->interp;
-
-#if 0
-       LOG_DEBUG("registering '%s'...", c->name);
-#endif
+       struct command *c = priv;
 
-       return Jim_CreateCommand(interp, c->name, command_unknown, c, NULL);
+       free(c->name);
+       free(c);
 }
 
 static struct command *register_command(struct command_context *context,
-       struct command *parent, const struct command_registration *cr)
+       const char *cmd_prefix, const struct command_registration *cr)
 {
+       char *full_name;
+
        if (!context || !cr->name)
                return NULL;
 
-       const char *name = cr->name;
-       struct command **head = command_list_for_parent(context, parent);
-       struct command *c = command_find(*head, name);
-       if (NULL != c) {
+       if (cmd_prefix)
+               full_name = alloc_printf("%s %s", cmd_prefix, cr->name);
+       else
+               full_name = strdup(cr->name);
+       if (!full_name)
+               return NULL;
+
+       struct command *c = command_find_from_name(context->interp, full_name);
+       if (c) {
                /* TODO: originally we treated attempting to register a cmd twice as an error
                 * Sometimes we need this behaviour, such as with flash banks.
                 * http://www.mail-archive.com/openocd-development@lists.berlios.de/msg11152.html */
-               LOG_DEBUG("command '%s' is already registered in '%s' context",
-                       name, parent ? parent->name : "<global>");
+               LOG_DEBUG("command '%s' is already registered", full_name);
+               free(full_name);
                return c;
        }
 
-       c = command_new(context, parent, cr);
-       if (NULL == c)
+       c = command_new(context, full_name, cr);
+       if (!c) {
+               free(full_name);
                return NULL;
+       }
 
-       if (cr->jim_handler || cr->handler) {
-               int retval = register_command_handler(context, command_root(c));
-               if (retval != JIM_OK) {
-                       unregister_command(context, parent, name);
-                       return NULL;
-               }
+       LOG_DEBUG("registering '%s'...", full_name);
+       int retval = Jim_CreateCommand(context->interp, full_name,
+                               jim_command_dispatch, c, command_free);
+       if (retval != JIM_OK) {
+               command_run_linef(context, "del_help_text {%s}", full_name);
+               command_run_linef(context, "del_usage_text {%s}", full_name);
+               free(c);
+               free(full_name);
+               return NULL;
        }
+
+       free(full_name);
        return c;
 }
 
-static int ___register_commands(struct command_context *cmd_ctx, struct command *parent,
+int __register_commands(struct command_context *cmd_ctx, const char *cmd_prefix,
        const struct command_registration *cmds, void *data,
        struct target *override_target)
 {
@@ -404,7 +368,7 @@ static int ___register_commands(struct command_context *cmd_ctx, struct command
 
                struct command *c = NULL;
                if (NULL != cr->name) {
-                       c = register_command(cmd_ctx, parent, cr);
+                       c = register_command(cmd_ctx, cmd_prefix, cr);
                        if (NULL == c) {
                                retval = ERROR_FAIL;
                                break;
@@ -413,73 +377,111 @@ static int ___register_commands(struct command_context *cmd_ctx, struct command
                        c->jim_override_target = override_target;
                }
                if (NULL != cr->chain) {
-                       struct command *p = c ? : parent;
-                       retval = ___register_commands(cmd_ctx, p, cr->chain, data, override_target);
+                       if (cr->name) {
+                               if (cmd_prefix) {
+                                       char *new_prefix = alloc_printf("%s %s", cmd_prefix, cr->name);
+                                       if (!new_prefix) {
+                                               retval = ERROR_FAIL;
+                                               break;
+                                       }
+                                       retval = __register_commands(cmd_ctx, new_prefix, cr->chain, data, override_target);
+                                       free(new_prefix);
+                               } else {
+                                       retval = __register_commands(cmd_ctx, cr->name, cr->chain, data, override_target);
+                               }
+                       } else {
+                               retval = __register_commands(cmd_ctx, cmd_prefix, cr->chain, data, override_target);
+                       }
                        if (ERROR_OK != retval)
                                break;
                }
        }
        if (ERROR_OK != retval) {
                for (unsigned j = 0; j < i; j++)
-                       unregister_command(cmd_ctx, parent, cmds[j].name);
+                       unregister_command(cmd_ctx, cmd_prefix, cmds[j].name);
        }
        return retval;
 }
 
-int __register_commands(struct command_context *cmd_ctx, const char *cmd_prefix,
-       const struct command_registration *cmds, void *data,
-       struct target *override_target)
+static __attribute__ ((format (PRINTF_ATTRIBUTE_FORMAT, 2, 3)))
+int unregister_commands_match(struct command_context *cmd_ctx, const char *format, ...)
 {
-       struct command *parent = NULL;
+       Jim_Interp *interp = cmd_ctx->interp;
+       va_list ap;
 
-       if (cmd_prefix)
-               parent = command_find(cmd_ctx->commands, cmd_prefix);
+       va_start(ap, format);
+       char *query = alloc_vprintf(format, ap);
+       va_end(ap);
+       if (!query)
+               return ERROR_FAIL;
+
+       char *query_cmd = alloc_printf("info commands {%s}", query);
+       free(query);
+       if (!query_cmd)
+               return ERROR_FAIL;
+
+       int retval = Jim_EvalSource(interp, __THIS__FILE__, __LINE__, query_cmd);
+       free(query_cmd);
+       if (retval != JIM_OK)
+               return ERROR_FAIL;
+
+       Jim_Obj *list = Jim_GetResult(interp);
+       Jim_IncrRefCount(list);
+
+       int len = Jim_ListLength(interp, list);
+       for (int i = 0; i < len; i++) {
+               Jim_Obj *elem = Jim_ListGetIndex(interp, list, i);
+               Jim_IncrRefCount(elem);
+
+               const char *name = Jim_GetString(elem, NULL);
+               struct command *c = command_find_from_name(interp, name);
+               if (!c) {
+                       /* not openocd command */
+                       Jim_DecrRefCount(interp, elem);
+                       continue;
+               }
+               LOG_DEBUG("delete command \"%s\"", name);
+#if JIM_VERSION >= 80
+               Jim_DeleteCommand(interp, elem);
+#else
+               Jim_DeleteCommand(interp, name);
+#endif
+
+               help_del_command(cmd_ctx, name);
+
+               Jim_DecrRefCount(interp, elem);
+       }
 
-       return ___register_commands(cmd_ctx, parent, cmds, data, override_target);
+       Jim_DecrRefCount(interp, list);
+       return ERROR_OK;
 }
 
 int unregister_all_commands(struct command_context *context,
-       struct command *parent)
+       const char *cmd_prefix)
 {
-       if (context == NULL)
+       if (!context)
                return ERROR_OK;
 
-       struct command **head = command_list_for_parent(context, parent);
-       while (NULL != *head) {
-               struct command *tmp = *head;
-               *head = tmp->next;
-               command_free(tmp);
-       }
+       if (!cmd_prefix || !*cmd_prefix)
+               return unregister_commands_match(context, "*");
 
-       return ERROR_OK;
+       int retval = unregister_commands_match(context, "%s *", cmd_prefix);
+       if (retval != ERROR_OK)
+               return retval;
+
+       return unregister_commands_match(context, "%s", cmd_prefix);
 }
 
 static int unregister_command(struct command_context *context,
-       struct command *parent, const char *name)
+       const char *cmd_prefix, const char *name)
 {
-       if ((!context) || (!name))
+       if (!context || !name)
                return ERROR_COMMAND_SYNTAX_ERROR;
 
-       struct command *p = NULL;
-       struct command **head = command_list_for_parent(context, parent);
-       for (struct command *c = *head; NULL != c; p = c, c = c->next) {
-               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
-                       *head = c->next;
+       if (!cmd_prefix || !*cmd_prefix)
+               return unregister_commands_match(context, "%s", name);
 
-               command_free(c);
-               return ERROR_OK;
-       }
-
-       return ERROR_OK;
+       return unregister_commands_match(context, "%s %s", cmd_prefix, name);
 }
 
 void command_output_text(struct command_context *context, const char *data)
@@ -536,34 +538,7 @@ void command_print(struct command_invocation *cmd, const char *format, ...)
        va_end(ap);
 }
 
-static char *__command_name(struct command *c, char delim, unsigned extra)
-{
-       char *name;
-       unsigned len = strlen(c->name);
-       if (NULL == c->parent) {
-               /* allocate enough for the name, child names, and '\0' */
-               name = malloc(len + extra + 1);
-               if (!name) {
-                       LOG_ERROR("Out of memory");
-                       return NULL;
-               }
-               strcpy(name, c->name);
-       } else {
-               /* parent's extra must include both the space and name */
-               name = __command_name(c->parent, delim, 1 + len + extra);
-               char dstr[2] = { delim, 0 };
-               strcat(name, dstr);
-               strcat(name, c->name);
-       }
-       return name;
-}
-
-static char *command_name(struct command *c, char delim)
-{
-       return __command_name(c, delim, 0);
-}
-
-static bool command_can_run(struct command_context *cmd_ctx, struct command *c)
+static bool command_can_run(struct command_context *cmd_ctx, struct command *c, const char *full_name)
 {
        if (c->mode == COMMAND_ANY || c->mode == cmd_ctx->mode)
                return true;
@@ -582,10 +557,8 @@ static bool command_can_run(struct command_context *cmd_ctx, struct command *c)
                        when = "if Cthulhu is summoned by";
                        break;
        }
-       char *full_name = command_name(c, ' ');
        LOG_ERROR("The '%s' command must be used %s 'init'.",
                        full_name ? full_name : c->name, when);
-       free(full_name);
        return false;
 }
 
@@ -606,20 +579,13 @@ static int run_command(struct command_context *context,
        int retval = c->handler(&cmd);
        if (retval == ERROR_COMMAND_SYNTAX_ERROR) {
                /* Print help for command */
-               char *full_name = command_name(c, ' ');
-               if (NULL != full_name) {
-                       command_run_linef(context, "usage %s", full_name);
-                       free(full_name);
-               }
+               command_run_linef(context, "usage %s", words[0]);
        } else if (retval == ERROR_COMMAND_CLOSE_CONNECTION) {
                /* just fall through for a shutdown request */
        } else {
-               if (retval != ERROR_OK) {
-                       char *full_name = command_name(c, ' ');
+               if (retval != ERROR_OK)
                        LOG_DEBUG("Command '%s' failed with error code %d",
-                                               full_name ? full_name : c->name, retval);
-                       free(full_name);
-               }
+                                               words[0], retval);
                /* Use the command output as the Tcl result */
                Jim_SetResult(context->interp, cmd.output);
        }
@@ -941,19 +907,6 @@ COMMAND_HANDLER(handle_help_command)
        return retval;
 }
 
-static int command_unknown_find(unsigned argc, Jim_Obj *const *argv,
-       struct command *head, struct command **out)
-{
-       if (0 == argc)
-               return argc;
-       const char *cmd_name = Jim_GetString(argv[0], NULL);
-       struct command *c = command_find(head, cmd_name);
-       if (NULL == c)
-               return argc;
-       *out = c;
-       return command_unknown_find(--argc, ++argv, (*out)->children, out);
-}
-
 static char *alloc_concatenate_strings(int argc, Jim_Obj * const *argv)
 {
        char *prev, *all;
@@ -980,41 +933,8 @@ static char *alloc_concatenate_strings(int argc, Jim_Obj * const *argv)
        return all;
 }
 
-static int run_usage(Jim_Interp *interp, int argc_valid, int argc, Jim_Obj * const *argv)
-{
-       struct command_context *cmd_ctx = current_command_context(interp);
-       char *command;
-       int retval;
-
-       assert(argc_valid >= 1);
-       assert(argc >= argc_valid);
-
-       command = alloc_concatenate_strings(argc_valid, argv);
-       if (!command)
-               return JIM_ERR;
-
-       retval = command_run_linef(cmd_ctx, "usage %s", command);
-       if (retval != ERROR_OK) {
-               LOG_ERROR("unable to execute command \"usage %s\"", command);
-               return JIM_ERR;
-       }
-
-       if (argc_valid == argc)
-               LOG_ERROR("%s: command requires more arguments", command);
-       else {
-               free(command);
-               command = alloc_concatenate_strings(argc - argc_valid, argv + argc_valid);
-               if (!command)
-                       return JIM_ERR;
-               LOG_ERROR("invalid subcommand \"%s\"", command);
-       }
-
-       free(command);
-       return retval;
-}
-
 static int exec_command(Jim_Interp *interp, struct command_context *cmd_ctx,
-               struct command *c, int argc, Jim_Obj *const *argv)
+               struct command *c, int argc, Jim_Obj * const *argv)
 {
        if (c->jim_handler)
                return c->jim_handler(interp, argc, argv);
@@ -1030,34 +950,34 @@ static int exec_command(Jim_Interp *interp, struct command_context *cmd_ctx,
        return command_retval_set(interp, retval);
 }
 
-static int command_unknown(Jim_Interp *interp, int argc, Jim_Obj *const *argv)
+static int jim_command_dispatch(Jim_Interp *interp, int argc, Jim_Obj * const *argv)
 {
        script_debug(interp, argc, argv);
 
-       struct command_context *cmd_ctx = current_command_context(interp);
-       struct command *c = cmd_ctx->commands;
-       int remaining = command_unknown_find(argc, argv, c, &c);
-       /* if nothing could be consumed, then it's really an unknown command */
-       if (remaining == argc) {
-               const char *cmd = Jim_GetString(argv[0], NULL);
-               LOG_ERROR("Unknown command:\n  %s", cmd);
-               return JIM_OK;
+       /* check subcommands */
+       if (argc > 1) {
+               char *s = alloc_printf("%s %s", Jim_GetString(argv[0], NULL), Jim_GetString(argv[1], NULL));
+               Jim_Obj *js = Jim_NewStringObj(interp, s, -1);
+               Jim_IncrRefCount(js);
+               free(s);
+               Jim_Cmd *cmd = Jim_GetCommand(interp, js, JIM_NONE);
+               if (cmd) {
+                       int retval = Jim_EvalObjPrefix(interp, js, argc - 2, argv + 2);
+                       Jim_DecrRefCount(interp, js);
+                       return retval;
+               }
+               Jim_DecrRefCount(interp, js);
        }
 
-       Jim_Obj *const *start;
-       unsigned count;
-       if (c->handler || c->jim_handler) {
-               /* include the command name in the list */
-               count = remaining + 1;
-               start = argv + (argc - remaining - 1);
-       } else {
-               count = argc - remaining;
-               start = argv;
-               run_usage(interp, count, argc, start);
+       struct command *c = jim_to_command(interp);
+       if (!c->jim_handler && !c->handler) {
+               Jim_EvalObjPrefix(interp, Jim_NewStringObj(interp, "usage", -1), 1, argv);
                return JIM_ERR;
        }
 
-       if (!command_can_run(cmd_ctx, c))
+       struct command_context *cmd_ctx = current_command_context(interp);
+
+       if (!command_can_run(cmd_ctx, c, Jim_GetString(argv[0], NULL)))
                return JIM_ERR;
 
        target_call_timer_callbacks_now();
@@ -1070,13 +990,13 @@ static int command_unknown(Jim_Interp *interp, int argc, Jim_Obj *const *argv)
         * current_target_override is used also for event handlers
         * therefore we prevent touching it if command has no prefix.
         * Previous override is saved and restored back to ensure
-        * correct work when command_unknown() is re-entered.
+        * correct work when jim_command_dispatch() is re-entered.
         */
        struct target *saved_target_override = cmd_ctx->current_target_override;
        if (c->jim_override_target)
                cmd_ctx->current_target_override = c->jim_override_target;
 
-       int retval = exec_command(interp, cmd_ctx, c, count, start);
+       int retval = exec_command(interp, cmd_ctx, c, argc, argv);
 
        if (c->jim_override_target)
                cmd_ctx->current_target_override = saved_target_override;
@@ -1098,18 +1018,19 @@ static int jim_command_mode(Jim_Interp *interp, int argc, Jim_Obj *const *argv)
                Jim_Cmd *cmd = Jim_GetCommand(interp, s, JIM_NONE);
                Jim_DecrRefCount(interp, s);
                free(full_name);
-               if (cmd && jimcmd_is_proc(cmd)) {
-                       Jim_SetResultString(interp, "any", -1);
-                       return JIM_OK;
-               }
-               struct command *c = cmd_ctx->commands;
-               int remaining = command_unknown_find(argc - 1, argv + 1, c, &c);
-               /* if nothing could be consumed, then it's an unknown command */
-               if (remaining == argc - 1) {
+               if (!cmd || !(jimcmd_is_proc(cmd) || jimcmd_is_ocd_command(cmd))) {
                        Jim_SetResultString(interp, "unknown", -1);
                        return JIM_OK;
                }
-               mode = c->mode;
+
+               if (jimcmd_is_proc(cmd)) {
+                       /* tcl proc */
+                       mode = COMMAND_ANY;
+               } else {
+                       struct command *c = jimcmd_privdata(cmd);
+
+                       mode = c->mode;
+               }
        } else
                mode = cmd_ctx->mode;
 

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)