shell_history.c 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <shell/shell_history.h>
  7. #include <string.h>
  8. /*
  9. * History must store strings (commands) and allow traversing them and adding
  10. * new string. When new item is added then first it is compared if it is not
  11. * the same as the last one (then it is not stored). If there is no room in the
  12. * buffer to store the new item, oldest one is removed until there is a room.
  13. *
  14. * Items are allocated and stored in the ring buffer. Items then a linked in
  15. * the list.
  16. *
  17. * Because stored strings must be copied and compared, it is more convenient to
  18. * store them in the ring buffer in a way that they are not split into two
  19. * chunks (when ring buffer wraps). To ensure that item is in a single chunk,
  20. * item includes padding. If continues area for new item cannot be allocated
  21. * then allocated space is increased by the padding.
  22. *
  23. * If item does not fit at the end of the ring buffer padding is added: *
  24. * +-----------+----------------+-----------------------------------+---------+
  25. * | header | "history item" | | padding |
  26. * | padding | | | |
  27. * +-----------+----------------+-----------------------------------+---------+
  28. *
  29. * If item fits in the ring buffer available space then there is no padding:
  30. * +-----------------+------------+----------------+--------------------------+
  31. * | | header | "history item" | |
  32. * | | no padding | | |
  33. * +-----------------+------------+----------------+--------------------------+
  34. */
  35. struct shell_history_item {
  36. sys_dnode_t dnode;
  37. uint16_t len;
  38. uint16_t padding;
  39. char data[0];
  40. };
  41. void z_shell_history_mode_exit(struct shell_history *history)
  42. {
  43. history->current = NULL;
  44. }
  45. bool z_shell_history_get(struct shell_history *history, bool up,
  46. uint8_t *dst, uint16_t *len)
  47. {
  48. struct shell_history_item *h_item; /* history item */
  49. sys_dnode_t *l_item; /* list item */
  50. if (sys_dlist_is_empty(&history->list)) {
  51. *len = 0U;
  52. return false;
  53. }
  54. if (!up) { /* button down */
  55. if (history->current == NULL) {
  56. /* Not in history mode. It is started by up button. */
  57. *len = 0U;
  58. return false;
  59. }
  60. l_item = sys_dlist_peek_prev_no_check(&history->list,
  61. history->current);
  62. } else { /* button up */
  63. l_item = (history->current == NULL) ?
  64. sys_dlist_peek_head_not_empty(&history->list) :
  65. sys_dlist_peek_next_no_check(&history->list, history->current);
  66. }
  67. history->current = l_item;
  68. h_item = CONTAINER_OF(l_item, struct shell_history_item, dnode);
  69. if (l_item) {
  70. memcpy(dst, h_item->data, h_item->len);
  71. *len = h_item->len;
  72. dst[*len] = '\0';
  73. return true;
  74. }
  75. *len = 0U;
  76. return false;
  77. }
  78. static void add_to_head(struct shell_history *history,
  79. struct shell_history_item *item,
  80. uint8_t *src, size_t len, uint16_t padding)
  81. {
  82. item->len = len;
  83. item->padding = padding;
  84. memcpy(item->data, src, len);
  85. sys_dlist_prepend(&history->list, &item->dnode);
  86. }
  87. /* Returns true if element was removed. */
  88. static bool remove_from_tail(struct shell_history *history)
  89. {
  90. sys_dnode_t *l_item; /* list item */
  91. struct shell_history_item *h_item;
  92. uint32_t total_len;
  93. if (sys_dlist_is_empty(&history->list)) {
  94. return false;
  95. }
  96. l_item = sys_dlist_peek_tail(&history->list);
  97. sys_dlist_remove(l_item);
  98. h_item = CONTAINER_OF(l_item, struct shell_history_item, dnode);
  99. total_len = offsetof(struct shell_history_item, data) +
  100. h_item->len + h_item->padding;
  101. ring_buf_get_finish(history->ring_buf, total_len);
  102. return true;
  103. }
  104. void z_shell_history_purge(struct shell_history *history)
  105. {
  106. while (remove_from_tail(history)) {
  107. }
  108. }
  109. void z_shell_history_put(struct shell_history *history, uint8_t *line,
  110. size_t len)
  111. {
  112. sys_dnode_t *l_item; /* list item */
  113. struct shell_history_item *h_item;
  114. uint32_t total_len = len + offsetof(struct shell_history_item, data);
  115. uint32_t claim_len;
  116. uint32_t claim2_len;
  117. uint16_t padding = (~total_len + 1) & (sizeof(void *) - 1);
  118. /* align to word. */
  119. total_len += padding;
  120. if (total_len > ring_buf_capacity_get(history->ring_buf)) {
  121. return;
  122. }
  123. z_shell_history_mode_exit(history);
  124. if (len == 0) {
  125. return;
  126. }
  127. l_item = sys_dlist_peek_head(&history->list);
  128. h_item = CONTAINER_OF(l_item, struct shell_history_item, dnode);
  129. if (l_item &&
  130. (h_item->len == len) &&
  131. (memcmp(h_item->data, line, len) == 0)) {
  132. /* Same command as before, do not store */
  133. return;
  134. }
  135. do {
  136. claim_len = ring_buf_put_claim(history->ring_buf,
  137. (uint8_t **)&h_item, total_len);
  138. /* second allocation may succeed if we were at the end of the
  139. * buffer.
  140. */
  141. if (claim_len < total_len) {
  142. claim2_len =
  143. ring_buf_put_claim(history->ring_buf,
  144. (uint8_t **)&h_item, total_len);
  145. if (claim2_len == total_len) {
  146. ring_buf_put_finish(history->ring_buf,
  147. claim_len);
  148. padding += claim_len;
  149. claim_len = total_len;
  150. }
  151. }
  152. if (claim_len == total_len) {
  153. add_to_head(history, h_item, line, len, padding);
  154. ring_buf_put_finish(history->ring_buf, claim_len);
  155. break;
  156. }
  157. ring_buf_put_finish(history->ring_buf, 0);
  158. if (remove_from_tail(history) == false) {
  159. __ASSERT_NO_MSG(ring_buf_is_empty(history->ring_buf));
  160. /* if history is empty reset ring buffer. Even when
  161. * ring buffer is empty, it is possible that available
  162. * continues memory in worst case equals half of the
  163. * ring buffer capacity. By reseting ring buffer we
  164. * ensure that it is capable to provide continues memory
  165. * of ring buffer capacity length.
  166. */
  167. ring_buf_reset(history->ring_buf);
  168. }
  169. } while (1);
  170. }
  171. void z_shell_history_init(struct shell_history *history)
  172. {
  173. sys_dlist_init(&history->list);
  174. history->current = NULL;
  175. }