cb_hsm_test.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. /*
  2. * SPDX-License-Identifier: Apache-2.0
  3. *
  4. * Change Logs:
  5. * Date Author Notes
  6. * 2023-06-03 tyx first implementation
  7. */
  8. #include <stdio.h>
  9. #include <string.h>
  10. #include <gtest/gtest.h>
  11. #include "cb_hsm.h"
  12. typedef struct sm_test_actor
  13. {
  14. cb_hsm_actor_t super;
  15. int foo;
  16. char buff[256];
  17. char *ptr;
  18. char* end;
  19. }sm_actor;
  20. enum QHsmTstSignals {
  21. A_SIG = CB_HSM_TOPIC_USER,
  22. B_SIG,
  23. C_SIG,
  24. D_SIG,
  25. E_SIG,
  26. F_SIG,
  27. G_SIG,
  28. H_SIG,
  29. I_SIG,
  30. TERMINATE_SIG,
  31. MAX_SIG
  32. };
  33. static sm_actor test_actor;
  34. cb_hsm_ret_t QHsmTst_initial(sm_actor *me, const cb_hsm_event_t * const e);
  35. cb_hsm_ret_t QHsmTst_s2(sm_actor *me, const cb_hsm_event_t * const e);
  36. cb_hsm_ret_t QHsmTst_s11(sm_actor *me, const cb_hsm_event_t * const e);
  37. cb_hsm_ret_t QHsmTst_s211(sm_actor *me, const cb_hsm_event_t * const e);
  38. const char* QHsmTst_ctor(void) {
  39. sm_actor *me = &test_actor;
  40. cb_hsm_init(&me->super);
  41. CB_HSM_START(&me->super, &QHsmTst_initial);
  42. return test_actor.buff;
  43. }
  44. cb_hsm_ret_t QHsmTst_initial(sm_actor *me, const cb_hsm_event_t * const e)
  45. {
  46. (void)e; /* suppress "e not used" compiler warning */
  47. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "top-INIT;");
  48. me->foo = 0; /* initialize extended state variable */
  49. return CB_HSM_TRAN(me, &QHsmTst_s2);
  50. }
  51. cb_hsm_ret_t QHsmTst_s(sm_actor *me, const cb_hsm_event_t * const e) {
  52. switch (e->topic) {
  53. case CB_HSM_TOPIC_ENTER: {
  54. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s-ENTRY;");
  55. return CB_HSM_HANDLED(me);
  56. }
  57. case CB_HSM_TOPIC_EXIT: {
  58. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s-EXIT;");
  59. return CB_HSM_HANDLED(me);
  60. }
  61. case CB_HSM_TOPIC_INIT: {
  62. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s-INIT;");
  63. return CB_HSM_TRAN(me, &QHsmTst_s11);
  64. }
  65. case E_SIG: {
  66. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s-E;");
  67. return CB_HSM_TRAN(me, &QHsmTst_s11);
  68. }
  69. case I_SIG: { /* internal transition with a guard */
  70. if (me->foo) {
  71. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s-I;");
  72. me->foo = 0;
  73. return CB_HSM_HANDLED(me);
  74. }
  75. break;
  76. }
  77. case TERMINATE_SIG: {
  78. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "exit\n\n;");
  79. return CB_HSM_HANDLED(me);
  80. }
  81. }
  82. return CB_HSM_SUPER(me, &cb_hsm_state_top);
  83. }
  84. cb_hsm_ret_t QHsmTst_s1(sm_actor *me, const cb_hsm_event_t * const e) {
  85. switch (e->topic) {
  86. case CB_HSM_TOPIC_ENTER: {
  87. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-ENTRY;");
  88. return CB_HSM_HANDLED(me);
  89. }
  90. case CB_HSM_TOPIC_EXIT: {
  91. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-EXIT;");
  92. return CB_HSM_HANDLED(me);
  93. }
  94. case CB_HSM_TOPIC_INIT: {
  95. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-INIT;");
  96. return CB_HSM_TRAN(me, &QHsmTst_s11);
  97. }
  98. case A_SIG: {
  99. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-A;");
  100. return CB_HSM_TRAN(me, &QHsmTst_s1);
  101. }
  102. case B_SIG: {
  103. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-B;");
  104. return CB_HSM_TRAN(me, &QHsmTst_s11);
  105. }
  106. case C_SIG: {
  107. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-C;");
  108. return CB_HSM_TRAN(me, &QHsmTst_s2);
  109. }
  110. case D_SIG: { /* transition with a gurad */
  111. if (!me->foo) {
  112. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-D;");
  113. me->foo = 1;
  114. return CB_HSM_TRAN(me, &QHsmTst_s);
  115. }
  116. break;
  117. }
  118. case F_SIG: {
  119. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-F;");
  120. return CB_HSM_TRAN(me, &QHsmTst_s211);
  121. }
  122. case I_SIG: { /* internal transition */
  123. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s1-I;");
  124. return CB_HSM_HANDLED(me);
  125. }
  126. }
  127. return CB_HSM_SUPER(me, &QHsmTst_s);
  128. }
  129. /*..........................................................................*/
  130. cb_hsm_ret_t QHsmTst_s11(sm_actor *me, const cb_hsm_event_t * const e) {
  131. switch (e->topic) {
  132. case CB_HSM_TOPIC_ENTER: {
  133. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s11-ENTRY;");
  134. return CB_HSM_HANDLED(me);
  135. }
  136. case CB_HSM_TOPIC_EXIT: {
  137. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s11-EXIT;");
  138. return CB_HSM_HANDLED(me);
  139. }
  140. case D_SIG: { /* transition with a gurad */
  141. if (me->foo) {
  142. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s11-D;");
  143. me->foo = 0;
  144. return CB_HSM_TRAN(me, &QHsmTst_s1);
  145. }
  146. break;
  147. }
  148. case G_SIG: {
  149. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s11-G;");
  150. return CB_HSM_TRAN(me, &QHsmTst_s211);
  151. }
  152. case H_SIG: {
  153. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s11-H;");
  154. return CB_HSM_TRAN(me, &QHsmTst_s);
  155. }
  156. }
  157. return CB_HSM_SUPER(me, &QHsmTst_s1);
  158. }
  159. /*..........................................................................*/
  160. cb_hsm_ret_t QHsmTst_s2(sm_actor *me, const cb_hsm_event_t * const e) {
  161. switch (e->topic) {
  162. case CB_HSM_TOPIC_ENTER: {
  163. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s2-ENTRY;");
  164. return CB_HSM_HANDLED(me);
  165. }
  166. case CB_HSM_TOPIC_EXIT: {
  167. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s2-EXIT;");
  168. return CB_HSM_HANDLED(me);
  169. }
  170. case CB_HSM_TOPIC_INIT: {
  171. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s2-INIT;");
  172. return CB_HSM_TRAN(me, &QHsmTst_s211);
  173. }
  174. case C_SIG: {
  175. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s2-C;");
  176. return CB_HSM_TRAN(me, &QHsmTst_s1);
  177. }
  178. case F_SIG: {
  179. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s2-F;");
  180. return CB_HSM_TRAN(me, &QHsmTst_s11);
  181. }
  182. case I_SIG: { /* internal transition with a guard */
  183. if (!me->foo) {
  184. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s2-I;");
  185. me->foo = 1;
  186. return CB_HSM_HANDLED(me);
  187. }
  188. break;
  189. }
  190. }
  191. return CB_HSM_SUPER(me, &QHsmTst_s);
  192. }
  193. /*..........................................................................*/
  194. cb_hsm_ret_t QHsmTst_s21(sm_actor *me, const cb_hsm_event_t * const e) {
  195. switch (e->topic) {
  196. case CB_HSM_TOPIC_ENTER: {
  197. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s21-ENTRY;");
  198. return CB_HSM_HANDLED(me);
  199. }
  200. case CB_HSM_TOPIC_EXIT: {
  201. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s21-EXIT;");
  202. return CB_HSM_HANDLED(me);
  203. }
  204. case CB_HSM_TOPIC_INIT: {
  205. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s21-INIT;");
  206. return CB_HSM_TRAN(me, &QHsmTst_s211);
  207. }
  208. case A_SIG: {
  209. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s21-A;");
  210. return CB_HSM_TRAN(me, &QHsmTst_s21);
  211. }
  212. case B_SIG: {
  213. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s21-B;");
  214. return CB_HSM_TRAN(me, &QHsmTst_s211);
  215. }
  216. case G_SIG: {
  217. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s21-G;");
  218. return CB_HSM_TRAN(me, &QHsmTst_s1);
  219. }
  220. }
  221. return CB_HSM_SUPER(me, &QHsmTst_s2);
  222. }
  223. /*..........................................................................*/
  224. cb_hsm_ret_t QHsmTst_s211(sm_actor *me, const cb_hsm_event_t * const e) {
  225. switch (e->topic) {
  226. case CB_HSM_TOPIC_ENTER: {
  227. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s211-ENTRY;");
  228. return CB_HSM_HANDLED(me);
  229. }
  230. case CB_HSM_TOPIC_EXIT: {
  231. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s211-EXIT;");
  232. return CB_HSM_HANDLED(me);
  233. }
  234. case D_SIG: {
  235. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s211-D;");
  236. return CB_HSM_TRAN(me, &QHsmTst_s21);
  237. }
  238. case H_SIG: {
  239. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "s211-H;");
  240. return CB_HSM_TRAN(me, &QHsmTst_s);
  241. }
  242. }
  243. return CB_HSM_SUPER(me, &QHsmTst_s21);
  244. }
  245. static void result_clean(void)
  246. {
  247. memset(test_actor.buff, 0, sizeof(test_actor.buff));
  248. test_actor.ptr = test_actor.buff;
  249. test_actor.end = &test_actor.buff[sizeof(test_actor.buff) / sizeof(test_actor.buff[0])];
  250. }
  251. static const char *dispatch(enum QHsmTstSignals sig) {
  252. cb_hsm_event_t e;
  253. result_clean();
  254. e.topic = sig & 0xFFFF;
  255. test_actor.ptr += snprintf(test_actor.ptr, test_actor.end - test_actor.ptr, "%c:", 'A' + sig - A_SIG);
  256. cb_hsm_event_handle(&test_actor.super, &e); /* dispatch the event */
  257. return test_actor.buff;
  258. }
  259. TEST(testCase, cb_hsm_test01)
  260. {
  261. result_clean();
  262. EXPECT_STREQ(QHsmTst_ctor(), "top-INIT;s-ENTRY;s2-ENTRY;s2-INIT;s21-ENTRY;s211-ENTRY;");
  263. EXPECT_STREQ(dispatch(G_SIG), "G:s21-G;s211-EXIT;s21-EXIT;s2-EXIT;s1-ENTRY;s1-INIT;s11-ENTRY;");
  264. EXPECT_STREQ(dispatch(I_SIG), "I:s1-I;");
  265. EXPECT_STREQ(dispatch(A_SIG), "A:s1-A;s11-EXIT;s1-EXIT;s1-ENTRY;s1-INIT;s11-ENTRY;");
  266. EXPECT_STREQ(dispatch(D_SIG), "D:s1-D;s11-EXIT;s1-EXIT;s-INIT;s1-ENTRY;s11-ENTRY;");
  267. EXPECT_STREQ(dispatch(D_SIG), "D:s11-D;s11-EXIT;s1-INIT;s11-ENTRY;");
  268. EXPECT_STREQ(dispatch(C_SIG), "C:s1-C;s11-EXIT;s1-EXIT;s2-ENTRY;s2-INIT;s21-ENTRY;s211-ENTRY;");
  269. EXPECT_STREQ(dispatch(E_SIG), "E:s-E;s211-EXIT;s21-EXIT;s2-EXIT;s1-ENTRY;s11-ENTRY;");
  270. EXPECT_STREQ(dispatch(E_SIG), "E:s-E;s11-EXIT;s1-EXIT;s1-ENTRY;s11-ENTRY;");
  271. EXPECT_STREQ(dispatch(G_SIG), "G:s11-G;s11-EXIT;s1-EXIT;s2-ENTRY;s21-ENTRY;s211-ENTRY;");
  272. EXPECT_STREQ(dispatch(I_SIG), "I:s2-I;");
  273. EXPECT_STREQ(dispatch(I_SIG), "I:s-I;");
  274. }