wait.cc 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. /* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
  2. *
  3. * Data Differential YATL (i.e. libtest) library
  4. *
  5. * Copyright (C) 2012 Data Differential, http://datadifferential.com/
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions are
  9. * met:
  10. *
  11. * * Redistributions of source code must retain the above copyright
  12. * notice, this list of conditions and the following disclaimer.
  13. *
  14. * * Redistributions in binary form must reproduce the above
  15. * copyright notice, this list of conditions and the following disclaimer
  16. * in the documentation and/or other materials provided with the
  17. * distribution.
  18. *
  19. * * The names of its contributors may not be used to endorse or
  20. * promote products derived from this software without specific prior
  21. * written permission.
  22. *
  23. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  24. * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  25. * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  26. * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  27. * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  28. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  29. * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  30. * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  31. * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  32. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  33. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  34. *
  35. */
  36. #include "libtest/yatlcon.h"
  37. #include <cstdlib>
  38. #include <fcntl.h>
  39. #include <getopt.h>
  40. #include <iostream>
  41. #include <sys/stat.h>
  42. #include <sys/types.h>
  43. #include <unistd.h>
  44. #include <libtest/wait.h>
  45. static void version_command(const char *command_name, int major_version, int minor_version)
  46. {
  47. std::cout << command_name << " " << major_version << "." << minor_version << std::endl;
  48. }
  49. static void help_command(const char *command_name,
  50. int major_version, int minor_version,
  51. const struct option *long_options)
  52. {
  53. std::cout << command_name << " " << major_version << "." << minor_version << std::endl;
  54. std::cout << "Current options. A '=' means the option takes a value." << std::endl << std::endl;
  55. for (uint32_t x= 0; long_options[x].name; x++)
  56. {
  57. std::cout << "\t --" << long_options[x].name << char(long_options[x].has_arg ? '=' : ' ') << std::endl;
  58. }
  59. std::cout << std::endl;
  60. }
  61. static void close_stdio(void)
  62. {
  63. int fd;
  64. if ((fd = open("/dev/null", O_RDWR, 0)) < 0)
  65. {
  66. return;
  67. }
  68. else
  69. {
  70. if (dup2(fd, STDIN_FILENO) < 0)
  71. {
  72. return;
  73. }
  74. if (dup2(fd, STDOUT_FILENO) < 0)
  75. {
  76. return;
  77. }
  78. if (dup2(fd, STDERR_FILENO) < 0)
  79. {
  80. return;
  81. }
  82. if (fd > STDERR_FILENO)
  83. {
  84. close(fd);
  85. }
  86. }
  87. }
  88. enum {
  89. OPT_HELP,
  90. OPT_QUIET,
  91. OPT_VERSION
  92. };
  93. static void options_parse(int argc, char *argv[])
  94. {
  95. static struct option long_options[]=
  96. {
  97. { "version", no_argument, NULL, OPT_VERSION},
  98. { "help", no_argument, NULL, OPT_HELP},
  99. { "quiet", no_argument, NULL, OPT_QUIET},
  100. {0, 0, 0, 0},
  101. };
  102. bool opt_version= false;
  103. bool opt_help= false;
  104. bool opt_quiet= false;
  105. int option_index= 0;
  106. while (1)
  107. {
  108. int option_rv= getopt_long(argc, argv, "", long_options, &option_index);
  109. if (option_rv == -1)
  110. {
  111. break;
  112. }
  113. switch (option_rv)
  114. {
  115. case OPT_HELP: /* --help or -h */
  116. opt_help= true;
  117. break;
  118. case OPT_VERSION: /* --version or -v */
  119. opt_version= true;
  120. break;
  121. case OPT_QUIET:
  122. opt_quiet= true;
  123. break;
  124. case '?':
  125. /* getopt_long already printed an error message. */
  126. exit(EXIT_FAILURE);
  127. default:
  128. help_command(argv[0], 1, 0, long_options);
  129. exit(EXIT_FAILURE);
  130. }
  131. }
  132. if (opt_quiet)
  133. {
  134. close_stdio();
  135. }
  136. if (opt_version)
  137. {
  138. version_command(argv[0], 1, 0);
  139. exit(EXIT_SUCCESS);
  140. }
  141. if (opt_help)
  142. {
  143. help_command(argv[0], 1, 0, long_options);
  144. exit(EXIT_SUCCESS);
  145. }
  146. }
  147. int main(int argc, char *argv[])
  148. {
  149. if (argc == 1)
  150. {
  151. return EXIT_FAILURE;
  152. }
  153. options_parse(argc, argv);
  154. int ret= EXIT_FAILURE;
  155. while (optind < argc)
  156. {
  157. libtest::Wait wait(argv[optind++]);
  158. if (wait.successful() == false)
  159. {
  160. return EXIT_FAILURE;
  161. }
  162. ret= EXIT_SUCCESS;
  163. }
  164. return ret;
  165. }