test_index.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. #include <stdio.h>
  2. #include <furi.h>
  3. #include <furi_hal.h>
  4. #include "minunit_vars.h"
  5. #include <notification/notification_messages.h>
  6. #include <cli/cli.h>
  7. #include <loader/loader.h>
  8. #define TAG "UnitTests"
  9. int run_minunit_test_furi();
  10. int run_minunit_test_furi_hal();
  11. int run_minunit_test_furi_string();
  12. int run_minunit_test_infrared();
  13. int run_minunit_test_rpc();
  14. int run_minunit_test_manifest();
  15. int run_minunit_test_flipper_format();
  16. int run_minunit_test_flipper_format_string();
  17. int run_minunit_test_stream();
  18. int run_minunit_test_storage();
  19. int run_minunit_test_subghz();
  20. int run_minunit_test_dirwalk();
  21. int run_minunit_test_power();
  22. int run_minunit_test_protocol_dict();
  23. int run_minunit_test_lfrfid_protocols();
  24. int run_minunit_test_nfc();
  25. int run_minunit_test_bit_lib();
  26. int run_minunit_test_float_tools();
  27. int run_minunit_test_bt();
  28. typedef int (*UnitTestEntry)();
  29. typedef struct {
  30. const char* name;
  31. const UnitTestEntry entry;
  32. } UnitTest;
  33. const UnitTest unit_tests[] = {
  34. {.name = "furi", .entry = run_minunit_test_furi},
  35. {.name = "furi_hal", .entry = run_minunit_test_furi_hal},
  36. {.name = "furi_string", .entry = run_minunit_test_furi_string},
  37. {.name = "storage", .entry = run_minunit_test_storage},
  38. {.name = "stream", .entry = run_minunit_test_stream},
  39. {.name = "dirwalk", .entry = run_minunit_test_dirwalk},
  40. {.name = "manifest", .entry = run_minunit_test_manifest},
  41. {.name = "flipper_format", .entry = run_minunit_test_flipper_format},
  42. {.name = "flipper_format_string", .entry = run_minunit_test_flipper_format_string},
  43. {.name = "rpc", .entry = run_minunit_test_rpc},
  44. {.name = "subghz", .entry = run_minunit_test_subghz},
  45. {.name = "infrared", .entry = run_minunit_test_infrared},
  46. {.name = "nfc", .entry = run_minunit_test_nfc},
  47. {.name = "power", .entry = run_minunit_test_power},
  48. {.name = "protocol_dict", .entry = run_minunit_test_protocol_dict},
  49. {.name = "lfrfid", .entry = run_minunit_test_lfrfid_protocols},
  50. {.name = "bit_lib", .entry = run_minunit_test_bit_lib},
  51. {.name = "float_tools", .entry = run_minunit_test_float_tools},
  52. {.name = "bt", .entry = run_minunit_test_bt},
  53. };
  54. void minunit_print_progress() {
  55. static const char progress[] = {'\\', '|', '/', '-'};
  56. static uint8_t progress_counter = 0;
  57. static TickType_t last_tick = 0;
  58. TickType_t current_tick = xTaskGetTickCount();
  59. if(current_tick - last_tick > 20) {
  60. last_tick = current_tick;
  61. printf("[%c]\033[3D", progress[++progress_counter % COUNT_OF(progress)]);
  62. fflush(stdout);
  63. }
  64. }
  65. void minunit_print_fail(const char* str) {
  66. printf(FURI_LOG_CLR_E "%s\r\n" FURI_LOG_CLR_RESET, str);
  67. }
  68. void unit_tests_cli(Cli* cli, FuriString* args, void* context) {
  69. UNUSED(cli);
  70. UNUSED(args);
  71. UNUSED(context);
  72. minunit_run = 0;
  73. minunit_assert = 0;
  74. minunit_fail = 0;
  75. minunit_status = 0;
  76. Loader* loader = furi_record_open(RECORD_LOADER);
  77. NotificationApp* notification = furi_record_open(RECORD_NOTIFICATION);
  78. // TODO: lock device while test running
  79. if(loader_is_locked(loader)) {
  80. printf("RPC: stop all applications to run tests\r\n");
  81. notification_message(notification, &sequence_blink_magenta_100);
  82. } else {
  83. notification_message_block(notification, &sequence_set_only_blue_255);
  84. uint32_t heap_before = memmgr_get_free_heap();
  85. uint32_t cycle_counter = furi_get_tick();
  86. for(size_t i = 0; i < COUNT_OF(unit_tests); i++) {
  87. if(cli_cmd_interrupt_received(cli)) {
  88. break;
  89. }
  90. if(furi_string_size(args)) {
  91. if(furi_string_cmp_str(args, unit_tests[i].name) == 0) {
  92. unit_tests[i].entry();
  93. } else {
  94. printf("Skipping %s\r\n", unit_tests[i].name);
  95. }
  96. } else {
  97. unit_tests[i].entry();
  98. }
  99. }
  100. if(minunit_run != 0) {
  101. printf("\r\nFailed tests: %u\r\n", minunit_fail);
  102. // Time report
  103. cycle_counter = (furi_get_tick() - cycle_counter);
  104. printf("Consumed: %lu ms\r\n", cycle_counter);
  105. // Wait for tested services and apps to deallocate memory
  106. furi_delay_ms(200);
  107. uint32_t heap_after = memmgr_get_free_heap();
  108. printf("Leaked: %ld\r\n", heap_before - heap_after);
  109. // Final Report
  110. if(minunit_fail == 0) {
  111. notification_message(notification, &sequence_success);
  112. printf("Status: PASSED\r\n");
  113. } else {
  114. notification_message(notification, &sequence_error);
  115. printf("Status: FAILED\r\n");
  116. }
  117. }
  118. }
  119. furi_record_close(RECORD_NOTIFICATION);
  120. furi_record_close(RECORD_LOADER);
  121. }
  122. void unit_tests_on_system_start() {
  123. #ifdef SRV_CLI
  124. Cli* cli = furi_record_open(RECORD_CLI);
  125. // We need to launch apps from tests, so we cannot lock loader
  126. cli_add_command(cli, "unit_tests", CliCommandFlagParallelSafe, unit_tests_cli, NULL);
  127. furi_record_close(RECORD_CLI);
  128. #endif
  129. }