stress.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. /*
  2. * libusb stress test program to perform simple stress tests
  3. * Copyright © 2012 Toby Gray <toby.gray@realvnc.com>
  4. *
  5. * This library is free software; you can redistribute it and/or
  6. * modify it under the terms of the GNU Lesser General Public
  7. * License as published by the Free Software Foundation; either
  8. * version 2.1 of the License, or (at your option) any later version.
  9. *
  10. * This library is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  13. * Lesser General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU Lesser General Public
  16. * License along with this library; if not, write to the Free Software
  17. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  18. */
  19. #include <config.h>
  20. #include <string.h>
  21. #include "libusb.h"
  22. #include "libusb_testlib.h"
  23. /** Test that creates and destroys a single concurrent context
  24. * 10000 times. */
  25. static libusb_testlib_result test_init_and_exit(void)
  26. {
  27. for (int i = 0; i < 10000; ++i) {
  28. libusb_context *ctx = NULL;
  29. int r;
  30. r = libusb_init(&ctx);
  31. if (r != LIBUSB_SUCCESS) {
  32. libusb_testlib_logf(
  33. "Failed to init libusb on iteration %d: %d",
  34. i, r);
  35. return TEST_STATUS_FAILURE;
  36. }
  37. libusb_exit(ctx);
  38. }
  39. return TEST_STATUS_SUCCESS;
  40. }
  41. /** Tests that devices can be listed 1000 times. */
  42. static libusb_testlib_result test_get_device_list(void)
  43. {
  44. libusb_context *ctx;
  45. int r;
  46. r = libusb_init(&ctx);
  47. if (r != LIBUSB_SUCCESS) {
  48. libusb_testlib_logf("Failed to init libusb: %d", r);
  49. return TEST_STATUS_FAILURE;
  50. }
  51. for (int i = 0; i < 1000; ++i) {
  52. libusb_device **device_list = NULL;
  53. ssize_t list_size = libusb_get_device_list(ctx, &device_list);
  54. if (list_size < 0 || !device_list) {
  55. libusb_testlib_logf(
  56. "Failed to get device list on iteration %d: %ld (%p)",
  57. i, (long)-list_size, device_list);
  58. libusb_exit(ctx);
  59. return TEST_STATUS_FAILURE;
  60. }
  61. libusb_free_device_list(device_list, 1);
  62. }
  63. libusb_exit(ctx);
  64. return TEST_STATUS_SUCCESS;
  65. }
  66. /** Tests that 100 concurrent device lists can be open at a time. */
  67. static libusb_testlib_result test_many_device_lists(void)
  68. {
  69. #define LIST_COUNT 100
  70. libusb_testlib_result result = TEST_STATUS_SUCCESS;
  71. libusb_context *ctx = NULL;
  72. libusb_device **device_lists[LIST_COUNT];
  73. int r;
  74. r = libusb_init(&ctx);
  75. if (r != LIBUSB_SUCCESS) {
  76. libusb_testlib_logf("Failed to init libusb: %d", r);
  77. return TEST_STATUS_FAILURE;
  78. }
  79. memset(device_lists, 0, sizeof(device_lists));
  80. /* Create the 100 device lists. */
  81. for (int i = 0; i < LIST_COUNT; ++i) {
  82. ssize_t list_size = libusb_get_device_list(ctx, &device_lists[i]);
  83. if (list_size < 0 || !device_lists[i]) {
  84. libusb_testlib_logf(
  85. "Failed to get device list on iteration %d: %ld (%p)",
  86. i, (long)-list_size, device_lists[i]);
  87. result = TEST_STATUS_FAILURE;
  88. break;
  89. }
  90. }
  91. /* Destroy the 100 device lists. */
  92. for (int i = 0; i < LIST_COUNT; ++i) {
  93. if (device_lists[i])
  94. libusb_free_device_list(device_lists[i], 1);
  95. }
  96. libusb_exit(ctx);
  97. return result;
  98. #undef LIST_COUNT
  99. }
  100. /** Tests that the default context (used for various things including
  101. * logging) works correctly when the first context created in a
  102. * process is destroyed. */
  103. static libusb_testlib_result test_default_context_change(void)
  104. {
  105. for (int i = 0; i < 100; ++i) {
  106. libusb_context *ctx = NULL;
  107. int r;
  108. /* First create a new context */
  109. r = libusb_init(&ctx);
  110. if (r != LIBUSB_SUCCESS) {
  111. libusb_testlib_logf("Failed to init libusb: %d", r);
  112. return TEST_STATUS_FAILURE;
  113. }
  114. /* Enable debug output on new context, to be sure to use the context */
  115. libusb_set_option(ctx, LIBUSB_OPTION_LOG_LEVEL, LIBUSB_LOG_LEVEL_DEBUG);
  116. /* Enable debug output on the default context. This should work even before
  117. * the context has been created. */
  118. libusb_set_option(NULL, LIBUSB_OPTION_LOG_LEVEL, LIBUSB_LOG_LEVEL_DEBUG);
  119. /* Now create a reference to the default context */
  120. r = libusb_init(NULL);
  121. if (r != LIBUSB_SUCCESS) {
  122. libusb_testlib_logf("Failed to init libusb: %d", r);
  123. libusb_exit(ctx);
  124. return TEST_STATUS_FAILURE;
  125. }
  126. /* Destroy the first context */
  127. libusb_exit(ctx);
  128. /* Destroy the default context */
  129. libusb_exit(NULL);
  130. }
  131. return TEST_STATUS_SUCCESS;
  132. }
  133. /* Fill in the list of tests. */
  134. static const libusb_testlib_test tests[] = {
  135. { "init_and_exit", &test_init_and_exit },
  136. { "get_device_list", &test_get_device_list },
  137. { "many_device_lists", &test_many_device_lists },
  138. { "default_context_change", &test_default_context_change },
  139. LIBUSB_NULL_TEST
  140. };
  141. int main(int argc, char *argv[])
  142. {
  143. return libusb_testlib_run_tests(argc, argv, tests);
  144. }