test_lwip_apps.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. /*
  2. * SPDX-FileCopyrightText: 2022 Espressif Systems (Shanghai) CO LTD
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include "freertos/FreeRTOS.h"
  7. #include "freertos/event_groups.h"
  8. #include "test_utils.h"
  9. #include "unity.h"
  10. #include "lwip/inet.h"
  11. #include "lwip/netdb.h"
  12. #include "lwip/sockets.h"
  13. #include "ping/ping_sock.h"
  14. #include "dhcpserver/dhcpserver.h"
  15. #define ETH_PING_END_BIT BIT(1)
  16. #define ETH_PING_DURATION_MS (5000)
  17. #define ETH_PING_END_TIMEOUT_MS (ETH_PING_DURATION_MS * 2)
  18. #define TEST_ICMP_DESTINATION_DOMAIN_NAME "127.0.0.1"
  19. #if !TEMPORARY_DISABLED_FOR_TARGETS(ESP32C2)
  20. //IDF-5047
  21. static void test_on_ping_success(esp_ping_handle_t hdl, void *args)
  22. {
  23. uint8_t ttl;
  24. uint16_t seqno;
  25. uint32_t elapsed_time, recv_len;
  26. ip_addr_t target_addr;
  27. esp_ping_get_profile(hdl, ESP_PING_PROF_SEQNO, &seqno, sizeof(seqno));
  28. esp_ping_get_profile(hdl, ESP_PING_PROF_TTL, &ttl, sizeof(ttl));
  29. esp_ping_get_profile(hdl, ESP_PING_PROF_IPADDR, &target_addr, sizeof(target_addr));
  30. esp_ping_get_profile(hdl, ESP_PING_PROF_SIZE, &recv_len, sizeof(recv_len));
  31. esp_ping_get_profile(hdl, ESP_PING_PROF_TIMEGAP, &elapsed_time, sizeof(elapsed_time));
  32. printf("%d bytes from %s icmp_seq=%d ttl=%d time=%d ms\n",
  33. recv_len, inet_ntoa(target_addr.u_addr.ip4), seqno, ttl, elapsed_time);
  34. }
  35. static void test_on_ping_timeout(esp_ping_handle_t hdl, void *args)
  36. {
  37. uint16_t seqno;
  38. ip_addr_t target_addr;
  39. esp_ping_get_profile(hdl, ESP_PING_PROF_SEQNO, &seqno, sizeof(seqno));
  40. esp_ping_get_profile(hdl, ESP_PING_PROF_IPADDR, &target_addr, sizeof(target_addr));
  41. printf("From %s icmp_seq=%d timeout\n", inet_ntoa(target_addr.u_addr.ip4), seqno);
  42. }
  43. static void test_on_ping_end(esp_ping_handle_t hdl, void *args)
  44. {
  45. EventGroupHandle_t eth_event_group = (EventGroupHandle_t)args;
  46. uint32_t transmitted;
  47. uint32_t received;
  48. uint32_t total_time_ms;
  49. esp_ping_get_profile(hdl, ESP_PING_PROF_REQUEST, &transmitted, sizeof(transmitted));
  50. esp_ping_get_profile(hdl, ESP_PING_PROF_REPLY, &received, sizeof(received));
  51. esp_ping_get_profile(hdl, ESP_PING_PROF_DURATION, &total_time_ms, sizeof(total_time_ms));
  52. printf("%d packets transmitted, %d received, time %dms\n", transmitted, received, total_time_ms);
  53. if (transmitted == received) {
  54. xEventGroupSetBits(eth_event_group, ETH_PING_END_BIT);
  55. }
  56. }
  57. TEST_CASE("localhost ping test", "[lwip]")
  58. {
  59. EventBits_t bits;
  60. EventGroupHandle_t eth_event_group = xEventGroupCreate();
  61. TEST_ASSERT(eth_event_group != NULL);
  62. test_case_uses_tcpip();
  63. // Parse IP address: Destination is a localhost address, so we don't need any interface (esp-netif/driver)
  64. ip_addr_t target_addr;
  65. struct addrinfo hint;
  66. struct addrinfo *res = NULL;
  67. memset(&hint, 0, sizeof(hint));
  68. memset(&target_addr, 0, sizeof(target_addr));
  69. /* convert URL to IP */
  70. TEST_ASSERT(getaddrinfo(TEST_ICMP_DESTINATION_DOMAIN_NAME, NULL, &hint, &res) == 0);
  71. struct in_addr addr4 = ((struct sockaddr_in *)(res->ai_addr))->sin_addr;
  72. inet_addr_to_ip4addr(ip_2_ip4(&target_addr), &addr4);
  73. freeaddrinfo(res);
  74. esp_ping_config_t ping_config = ESP_PING_DEFAULT_CONFIG();
  75. ping_config.timeout_ms = 2000;
  76. ping_config.target_addr = target_addr;
  77. ping_config.count = 0; // ping in infinite mode
  78. /* set callback functions */
  79. esp_ping_callbacks_t cbs;
  80. cbs.on_ping_success = test_on_ping_success;
  81. cbs.on_ping_timeout = test_on_ping_timeout;
  82. cbs.on_ping_end = test_on_ping_end;
  83. cbs.cb_args = eth_event_group;
  84. esp_ping_handle_t ping;
  85. TEST_ESP_OK(esp_ping_new_session(&ping_config, &cbs, &ping));
  86. /* start ping */
  87. TEST_ESP_OK(esp_ping_start(ping));
  88. /* ping for a while */
  89. vTaskDelay(pdMS_TO_TICKS(ETH_PING_DURATION_MS));
  90. /* stop ping */
  91. TEST_ESP_OK(esp_ping_stop(ping));
  92. /* wait for end of ping */
  93. bits = xEventGroupWaitBits(eth_event_group, ETH_PING_END_BIT, true, true, pdMS_TO_TICKS(ETH_PING_END_TIMEOUT_MS));
  94. TEST_ASSERT((bits & ETH_PING_END_BIT) == ETH_PING_END_BIT);
  95. /* restart ping */
  96. TEST_ESP_OK(esp_ping_start(ping));
  97. vTaskDelay(pdMS_TO_TICKS(ETH_PING_DURATION_MS));
  98. TEST_ESP_OK(esp_ping_stop(ping));
  99. bits = xEventGroupWaitBits(eth_event_group, ETH_PING_END_BIT, true, true, pdMS_TO_TICKS(ETH_PING_END_TIMEOUT_MS));
  100. TEST_ASSERT((bits & ETH_PING_END_BIT) == ETH_PING_END_BIT);
  101. /* de-initialize ping process */
  102. TEST_ESP_OK(esp_ping_delete_session(ping));
  103. vEventGroupDelete(eth_event_group);
  104. }
  105. #endif //!TEMPORARY_DISABLED_FOR_TARGETS(ESP32C2)
  106. TEST_CASE("dhcp server init/deinit", "[lwip][leaks=0]")
  107. {
  108. dhcps_t *dhcps = dhcps_new();
  109. TEST_ASSERT_NOT_NULL(dhcps);
  110. ip4_addr_t ip = { .addr = IPADDR_ANY };
  111. TEST_ASSERT(dhcps_start(dhcps, NULL, ip) == ERR_ARG);
  112. TEST_ASSERT(dhcps_stop(dhcps, NULL) == ERR_ARG);
  113. dhcps_delete(dhcps);
  114. }
  115. TEST_CASE("dhcp server start/stop on localhost", "[lwip]")
  116. {
  117. test_case_uses_tcpip();
  118. dhcps_t *dhcps = dhcps_new();
  119. struct netif *netif;
  120. NETIF_FOREACH(netif) {
  121. if (netif->name[0] == 'l' && netif->name[1] == 'o') {
  122. break;
  123. }
  124. }
  125. TEST_ASSERT_NOT_NULL(netif);
  126. ip4_addr_t ip = { .addr = 0x7f0001 };
  127. TEST_ASSERT(dhcps_start(dhcps, netif, ip) == ERR_OK);
  128. TEST_ASSERT(dhcps_stop(dhcps, netif) == ERR_OK);
  129. dhcps_delete(dhcps);
  130. }