shell_wildcard.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <string.h>
  7. #include <fnmatch.h>
  8. #include "shell_wildcard.h"
  9. #include "shell_utils.h"
  10. #include "shell_ops.h"
  11. static enum shell_wildcard_status command_add(char *buff, uint16_t *buff_len,
  12. char const *cmd,
  13. char const *pattern)
  14. {
  15. uint16_t cmd_len = z_shell_strlen(cmd);
  16. char *completion_addr;
  17. uint16_t shift;
  18. /* +1 for space */
  19. if ((*buff_len + cmd_len + 1) > CONFIG_SHELL_CMD_BUFF_SIZE) {
  20. return SHELL_WILDCARD_CMD_MISSING_SPACE;
  21. }
  22. completion_addr = strstr(buff, pattern);
  23. if (!completion_addr) {
  24. return SHELL_WILDCARD_CMD_NO_MATCH_FOUND;
  25. }
  26. shift = z_shell_strlen(completion_addr);
  27. /* make place for new command: + 1 for space + 1 for EOS */
  28. memmove(completion_addr + cmd_len + 1, completion_addr, shift + 1);
  29. memcpy(completion_addr, cmd, cmd_len);
  30. /* adding space to not brake next command in the buffer */
  31. completion_addr[cmd_len] = ' ';
  32. *buff_len += cmd_len + 1; /* + 1 for space */
  33. return SHELL_WILDCARD_CMD_ADDED;
  34. }
  35. /**
  36. * @internal @brief Function for searching and adding commands to the temporary
  37. * shell buffer matching to wildcard pattern.
  38. *
  39. * Function will search commands tree for commands matching wildcard pattern
  40. * stored in argv[cmd_lvl]. When match is found wildcard pattern will be
  41. * replaced by matching commands. If there is no space in the buffer to add all
  42. * matching commands function will add as many as possible. Next it will
  43. * continue to search for next wildcard pattern and it will try to add matching
  44. * commands.
  45. *
  46. *
  47. * This function is internal to shell module and shall be not called directly.
  48. *
  49. * @param[in/out] shell Pointer to the CLI instance.
  50. * @param[in] cmd Pointer to command which will be processed
  51. * @param[in] pattern Pointer to wildcard pattern.
  52. *
  53. * @retval WILDCARD_CMD_ADDED All matching commands added to the buffer.
  54. * @retval WILDCARD_CMD_ADDED_MISSING_SPACE Not all matching commands added
  55. * because CONFIG_SHELL_CMD_BUFF_SIZE
  56. * is too small.
  57. * @retval WILDCARD_CMD_NO_MATCH_FOUND No matching command found.
  58. */
  59. static enum shell_wildcard_status commands_expand(const struct shell *shell,
  60. const struct shell_static_entry *cmd,
  61. const char *pattern)
  62. {
  63. enum shell_wildcard_status ret_val = SHELL_WILDCARD_CMD_NO_MATCH_FOUND;
  64. struct shell_static_entry const *entry = NULL;
  65. struct shell_static_entry dloc;
  66. size_t cmd_idx = 0;
  67. size_t cnt = 0;
  68. while ((entry = z_shell_cmd_get(cmd, cmd_idx++, &dloc)) != NULL) {
  69. if (fnmatch(pattern, entry->syntax, 0) == 0) {
  70. ret_val = command_add(shell->ctx->temp_buff,
  71. &shell->ctx->cmd_tmp_buff_len,
  72. entry->syntax, pattern);
  73. if (ret_val == SHELL_WILDCARD_CMD_MISSING_SPACE) {
  74. z_shell_fprintf(shell, SHELL_WARNING,
  75. "Command buffer is too short to"
  76. " expand all commands matching"
  77. " wildcard pattern: %s\n", pattern);
  78. break;
  79. } else if (ret_val != SHELL_WILDCARD_CMD_ADDED) {
  80. break;
  81. }
  82. cnt++;
  83. }
  84. }
  85. if (cnt > 0) {
  86. z_shell_pattern_remove(shell->ctx->temp_buff,
  87. &shell->ctx->cmd_tmp_buff_len, pattern);
  88. }
  89. return ret_val;
  90. }
  91. bool z_shell_has_wildcard(const char *str)
  92. {
  93. uint16_t str_len = z_shell_strlen(str);
  94. for (size_t i = 0; i < str_len; i++) {
  95. if ((str[i] == '?') || (str[i] == '*')) {
  96. return true;
  97. }
  98. }
  99. return false;
  100. }
  101. void z_shell_wildcard_prepare(const struct shell *shell)
  102. {
  103. /* Wildcard can be correctly handled under following conditions:
  104. * - wildcard command does not have a handler
  105. * - wildcard command is on the deepest commands level
  106. * - other commands on the same level as wildcard command shall also not
  107. * have a handler
  108. *
  109. * Algorithm:
  110. * 1. Command buffer: ctx->cmd_buff is copied to temporary buffer:
  111. * ctx->temp_buff.
  112. * 2. Algorithm goes through command buffer to find handlers and
  113. * subcommands.
  114. * 3. If algorithm will find a wildcard character it switches to
  115. * Temporary buffer.
  116. * 4. In the Temporary buffer command containing wildcard character is
  117. * replaced by matching command(s).
  118. * 5. Algorithm switch back to Command buffer and analyzes next command.
  119. * 6. When all arguments are analyzed from Command buffer, Temporary
  120. * buffer with all expanded commands is copied to Command buffer.
  121. * 7. Deepest found handler is executed and all lower level commands,
  122. * including expanded commands, are passed as arguments.
  123. */
  124. memset(shell->ctx->temp_buff, 0, sizeof(shell->ctx->temp_buff));
  125. memcpy(shell->ctx->temp_buff,
  126. shell->ctx->cmd_buff,
  127. shell->ctx->cmd_buff_len);
  128. /* Function shell_spaces_trim must be used instead of shell_make_argv.
  129. * At this point it is important to keep temp_buff as one string.
  130. * It will allow to find wildcard commands easily with strstr function.
  131. */
  132. z_shell_spaces_trim(shell->ctx->temp_buff);
  133. /* +1 for EOS*/
  134. shell->ctx->cmd_tmp_buff_len = z_shell_strlen(shell->ctx->temp_buff) + 1;
  135. }
  136. enum shell_wildcard_status z_shell_wildcard_process(const struct shell *shell,
  137. const struct shell_static_entry *cmd,
  138. const char *pattern)
  139. {
  140. enum shell_wildcard_status ret_val = SHELL_WILDCARD_NOT_FOUND;
  141. if (cmd == NULL) {
  142. return ret_val;
  143. }
  144. if (!z_shell_has_wildcard(pattern)) {
  145. return ret_val;
  146. }
  147. /* Function will search commands tree for commands matching wildcard
  148. * pattern stored in argv[cmd_lvl]. When match is found wildcard pattern
  149. * will be replaced by matching commands. If there is no space in the
  150. * buffer to add all matching commands function will add as many as
  151. * possible. Next it will continue to search for next wildcard pattern
  152. * and it will try to add matching commands.
  153. */
  154. ret_val = commands_expand(shell, cmd, pattern);
  155. return ret_val;
  156. }
  157. void z_shell_wildcard_finalize(const struct shell *shell)
  158. {
  159. memcpy(shell->ctx->cmd_buff,
  160. shell->ctx->temp_buff,
  161. shell->ctx->cmd_tmp_buff_len);
  162. shell->ctx->cmd_buff_len = shell->ctx->cmd_tmp_buff_len;
  163. }