testautomation_stdlib.c 30 KB

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