test_mem.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. #include "test_mem.h"
  2. #include "lwip/mem.h"
  3. #include "lwip/stats.h"
  4. #if !LWIP_STATS || !MEM_STATS
  5. #error "This tests needs MEM-statistics enabled"
  6. #endif
  7. /* Setups/teardown functions */
  8. static void
  9. mem_setup(void)
  10. {
  11. lwip_check_ensure_no_alloc(SKIP_POOL(MEMP_SYS_TIMEOUT));
  12. }
  13. static void
  14. mem_teardown(void)
  15. {
  16. lwip_check_ensure_no_alloc(SKIP_POOL(MEMP_SYS_TIMEOUT));
  17. }
  18. /* Test functions */
  19. /** Call mem_malloc, mem_free and mem_trim and check stats */
  20. START_TEST(test_mem_one)
  21. {
  22. #define SIZE1 16
  23. #define SIZE1_2 12
  24. #define SIZE2 16
  25. void *p1, *p2;
  26. mem_size_t s1, s2;
  27. LWIP_UNUSED_ARG(_i);
  28. fail_unless(lwip_stats.mem.used == 0);
  29. p1 = mem_malloc(SIZE1);
  30. fail_unless(p1 != NULL);
  31. fail_unless(lwip_stats.mem.used >= SIZE1);
  32. s1 = lwip_stats.mem.used;
  33. p2 = mem_malloc(SIZE2);
  34. fail_unless(p2 != NULL);
  35. fail_unless(lwip_stats.mem.used >= SIZE2 + s1);
  36. s2 = lwip_stats.mem.used;
  37. mem_trim(p1, SIZE1_2);
  38. mem_free(p2);
  39. fail_unless(lwip_stats.mem.used <= s2 - SIZE2);
  40. mem_free(p1);
  41. fail_unless(lwip_stats.mem.used == 0);
  42. }
  43. END_TEST
  44. static void malloc_keep_x(int x, int num, int size, int freestep)
  45. {
  46. int i;
  47. void* p[16];
  48. LWIP_ASSERT("invalid size", size >= 0 && size < (mem_size_t)-1);
  49. memset(p, 0, sizeof(p));
  50. for(i = 0; i < num && i < 16; i++) {
  51. p[i] = mem_malloc((mem_size_t)size);
  52. fail_unless(p[i] != NULL);
  53. }
  54. for(i = 0; i < num && i < 16; i += freestep) {
  55. if (i == x) {
  56. continue;
  57. }
  58. mem_free(p[i]);
  59. p[i] = NULL;
  60. }
  61. for(i = 0; i < num && i < 16; i++) {
  62. if (i == x) {
  63. continue;
  64. }
  65. if (p[i] != NULL) {
  66. mem_free(p[i]);
  67. p[i] = NULL;
  68. }
  69. }
  70. fail_unless(p[x] != NULL);
  71. mem_free(p[x]);
  72. }
  73. START_TEST(test_mem_random)
  74. {
  75. const int num = 16;
  76. int x;
  77. int size;
  78. int freestep;
  79. LWIP_UNUSED_ARG(_i);
  80. fail_unless(lwip_stats.mem.used == 0);
  81. for (x = 0; x < num; x++) {
  82. for (size = 1; size < 32; size++) {
  83. for (freestep = 1; freestep <= 3; freestep++) {
  84. fail_unless(lwip_stats.mem.used == 0);
  85. malloc_keep_x(x, num, size, freestep);
  86. fail_unless(lwip_stats.mem.used == 0);
  87. }
  88. }
  89. }
  90. }
  91. END_TEST
  92. START_TEST(test_mem_invalid_free)
  93. {
  94. u8_t *ptr, *ptr_low, *ptr_high;
  95. LWIP_UNUSED_ARG(_i);
  96. fail_unless(lwip_stats.mem.used == 0);
  97. fail_unless(lwip_stats.mem.illegal == 0);
  98. ptr = (u8_t *)mem_malloc(1);
  99. fail_unless(ptr != NULL);
  100. fail_unless(lwip_stats.mem.used != 0);
  101. ptr_low = ptr - 0x10;
  102. mem_free(ptr_low);
  103. fail_unless(lwip_stats.mem.illegal == 1);
  104. lwip_stats.mem.illegal = 0;
  105. ptr_high = ptr + (MEM_SIZE * 2);
  106. mem_free(ptr_high);
  107. fail_unless(lwip_stats.mem.illegal == 1);
  108. lwip_stats.mem.illegal = 0;
  109. mem_free(ptr);
  110. fail_unless(lwip_stats.mem.illegal == 0);
  111. fail_unless(lwip_stats.mem.used == 0);
  112. }
  113. END_TEST
  114. START_TEST(test_mem_double_free)
  115. {
  116. u8_t *ptr1b, *ptr1, *ptr2, *ptr3;
  117. LWIP_UNUSED_ARG(_i);
  118. fail_unless(lwip_stats.mem.used == 0);
  119. fail_unless(lwip_stats.mem.illegal == 0);
  120. ptr1 = (u8_t *)mem_malloc(1);
  121. fail_unless(ptr1 != NULL);
  122. fail_unless(lwip_stats.mem.used != 0);
  123. ptr2 = (u8_t *)mem_malloc(1);
  124. fail_unless(ptr2 != NULL);
  125. fail_unless(lwip_stats.mem.used != 0);
  126. ptr3 = (u8_t *)mem_malloc(1);
  127. fail_unless(ptr3 != NULL);
  128. fail_unless(lwip_stats.mem.used != 0);
  129. /* free the middle mem */
  130. mem_free(ptr2);
  131. fail_unless(lwip_stats.mem.illegal == 0);
  132. /* double-free of middle mem: should fail */
  133. mem_free(ptr2);
  134. fail_unless(lwip_stats.mem.illegal == 1);
  135. lwip_stats.mem.illegal = 0;
  136. /* free upper memory and try again */
  137. mem_free(ptr3);
  138. fail_unless(lwip_stats.mem.illegal == 0);
  139. mem_free(ptr2);
  140. fail_unless(lwip_stats.mem.illegal == 1);
  141. lwip_stats.mem.illegal = 0;
  142. /* free lower memory and try again */
  143. mem_free(ptr1);
  144. fail_unless(lwip_stats.mem.illegal == 0);
  145. fail_unless(lwip_stats.mem.used == 0);
  146. mem_free(ptr2);
  147. fail_unless(lwip_stats.mem.illegal == 1);
  148. fail_unless(lwip_stats.mem.used == 0);
  149. lwip_stats.mem.illegal = 0;
  150. /* reallocate lowest memory, now overlapping already freed ptr2 */
  151. #ifndef MIN_SIZE
  152. #define MIN_SIZE 12
  153. #endif
  154. ptr1b = (u8_t *)mem_malloc(MIN_SIZE * 2);
  155. fail_unless(ptr1b != NULL);
  156. fail_unless(lwip_stats.mem.used != 0);
  157. mem_free(ptr2);
  158. fail_unless(lwip_stats.mem.illegal == 1);
  159. lwip_stats.mem.illegal = 0;
  160. memset(ptr1b, 1, MIN_SIZE * 2);
  161. mem_free(ptr2);
  162. fail_unless(lwip_stats.mem.illegal == 1);
  163. lwip_stats.mem.illegal = 0;
  164. mem_free(ptr1b);
  165. fail_unless(lwip_stats.mem.illegal == 0);
  166. fail_unless(lwip_stats.mem.used == 0);
  167. }
  168. END_TEST
  169. /** Create the suite including all tests for this module */
  170. Suite *
  171. mem_suite(void)
  172. {
  173. testfunc tests[] = {
  174. TESTFUNC(test_mem_one),
  175. TESTFUNC(test_mem_random),
  176. TESTFUNC(test_mem_invalid_free),
  177. TESTFUNC(test_mem_double_free)
  178. };
  179. return create_suite("MEM", tests, sizeof(tests)/sizeof(testfunc), mem_setup, mem_teardown);
  180. }