module_links.c 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. #include <tools/ccanlint/ccanlint.h>
  2. #include <tools/tools.h>
  3. #include <ccan/talloc/talloc.h>
  4. #include <ccan/str/str.h>
  5. #include <sys/types.h>
  6. #include <sys/stat.h>
  7. #include <fcntl.h>
  8. #include <unistd.h>
  9. #include <limits.h>
  10. #include <errno.h>
  11. #include <stdlib.h>
  12. #include <stdio.h>
  13. #include <err.h>
  14. #include <string.h>
  15. #include <ctype.h>
  16. static const char *can_build(struct manifest *m)
  17. {
  18. if (safe_mode)
  19. return "Safe mode enabled";
  20. return NULL;
  21. }
  22. static char *obj_list(const struct manifest *m)
  23. {
  24. char *list = talloc_strdup(m, "");
  25. struct manifest *i;
  26. /* Other CCAN deps. */
  27. list_for_each(&m->deps, i, list) {
  28. if (i->compiled[COMPILE_NORMAL])
  29. list = talloc_asprintf_append(list, "%s ",
  30. i->compiled
  31. [COMPILE_NORMAL]);
  32. }
  33. return list;
  34. }
  35. static char *lib_list(const struct manifest *m)
  36. {
  37. unsigned int i, num;
  38. char **libs = get_libs(m, ".",
  39. &num, &m->info_file->compiled[COMPILE_NORMAL]);
  40. char *ret = talloc_strdup(m, "");
  41. for (i = 0; i < num; i++)
  42. ret = talloc_asprintf_append(ret, "-l%s ", libs[i]);
  43. return ret;
  44. }
  45. static void check_use_build(struct manifest *m,
  46. bool keep,
  47. unsigned int *timeleft, struct score *score)
  48. {
  49. char *contents;
  50. char *tmpfile, *cmdout;
  51. char *basename = talloc_asprintf(m, "%s/example.c", m->dir);
  52. int fd;
  53. tmpfile = maybe_temp_file(m, ".c", keep, basename);
  54. fd = open(tmpfile, O_WRONLY | O_CREAT | O_EXCL, 0600);
  55. if (fd < 0)
  56. err(1, "Creating temporary file %s", tmpfile);
  57. contents = talloc_asprintf(tmpfile,
  58. "#include <ccan/%s/%s.h>\n"
  59. "int main(void)\n"
  60. "{\n"
  61. " return 0;\n"
  62. "}\n",
  63. m->basename, m->basename);
  64. if (write(fd, contents, strlen(contents)) != strlen(contents))
  65. err(1, "Failure writing to temporary file %s", tmpfile);
  66. close(fd);
  67. if (compile_and_link(score, tmpfile, ccan_dir, obj_list(m),
  68. compiler, cflags, lib_list(m),
  69. maybe_temp_file(m, "", keep, tmpfile),
  70. &cmdout)) {
  71. score->pass = true;
  72. score->score = score->total;
  73. } else {
  74. score->error = cmdout;
  75. }
  76. }
  77. struct ccanlint module_links = {
  78. .key = "module_links",
  79. .name = "Module can be linked against trivial program",
  80. .check = check_use_build,
  81. .can_run = can_build,
  82. .needs = "module_builds depends_build"
  83. };
  84. REGISTER_TEST(module_links);