testautomation_stdlib.c 24 KB

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