testautomation_stdlib.c 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609
  1. /**
  2. * Standard C library routine test suite
  3. */
  4. #include <SDL3/SDL.h>
  5. #include <SDL3/SDL_test.h>
  6. /* Test case functions */
  7. /**
  8. * @brief Call to SDL_strlcpy
  9. */
  10. #undef SDL_strlcpy
  11. int stdlib_strlcpy(void *arg)
  12. {
  13. size_t result;
  14. char text[1024];
  15. const char *expected;
  16. result = SDL_strlcpy(text, "foo", sizeof(text));
  17. expected = "foo";
  18. SDLTest_AssertPass("Call to SDL_strlcpy(\"foo\")");
  19. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  20. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), (int)result);
  21. result = SDL_strlcpy(text, "foo", 2);
  22. expected = "f";
  23. SDLTest_AssertPass("Call to SDL_strlcpy(\"foo\") with buffer size 2");
  24. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  25. SDLTest_AssertCheck(result == 3, "Check result value, expected: 3, got: %d", (int)result);
  26. return TEST_COMPLETED;
  27. }
  28. #if defined(HAVE_WFORMAT) || defined(HAVE_WFORMAT_EXTRA_ARGS)
  29. #pragma GCC diagnostic push
  30. #if defined(HAVE_WFORMAT)
  31. #pragma GCC diagnostic ignored "-Wformat"
  32. #endif
  33. #if defined(HAVE_WFORMAT_EXTRA_ARGS)
  34. #pragma GCC diagnostic ignored "-Wformat-extra-args"
  35. #endif
  36. #endif
  37. /**
  38. * @brief Call to SDL_snprintf
  39. */
  40. #undef SDL_snprintf
  41. int stdlib_snprintf(void *arg)
  42. {
  43. int result;
  44. int predicted;
  45. char text[1024];
  46. const char *expected;
  47. size_t size;
  48. result = SDL_snprintf(text, sizeof(text), "%s", "foo");
  49. expected = "foo";
  50. SDLTest_AssertPass("Call to SDL_snprintf(\"%%s\", \"foo\")");
  51. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  52. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  53. result = SDL_snprintf(text, 2, "%s", "foo");
  54. expected = "f";
  55. SDLTest_AssertPass("Call to SDL_snprintf(\"%%s\", \"foo\") with buffer size 2");
  56. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  57. SDLTest_AssertCheck(result == 3, "Check result value, expected: 3, got: %d", result);
  58. result = SDL_snprintf(NULL, 0, "%s", "foo");
  59. SDLTest_AssertPass("Call to SDL_snprintf(NULL, 0, \"%%s\", \"foo\")");
  60. SDLTest_AssertCheck(result == 3, "Check result value, expected: 3, got: %d", result);
  61. result = SDL_snprintf(text, 2, "%s\n", "foo");
  62. expected = "f";
  63. SDLTest_AssertPass("Call to SDL_snprintf(\"%%s\\n\", \"foo\") with buffer size 2");
  64. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  65. SDLTest_AssertCheck(result == 4, "Check result value, expected: 4, got: %d", result);
  66. result = SDL_snprintf(text, sizeof(text), "%f", 0.0);
  67. predicted = SDL_snprintf(NULL, 0, "%f", 0.0);
  68. expected = "0.000000";
  69. SDLTest_AssertPass("Call to SDL_snprintf(\"%%f\", 0.0)");
  70. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  71. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  72. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  73. result = SDL_snprintf(text, sizeof(text), "%f", 1.0);
  74. predicted = SDL_snprintf(NULL, 0, "%f", 1.0);
  75. expected = "1.000000";
  76. SDLTest_AssertPass("Call to SDL_snprintf(\"%%f\", 1.0)");
  77. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  78. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  79. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  80. result = SDL_snprintf(text, sizeof(text), "%.f", 1.0);
  81. predicted = SDL_snprintf(NULL, 0, "%.f", 1.0);
  82. expected = "1";
  83. SDLTest_AssertPass("Call to SDL_snprintf(\"%%.f\", 1.0)");
  84. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  85. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  86. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  87. result = SDL_snprintf(text, sizeof(text), "%#.f", 1.0);
  88. predicted = SDL_snprintf(NULL, 0, "%#.f", 1.0);
  89. expected = "1.";
  90. SDLTest_AssertPass("Call to SDL_snprintf(\"%%#.f\", 1.0)");
  91. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  92. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  93. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  94. result = SDL_snprintf(text, sizeof(text), "%f", 1.0 + 1.0 / 3.0);
  95. predicted = SDL_snprintf(NULL, 0, "%f", 1.0 + 1.0 / 3.0);
  96. expected = "1.333333";
  97. SDLTest_AssertPass("Call to SDL_snprintf(\"%%f\", 1.0 + 1.0 / 3.0)");
  98. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  99. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  100. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  101. result = SDL_snprintf(text, sizeof(text), "%+f", 1.0 + 1.0 / 3.0);
  102. predicted = SDL_snprintf(NULL, 0, "%+f", 1.0 + 1.0 / 3.0);
  103. expected = "+1.333333";
  104. SDLTest_AssertPass("Call to SDL_snprintf(\"%%+f\", 1.0 + 1.0 / 3.0)");
  105. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  106. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  107. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  108. result = SDL_snprintf(text, sizeof(text), "%.2f", 1.0 + 1.0 / 3.0);
  109. predicted = SDL_snprintf(NULL, 0, "%.2f", 1.0 + 1.0 / 3.0);
  110. expected = "1.33";
  111. SDLTest_AssertPass("Call to SDL_snprintf(\"%%.2f\", 1.0 + 1.0 / 3.0)");
  112. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: %s, got: %s", expected, text);
  113. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  114. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  115. result = SDL_snprintf(text, sizeof(text), "%6.2f", 1.0 + 1.0 / 3.0);
  116. predicted = SDL_snprintf(NULL, 0, "%6.2f", 1.0 + 1.0 / 3.0);
  117. expected = " 1.33";
  118. SDLTest_AssertPass("Call to SDL_snprintf(\"%%6.2f\", 1.0 + 1.0 / 3.0)");
  119. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: '%s', got: '%s'", expected, text);
  120. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  121. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  122. result = SDL_snprintf(text, sizeof(text), "%06.2f", 1.0 + 1.0 / 3.0);
  123. predicted = SDL_snprintf(NULL, 0, "%06.2f", 1.0 + 1.0 / 3.0);
  124. expected = "001.33";
  125. SDLTest_AssertPass("Call to SDL_snprintf(\"%%06.2f\", 1.0 + 1.0 / 3.0)");
  126. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: '%s', got: '%s'", expected, text);
  127. SDLTest_AssertCheck(result == SDL_strlen(text), "Check result value, expected: %d, got: %d", (int)SDL_strlen(text), result);
  128. SDLTest_AssertCheck(predicted == result, "Check predicted value, expected: %d, got: %d", result, predicted);
  129. result = SDL_snprintf(text, 5, "%06.2f", 1.0 + 1.0 / 3.0);
  130. expected = "001.";
  131. SDLTest_AssertPass("Call to SDL_snprintf(\"%%06.2f\", 1.0 + 1.0 / 3.0) with buffer size 5");
  132. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: '%s', got: '%s'", expected, text);
  133. SDLTest_AssertCheck(result == 6, "Check result value, expected: 6, got: %d", result);
  134. result = SDL_snprintf(text, sizeof(text), "%g", 100.0);
  135. expected = "100";
  136. SDLTest_AssertPass("Call to SDL_snprintf(\"%%g\", 100.0)");
  137. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: '%s', got: '%s'", expected, text);
  138. SDLTest_AssertCheck(result == 3, "Check result value, expected: 3, got: %d", result);
  139. result = SDL_snprintf(text, sizeof(text), "%g", 100.75);
  140. expected = "100.75";
  141. SDLTest_AssertPass("Call to SDL_snprintf(\"%%g\", 100.75)");
  142. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: '%s', got: '%s'", expected, text);
  143. SDLTest_AssertCheck(result == 6, "Check result value, expected: 6, got: %d", result);
  144. size = 64;
  145. result = SDL_snprintf(text, sizeof(text), "%zu %s", size, "test");
  146. expected = "64 test";
  147. SDLTest_AssertPass("Call to SDL_snprintf(text, sizeof(text), \"%%zu %%s\", size, \"test\")");
  148. SDLTest_AssertCheck(SDL_strcmp(text, expected) == 0, "Check text, expected: '%s', got: '%s'", expected, text);
  149. SDLTest_AssertCheck(result == 7, "Check result value, expected: 7, got: %d", result);
  150. return TEST_COMPLETED;
  151. }
  152. #if defined(HAVE_WFORMAT) || defined(HAVE_WFORMAT_EXTRA_ARGS)
  153. #pragma GCC diagnostic pop
  154. #endif
  155. /**
  156. * @brief Call to SDL_getenv and SDL_setenv
  157. */
  158. int stdlib_getsetenv(void *arg)
  159. {
  160. const int nameLen = 16;
  161. char name[17];
  162. int counter;
  163. int result;
  164. char *value1;
  165. char *value2;
  166. char *expected;
  167. int overwrite;
  168. char *text;
  169. /* Create a random name. This tests SDL_getenv, since we need to */
  170. /* make sure the variable is not set yet (it shouldn't). */
  171. do {
  172. for (counter = 0; counter < nameLen; counter++) {
  173. name[counter] = (char)SDLTest_RandomIntegerInRange(65, 90);
  174. }
  175. name[nameLen] = '\0';
  176. text = SDL_getenv(name);
  177. SDLTest_AssertPass("Call to SDL_getenv('%s')", name);
  178. if (text != NULL) {
  179. SDLTest_Log("Expected: NULL, Got: '%s' (%i)", text, (int)SDL_strlen(text));
  180. }
  181. } while (text != NULL);
  182. /* Create random values to set */
  183. value1 = SDLTest_RandomAsciiStringOfSize(10);
  184. value2 = SDLTest_RandomAsciiStringOfSize(10);
  185. /* Set value 1 without overwrite */
  186. overwrite = 0;
  187. expected = value1;
  188. result = SDL_setenv(name, value1, overwrite);
  189. SDLTest_AssertPass("Call to SDL_setenv('%s','%s', %i)", name, value1, overwrite);
  190. SDLTest_AssertCheck(result == 0, "Check result, expected: 0, got: %i", result);
  191. /* Check value */
  192. text = SDL_getenv(name);
  193. SDLTest_AssertPass("Call to SDL_getenv('%s')", name);
  194. SDLTest_AssertCheck(text != NULL, "Verify returned text is not NULL");
  195. if (text != NULL) {
  196. SDLTest_AssertCheck(
  197. SDL_strcmp(text, expected) == 0,
  198. "Verify returned text, expected: %s, got: %s",
  199. expected,
  200. text);
  201. }
  202. /* Set value 2 with overwrite */
  203. overwrite = 1;
  204. expected = value2;
  205. result = SDL_setenv(name, value2, overwrite);
  206. SDLTest_AssertPass("Call to SDL_setenv('%s','%s', %i)", name, value2, overwrite);
  207. SDLTest_AssertCheck(result == 0, "Check result, expected: 0, got: %i", result);
  208. /* Check value */
  209. text = SDL_getenv(name);
  210. SDLTest_AssertPass("Call to SDL_getenv('%s')", name);
  211. SDLTest_AssertCheck(text != NULL, "Verify returned text is not NULL");
  212. if (text != NULL) {
  213. SDLTest_AssertCheck(
  214. SDL_strcmp(text, expected) == 0,
  215. "Verify returned text, expected: %s, got: %s",
  216. expected,
  217. text);
  218. }
  219. /* Set value 1 without overwrite */
  220. overwrite = 0;
  221. expected = value2;
  222. result = SDL_setenv(name, value1, overwrite);
  223. SDLTest_AssertPass("Call to SDL_setenv('%s','%s', %i)", name, value1, overwrite);
  224. SDLTest_AssertCheck(result == 0, "Check result, expected: 0, got: %i", result);
  225. /* Check value */
  226. text = SDL_getenv(name);
  227. SDLTest_AssertPass("Call to SDL_getenv('%s')", name);
  228. SDLTest_AssertCheck(text != NULL, "Verify returned text is not NULL");
  229. if (text != NULL) {
  230. SDLTest_AssertCheck(
  231. SDL_strcmp(text, expected) == 0,
  232. "Verify returned text, expected: %s, got: %s",
  233. expected,
  234. text);
  235. }
  236. /* Set value 1 without overwrite */
  237. overwrite = 1;
  238. expected = value1;
  239. result = SDL_setenv(name, value1, overwrite);
  240. SDLTest_AssertPass("Call to SDL_setenv('%s','%s', %i)", name, value1, overwrite);
  241. SDLTest_AssertCheck(result == 0, "Check result, expected: 0, got: %i", result);
  242. /* Check value */
  243. text = SDL_getenv(name);
  244. SDLTest_AssertPass("Call to SDL_getenv('%s')", name);
  245. SDLTest_AssertCheck(text != NULL, "Verify returned text is not NULL");
  246. if (text != NULL) {
  247. SDLTest_AssertCheck(
  248. SDL_strcmp(text, expected) == 0,
  249. "Verify returned text, expected: %s, got: %s",
  250. expected,
  251. text);
  252. }
  253. /* Negative cases */
  254. for (overwrite = 0; overwrite <= 1; overwrite++) {
  255. result = SDL_setenv(NULL, value1, overwrite);
  256. SDLTest_AssertPass("Call to SDL_setenv(NULL,'%s', %i)", value1, overwrite);
  257. SDLTest_AssertCheck(result == -1, "Check result, expected: -1, got: %i", result);
  258. result = SDL_setenv("", value1, overwrite);
  259. SDLTest_AssertPass("Call to SDL_setenv('','%s', %i)", value1, overwrite);
  260. SDLTest_AssertCheck(result == -1, "Check result, expected: -1, got: %i", result);
  261. result = SDL_setenv("=", value1, overwrite);
  262. SDLTest_AssertPass("Call to SDL_setenv('=','%s', %i)", value1, overwrite);
  263. SDLTest_AssertCheck(result == -1, "Check result, expected: -1, got: %i", result);
  264. result = SDL_setenv(name, NULL, overwrite);
  265. SDLTest_AssertPass("Call to SDL_setenv('%s', NULL, %i)", name, overwrite);
  266. SDLTest_AssertCheck(result == -1, "Check result, expected: -1, got: %i", result);
  267. }
  268. /* Clean up */
  269. SDL_free(value1);
  270. SDL_free(value2);
  271. return TEST_COMPLETED;
  272. }
  273. #if defined(HAVE_WFORMAT) || defined(HAVE_WFORMAT_EXTRA_ARGS)
  274. #pragma GCC diagnostic push
  275. #if defined(HAVE_WFORMAT)
  276. #pragma GCC diagnostic ignored "-Wformat"
  277. #endif
  278. #if defined(HAVE_WFORMAT_EXTRA_ARGS)
  279. #pragma GCC diagnostic ignored "-Wformat-extra-args"
  280. #endif
  281. #endif
  282. /**
  283. * @brief Call to SDL_sscanf
  284. */
  285. #undef SDL_sscanf
  286. int stdlib_sscanf(void *arg)
  287. {
  288. int output;
  289. int result;
  290. int expected_output;
  291. int expected_result;
  292. short short_output, expected_short_output;
  293. long long_output, expected_long_output;
  294. long long long_long_output, expected_long_long_output;
  295. size_t size_output, expected_size_output;
  296. char text[128];
  297. expected_output = output = 123;
  298. expected_result = -1;
  299. result = SDL_sscanf("", "%i", &output);
  300. SDLTest_AssertPass("Call to SDL_sscanf(\"\", \"%%i\", &output)");
  301. SDLTest_AssertCheck(expected_output == output, "Check output, expected: %i, got: %i", expected_output, output);
  302. SDLTest_AssertCheck(expected_result == result, "Check return value, expected: %i, got: %i", expected_result, result);
  303. expected_output = output = 123;
  304. expected_result = 0;
  305. result = SDL_sscanf("a", "%i", &output);
  306. SDLTest_AssertPass("Call to SDL_sscanf(\"a\", \"%%i\", &output)");
  307. SDLTest_AssertCheck(expected_output == output, "Check output, expected: %i, got: %i", expected_output, output);
  308. SDLTest_AssertCheck(expected_result == result, "Check return value, expected: %i, got: %i", expected_result, result);
  309. output = 123;
  310. expected_output = 2;
  311. expected_result = 1;
  312. result = SDL_sscanf("2", "%i", &output);
  313. SDLTest_AssertPass("Call to SDL_sscanf(\"2\", \"%%i\", &output)");
  314. SDLTest_AssertCheck(expected_output == output, "Check output, expected: %i, got: %i", expected_output, output);
  315. SDLTest_AssertCheck(expected_result == result, "Check return value, expected: %i, got: %i", expected_result, result);
  316. output = 123;
  317. expected_output = 0xa;
  318. expected_result = 1;
  319. result = SDL_sscanf("aa", "%1x", &output);
  320. SDLTest_AssertPass("Call to SDL_sscanf(\"aa\", \"%%1x\", &output)");
  321. SDLTest_AssertCheck(expected_output == output, "Check output, expected: %i, got: %i", expected_output, output);
  322. SDLTest_AssertCheck(expected_result == result, "Check return value, expected: %i, got: %i", expected_result, result);
  323. #define SIZED_TEST_CASE(type, var, format_specifier) \
  324. var##_output = 123; \
  325. expected_##var##_output = (type)(((unsigned type)(~0)) >> 1); \
  326. expected_result = 1; \
  327. result = SDL_snprintf(text, sizeof(text), format_specifier, expected_##var##_output); \
  328. result = SDL_sscanf(text, format_specifier, &var##_output); \
  329. SDLTest_AssertPass("Call to SDL_sscanf(\"%s\", \"%s\", &output)", text, #format_specifier); \
  330. SDLTest_AssertCheck(expected_##var##_output == var##_output, "Check output, expected: " format_specifier ", got: " format_specifier, expected_##var##_output, var##_output); \
  331. SDLTest_AssertCheck(expected_result == result, "Check return value, expected: %i, got: %i", expected_result, result); \
  332. \
  333. var##_output = 123; \
  334. expected_##var##_output = ~(type)(((unsigned type)(~0)) >> 1); \
  335. expected_result = 1; \
  336. result = SDL_snprintf(text, sizeof(text), format_specifier, expected_##var##_output); \
  337. result = SDL_sscanf(text, format_specifier, &var##_output); \
  338. SDLTest_AssertPass("Call to SDL_sscanf(\"%s\", \"%s\", &output)", text, #format_specifier); \
  339. SDLTest_AssertCheck(expected_##var##_output == var##_output, "Check output, expected: " format_specifier ", got: " format_specifier, expected_##var##_output, var##_output); \
  340. SDLTest_AssertCheck(expected_result == result, "Check return value, expected: %i, got: %i", expected_result, result);
  341. SIZED_TEST_CASE(short, short, "%hd")
  342. SIZED_TEST_CASE(long, long, "%ld")
  343. SIZED_TEST_CASE(long long, long_long, "%lld")
  344. size_output = 123;
  345. expected_size_output = ~((size_t)0);
  346. expected_result = 1;
  347. result = SDL_snprintf(text, sizeof(text), "%zu", expected_size_output);
  348. result = SDL_sscanf(text, "%zu", &size_output);
  349. SDLTest_AssertPass("Call to SDL_sscanf(\"%s\", \"%%zu\", &output)", text);
  350. SDLTest_AssertCheck(expected_size_output == size_output, "Check output, expected: %zu, got: %zu", expected_size_output, size_output);
  351. SDLTest_AssertCheck(expected_result == result, "Check return value, expected: %i, got: %i", expected_result, result);
  352. return TEST_COMPLETED;
  353. }
  354. #if defined(HAVE_WFORMAT) || defined(HAVE_WFORMAT_EXTRA_ARGS)
  355. #pragma GCC diagnostic pop
  356. #endif
  357. #if defined(_WIN64)
  358. #define SIZE_FORMAT "I64u"
  359. #elif defined(__WIN32__)
  360. #define SIZE_FORMAT "I32u"
  361. #else
  362. #define SIZE_FORMAT "zu"
  363. #endif
  364. typedef struct
  365. {
  366. size_t a;
  367. size_t b;
  368. size_t result;
  369. int status;
  370. } overflow_test;
  371. static const overflow_test multiplications[] = {
  372. { 1, 1, 1, 0 },
  373. { 0, 0, 0, 0 },
  374. { SDL_SIZE_MAX, 0, 0, 0 },
  375. { SDL_SIZE_MAX, 1, SDL_SIZE_MAX, 0 },
  376. { SDL_SIZE_MAX / 2, 2, SDL_SIZE_MAX - (SDL_SIZE_MAX % 2), 0 },
  377. { SDL_SIZE_MAX / 23, 23, SDL_SIZE_MAX - (SDL_SIZE_MAX % 23), 0 },
  378. { (SDL_SIZE_MAX / 2) + 1, 2, 0, -1 },
  379. { (SDL_SIZE_MAX / 23) + 42, 23, 0, -1 },
  380. { SDL_SIZE_MAX, SDL_SIZE_MAX, 0, -1 },
  381. };
  382. static const overflow_test additions[] = {
  383. { 1, 1, 2, 0 },
  384. { 0, 0, 0, 0 },
  385. { SDL_SIZE_MAX, 0, SDL_SIZE_MAX, 0 },
  386. { SDL_SIZE_MAX - 1, 1, SDL_SIZE_MAX, 0 },
  387. { SDL_SIZE_MAX - 42, 23, SDL_SIZE_MAX - (42 - 23), 0 },
  388. { SDL_SIZE_MAX, 1, 0, -1 },
  389. { SDL_SIZE_MAX, 23, 0, -1 },
  390. { SDL_SIZE_MAX, SDL_SIZE_MAX, 0, -1 },
  391. };
  392. static int
  393. stdlib_overflow(void *arg)
  394. {
  395. size_t i;
  396. size_t useBuiltin;
  397. for (useBuiltin = 0; useBuiltin < 2; useBuiltin++) {
  398. if (useBuiltin) {
  399. SDLTest_Log("Using gcc/clang builtins if possible");
  400. } else {
  401. SDLTest_Log("Not using gcc/clang builtins");
  402. }
  403. for (i = 0; i < SDL_arraysize(multiplications); i++) {
  404. const overflow_test *t = &multiplications[i];
  405. int status;
  406. size_t result = ~t->result;
  407. if (useBuiltin) {
  408. status = SDL_size_mul_overflow(t->a, t->b, &result);
  409. } else {
  410. /* This disables the macro that tries to use a gcc/clang
  411. * builtin, so we test the fallback implementation instead. */
  412. status = (SDL_size_mul_overflow)(t->a, t->b, &result);
  413. }
  414. if (t->status == 0) {
  415. SDLTest_AssertCheck(status == 0,
  416. "(%" SIZE_FORMAT " * %" SIZE_FORMAT ") should succeed",
  417. t->a, t->b);
  418. SDLTest_AssertCheck(result == t->result,
  419. "(%" SIZE_FORMAT " * %" SIZE_FORMAT "): expected %" SIZE_FORMAT ", got %" SIZE_FORMAT,
  420. t->a, t->b, t->result, result);
  421. } else {
  422. SDLTest_AssertCheck(status == -1,
  423. "(%" SIZE_FORMAT " * %" SIZE_FORMAT ") should fail",
  424. t->a, t->b);
  425. }
  426. if (t->a == t->b) {
  427. continue;
  428. }
  429. result = ~t->result;
  430. if (useBuiltin) {
  431. status = SDL_size_mul_overflow(t->b, t->a, &result);
  432. } else {
  433. status = (SDL_size_mul_overflow)(t->b, t->a, &result);
  434. }
  435. if (t->status == 0) {
  436. SDLTest_AssertCheck(status == 0,
  437. "(%" SIZE_FORMAT " * %" SIZE_FORMAT ") should succeed",
  438. t->b, t->a);
  439. SDLTest_AssertCheck(result == t->result,
  440. "(%" SIZE_FORMAT " * %" SIZE_FORMAT "): expected %" SIZE_FORMAT ", got %" SIZE_FORMAT,
  441. t->b, t->a, t->result, result);
  442. } else {
  443. SDLTest_AssertCheck(status == -1,
  444. "(%" SIZE_FORMAT " * %" SIZE_FORMAT ") should fail",
  445. t->b, t->a);
  446. }
  447. }
  448. for (i = 0; i < SDL_arraysize(additions); i++) {
  449. const overflow_test *t = &additions[i];
  450. int status;
  451. size_t result = ~t->result;
  452. if (useBuiltin) {
  453. status = SDL_size_add_overflow(t->a, t->b, &result);
  454. } else {
  455. status = (SDL_size_add_overflow)(t->a, t->b, &result);
  456. }
  457. if (t->status == 0) {
  458. SDLTest_AssertCheck(status == 0,
  459. "(%" SIZE_FORMAT " + %" SIZE_FORMAT ") should succeed",
  460. t->a, t->b);
  461. SDLTest_AssertCheck(result == t->result,
  462. "(%" SIZE_FORMAT " + %" SIZE_FORMAT "): expected %" SIZE_FORMAT ", got %" SIZE_FORMAT,
  463. t->a, t->b, t->result, result);
  464. } else {
  465. SDLTest_AssertCheck(status == -1,
  466. "(%" SIZE_FORMAT " + %" SIZE_FORMAT ") should fail",
  467. t->a, t->b);
  468. }
  469. if (t->a == t->b) {
  470. continue;
  471. }
  472. result = ~t->result;
  473. if (useBuiltin) {
  474. status = SDL_size_add_overflow(t->b, t->a, &result);
  475. } else {
  476. status = (SDL_size_add_overflow)(t->b, t->a, &result);
  477. }
  478. if (t->status == 0) {
  479. SDLTest_AssertCheck(status == 0,
  480. "(%" SIZE_FORMAT " + %" SIZE_FORMAT ") should succeed",
  481. t->b, t->a);
  482. SDLTest_AssertCheck(result == t->result,
  483. "(%" SIZE_FORMAT " + %" SIZE_FORMAT "): expected %" SIZE_FORMAT ", got %" SIZE_FORMAT,
  484. t->b, t->a, t->result, result);
  485. } else {
  486. SDLTest_AssertCheck(status == -1,
  487. "(%" SIZE_FORMAT " + %" SIZE_FORMAT ") should fail",
  488. t->b, t->a);
  489. }
  490. }
  491. }
  492. return TEST_COMPLETED;
  493. }
  494. /* ================= Test References ================== */
  495. /* Standard C routine test cases */
  496. static const SDLTest_TestCaseReference stdlibTest1 = {
  497. (SDLTest_TestCaseFp)stdlib_strlcpy, "stdlib_strlcpy", "Call to SDL_strlcpy", TEST_ENABLED
  498. };
  499. static const SDLTest_TestCaseReference stdlibTest2 = {
  500. (SDLTest_TestCaseFp)stdlib_snprintf, "stdlib_snprintf", "Call to SDL_snprintf", TEST_ENABLED
  501. };
  502. static const SDLTest_TestCaseReference stdlibTest3 = {
  503. (SDLTest_TestCaseFp)stdlib_getsetenv, "stdlib_getsetenv", "Call to SDL_getenv and SDL_setenv", TEST_ENABLED
  504. };
  505. static const SDLTest_TestCaseReference stdlibTest4 = {
  506. (SDLTest_TestCaseFp)stdlib_sscanf, "stdlib_sscanf", "Call to SDL_sscanf", TEST_ENABLED
  507. };
  508. static const SDLTest_TestCaseReference stdlibTestOverflow = {
  509. stdlib_overflow, "stdlib_overflow", "Overflow detection", TEST_ENABLED
  510. };
  511. /* Sequence of Standard C routine test cases */
  512. static const SDLTest_TestCaseReference *stdlibTests[] = {
  513. &stdlibTest1,
  514. &stdlibTest2,
  515. &stdlibTest3,
  516. &stdlibTest4,
  517. &stdlibTestOverflow,
  518. NULL
  519. };
  520. /* Standard C routine test suite (global) */
  521. SDLTest_TestSuiteReference stdlibTestSuite = {
  522. "Stdlib",
  523. NULL,
  524. stdlibTests,
  525. NULL
  526. };