You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

service.cpp 5.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. // \file service.cpp
  2. //
  3. // Copyright (C) 2014 MicroNeil Research Corporation.
  4. //
  5. // This program is part of the MicroNeil Research Open Library Project. For
  6. // more information go to http://www.microneil.com/OpenLibrary/index.html
  7. //
  8. // This program is free software; you can redistribute it and/or modify it
  9. // under the terms of the GNU General Public License as published by the
  10. // Free Software Foundation; either version 2 of the License, or (at your
  11. // option) any later version.
  12. //
  13. // This program is distributed in the hope that it will be useful, but WITHOUT
  14. // ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  15. // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  16. // more details.
  17. //
  18. // You should have received a copy of the GNU General Public License along with
  19. // this program; if not, write to the Free Software Foundation, Inc., 59 Temple
  20. // Place, Suite 330, Boston, MA 02111-1307 USA
  21. //==============================================================================
  22. #include <unistd.h>
  23. #include <sys/types.h>
  24. #include <sys/stat.h>
  25. #include <fcntl.h>
  26. #include <signal.h>
  27. #include <iostream>
  28. #include <cstdio>
  29. #include <cstdlib>
  30. #include <thread>
  31. #include <chrono>
  32. #include "service.hpp"
  33. // Main program for *nix daemon.
  34. int main(int argc, char *argv[]) {
  35. CodeDweller::Service &service = CodeDweller::Service::getInstance();
  36. return service.main(argc, argv);
  37. }
  38. namespace CodeDweller {
  39. Service::Service() :
  40. lastMessage(Message::None) {
  41. }
  42. int Service::main(int argc, char *argv[]) {
  43. // Load the arguments.
  44. loadArguments(argc, argv);
  45. pid_t pid;
  46. // Fork the process.
  47. pid = fork();
  48. if (pid < 0) {
  49. perror("Error from first fork");
  50. return(EXIT_FAILURE);
  51. }
  52. // Terminate the parent.
  53. if (pid > 0) {
  54. return(EXIT_SUCCESS);
  55. }
  56. // This is the child. Become the session leader.
  57. if (setsid() < 0) {
  58. perror("Error from setsid");
  59. return(EXIT_FAILURE);
  60. }
  61. // Fork again.
  62. pid = fork();
  63. if (pid < 0) {
  64. perror("Error from second fork");
  65. return(EXIT_FAILURE);
  66. }
  67. // Terminate the parent.
  68. if (pid > 0) {
  69. return(EXIT_SUCCESS);
  70. }
  71. // Set default file permissions. This call always succeeds.
  72. umask(0);
  73. // Disassociate the process from the default directory of the
  74. // grandparent.
  75. if (chdir("/") != 0) {
  76. perror("Error from chdir");
  77. return(EXIT_FAILURE);
  78. }
  79. // Initialize the set of signals to wait for.
  80. if (sigemptyset(&signalSet) != 0) {
  81. perror("Error from sigemptyset");
  82. return(EXIT_FAILURE);
  83. }
  84. if ((sigaddset(&signalSet, SIGTSTP) != 0) ||
  85. (sigaddset(&signalSet, SIGCONT) != 0) ||
  86. (sigaddset(&signalSet, SIGHUP) != 0) ||
  87. (sigaddset(&signalSet, SIGTERM) != 0)) {
  88. perror("Error from sigaddset");
  89. return(EXIT_FAILURE);
  90. }
  91. // Block the signals.
  92. if (sigprocmask(SIG_BLOCK, &signalSet, NULL) != 0) {
  93. perror("Error from sigprocmask");
  94. return(EXIT_FAILURE);
  95. }
  96. // Close all open file descriptors.
  97. for (int fd = 2; fd >= 0; fd--) {
  98. if (close(fd) != 0) {
  99. // There is no controlling terminal to send any message to.
  100. return(EXIT_FAILURE);
  101. }
  102. }
  103. // Connect standard I/O to the null device.
  104. int fd;
  105. // stdin.
  106. fd = open("/dev/null", O_RDONLY);
  107. if (fd < 0) {
  108. return(EXIT_FAILURE);
  109. }
  110. // stdout.
  111. fd = open("/dev/null", O_WRONLY);
  112. if (fd < 0) {
  113. return(EXIT_FAILURE);
  114. }
  115. // stderr.
  116. fd = open("/dev/null", O_WRONLY);
  117. if (fd < 0) {
  118. return(EXIT_FAILURE);
  119. }
  120. // Start the thread to process messages.
  121. std::thread messageThread(&Service::processMessages, this);
  122. // Run the service.
  123. int status;
  124. status = run();
  125. messageThread.join();
  126. return(status);
  127. }
  128. void Service::loadArguments(int argc, char *argv[]) {
  129. for (int i = 0; i < argc; i++) {
  130. cmdLineArgs.push_back(argv[i]);
  131. }
  132. }
  133. const std::vector<std::string> &Service::arguments() {
  134. return cmdLineArgs;
  135. }
  136. void Service::onStopCall(Callback *stopFunctor) {
  137. stopCallbacks.push_back(stopFunctor);
  138. }
  139. bool Service::receivedStop() {
  140. return (Message::Stop == lastMessage);
  141. }
  142. void Service::processMessages() {
  143. int sigNum;
  144. int status;
  145. while (true) {
  146. // Wait for a signal.
  147. status = sigwait(&signalSet, &sigNum);
  148. if (0 != status) {
  149. perror("Error from sigwait");
  150. // TODO: Implement recovery.
  151. ;
  152. }
  153. // Process the message.
  154. switch (sigNum) {
  155. case SIGTSTP:
  156. lastMessage = Message::Pause;
  157. break;
  158. case SIGCONT:
  159. lastMessage = Message::Resume;
  160. break;
  161. case SIGHUP:
  162. lastMessage = Message::Restart;
  163. break;
  164. case SIGTERM:
  165. lastMessage = Message::Stop;
  166. for (auto callback : stopCallbacks) {
  167. (*callback)();
  168. }
  169. // Exit.
  170. return;
  171. break;
  172. default:
  173. ;
  174. } // switch.
  175. } // while.
  176. }
  177. }