logo

utils-std

Collection of commonly available Unix tools git clone https://anongit.hacktivis.me/git/utils-std.git/

time.c (3774B)


  1. // utils-std: Collection of commonly available Unix tools
  2. // SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me>
  3. // SPDX-License-Identifier: MPL-2.0
  4. #define _POSIX_C_SOURCE 200809L
  5. #include "../lib/getopt_nolong.h"
  6. #include <errno.h> // errno
  7. #include <stdio.h> // perror, fprintf
  8. #include <stdlib.h> // abort
  9. #include <string.h> // strerror
  10. #include <sys/resource.h> // getrusage
  11. #include <sys/times.h> // times
  12. #include <sys/wait.h> // waitpid
  13. #include <unistd.h> // sysconf, fork, execvp, getopt
  14. const char *argv0 = "time";
  15. enum cmd_time_mode
  16. {
  17. CMD_TIME_POSIX = 0,
  18. CMD_TIME_VERBOSE = 1,
  19. };
  20. static void
  21. usage(void)
  22. {
  23. fprintf(stderr, "Usage: time [-p|-v] command [argument ...]\n");
  24. }
  25. int
  26. main(int argc, char *argv[])
  27. {
  28. struct tms tms;
  29. int ret = 0;
  30. enum cmd_time_mode mode = CMD_TIME_POSIX;
  31. if(argc <= 1)
  32. {
  33. usage();
  34. return 0;
  35. }
  36. for(int c = -1; (c = getopt_nolong(argc, argv, ":pv")) != -1;)
  37. {
  38. switch(c)
  39. {
  40. case 'p': // POSIX format (default)
  41. mode = CMD_TIME_POSIX;
  42. break;
  43. case 'v':
  44. mode = CMD_TIME_VERBOSE;
  45. break;
  46. case ':':
  47. fprintf(stderr, "time: error: Missing operand for option: '-%c'\n", optopt);
  48. usage();
  49. return 1;
  50. case '?':
  51. if(!got_long_opt) fprintf(stderr, "time: error: Unrecognised option: '-%c'\n", optopt);
  52. usage();
  53. return 1;
  54. }
  55. }
  56. argc -= optind;
  57. argv += optind;
  58. (void)argc;
  59. long ticks = sysconf(_SC_CLK_TCK);
  60. if(ticks <= 0)
  61. {
  62. perror("time: error: sysconf(_SC_CLK_TCK)");
  63. return 1;
  64. }
  65. clock_t t0 = times(&tms);
  66. if(t0 == (clock_t)-1)
  67. {
  68. perror("time: error: times");
  69. return 1;
  70. }
  71. // Note: posix_spawnp seems to lack enough error information
  72. pid_t pid = fork();
  73. switch(pid)
  74. {
  75. case -1:
  76. perror("time: error: fork");
  77. return 1;
  78. case 0:
  79. execvp(argv[0], argv);
  80. ret = 126 + (errno == ENOENT);
  81. fprintf(stderr, "time: error: Failed executing '%s': %s\n", argv[0], strerror(errno));
  82. return ret;
  83. default:
  84. break;
  85. }
  86. int status = 0;
  87. waitpid(pid, &status, 0);
  88. int t1 = times(&tms);
  89. if(t1 == (clock_t)-1)
  90. {
  91. perror("time: error: times");
  92. return 1;
  93. }
  94. if(WIFSIGNALED(status))
  95. {
  96. fprintf(stderr, "time: error: Command terminated by signal %d\n", WTERMSIG(status));
  97. ret = 128 + WTERMSIG(status);
  98. }
  99. struct rusage usage;
  100. if(getrusage(RUSAGE_CHILDREN, &usage) != 0)
  101. {
  102. fprintf(
  103. stderr, "time: error: Failed getting resource usage of children: %s\n", strerror(errno));
  104. return 1;
  105. }
  106. double utime_s = usage.ru_utime.tv_sec + (usage.ru_utime.tv_usec * 0.000001);
  107. double stime_s = usage.ru_stime.tv_sec + (usage.ru_stime.tv_usec * 0.000001);
  108. switch(mode)
  109. {
  110. case CMD_TIME_POSIX:
  111. fprintf(stderr, "real %f\nuser %f\nsys %f\n", (t1 - t0) / (double)ticks, utime_s, stime_s);
  112. break;
  113. case CMD_TIME_VERBOSE:
  114. fprintf(stderr, "\tExecutable: %s\n", argv[0]);
  115. fprintf(stderr, "\tReal time : %06f\n", (t1 - t0) / (double)ticks);
  116. fprintf(stderr, "\tUser time : %06f\n", utime_s);
  117. fprintf(stderr, "\tSystem time: %06f\n", stime_s);
  118. #ifdef __linux__
  119. fprintf(stderr, "\tMaximum resident set size (maxrss): %ld kB\n", usage.ru_maxrss);
  120. fprintf(stderr, "\tMinor (no I/O required) page faults (minflt): %ld\n", usage.ru_minflt);
  121. fprintf(stderr, "\tMajor (I/O required) page faults (majflt): %ld\n", usage.ru_majflt);
  122. fprintf(stderr, "\tMajor (I/O required) page faults (majflt): %ld\n", usage.ru_majflt);
  123. fprintf(stderr, "\tVoluntary context switch (nvcsw): %ld\n", usage.ru_nvcsw);
  124. fprintf(stderr, "\tInvoluntary context switch (nivcsw): %ld\n", usage.ru_nivcsw);
  125. #endif
  126. if(WIFEXITED(status)) fprintf(stderr, "\tExit status: %d\n", WEXITSTATUS(status));
  127. break;
  128. default:
  129. abort();
  130. }
  131. if(WIFEXITED(status))
  132. {
  133. ret = WEXITSTATUS(status);
  134. }
  135. return ret;
  136. }