mirror of
https://github.com/pygos/init.git
synced 2024-11-22 19:19:47 +01:00
David Oberhollenzer
028394b8a5
This commit add the ability to initd to reload the service configuration while running. The new configuration is merged with the existing one as follows: For each target: - If the existing service list is not NULL, we have not started that target yet. Simply replace it with the new list. - If it is NULL, the services have already been started. - First, remove all entries for services in that target that no loner exist (except from the 'running' list). - Second, add new services that we don't have yet. Treat them as recently diseased and let the user start them manualy. Signed-off-by: David Oberhollenzer <goliath@infraroot.at>
147 lines
2.6 KiB
C
147 lines
2.6 KiB
C
/* SPDX-License-Identifier: ISC */
|
|
#include "init.h"
|
|
|
|
static int sigfd = -1;
|
|
static int sockfd = -1;
|
|
|
|
static void handle_signal(void)
|
|
{
|
|
struct signalfd_siginfo info;
|
|
int status;
|
|
pid_t pid;
|
|
|
|
if (read(sigfd, &info, sizeof(info)) != sizeof(info)) {
|
|
perror("read on signal fd");
|
|
return;
|
|
}
|
|
|
|
switch (info.ssi_signo) {
|
|
case SIGCHLD:
|
|
while ((pid = waitpid(-1, &status, WNOHANG)) > 0) {
|
|
status = WIFEXITED(status) ? WEXITSTATUS(status) :
|
|
EXIT_FAILURE;
|
|
|
|
supervisor_handle_exited(pid, status);
|
|
}
|
|
break;
|
|
case SIGTERM:
|
|
supervisor_set_target(TGT_SHUTDOWN);
|
|
break;
|
|
case SIGINT:
|
|
supervisor_set_target(TGT_REBOOT);
|
|
break;
|
|
case SIGHUP:
|
|
supervisor_reload_config();
|
|
break;
|
|
case SIGUSR1:
|
|
if (sockfd >= 0) {
|
|
close(sockfd);
|
|
unlink(INIT_SOCK_PATH);
|
|
sockfd = -1;
|
|
}
|
|
sockfd = init_socket_create();
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void handle_request(void)
|
|
{
|
|
struct sockaddr_un addr;
|
|
init_request_t rq;
|
|
socklen_t addrlen;
|
|
ssize_t ret;
|
|
retry:
|
|
memset(&rq, 0, sizeof(rq));
|
|
addrlen = sizeof(addr);
|
|
ret = recvfrom(sockfd, &rq, sizeof(rq), MSG_DONTWAIT | MSG_TRUNC,
|
|
(struct sockaddr *)&addr, &addrlen);
|
|
|
|
if (ret < 0 && errno == EINTR)
|
|
goto retry;
|
|
|
|
if ((size_t)ret < sizeof(rq))
|
|
return;
|
|
|
|
switch (rq.rq) {
|
|
case EIR_STATUS:
|
|
supervisor_answer_status_request(sockfd, &addr, addrlen,
|
|
rq.arg.status.filter);
|
|
break;
|
|
case EIR_START:
|
|
rq.arg.startstop.id = be32toh(rq.arg.startstop.id);
|
|
supervisor_start(rq.arg.startstop.id);
|
|
break;
|
|
case EIR_STOP:
|
|
rq.arg.startstop.id = be32toh(rq.arg.startstop.id);
|
|
supervisor_stop(rq.arg.startstop.id);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void target_completed(int target)
|
|
{
|
|
switch (target) {
|
|
case TGT_BOOT:
|
|
if (sockfd < 0)
|
|
sockfd = init_socket_create();
|
|
break;
|
|
case TGT_SHUTDOWN:
|
|
for (;;)
|
|
reboot(RB_POWER_OFF);
|
|
break;
|
|
case TGT_REBOOT:
|
|
for (;;)
|
|
reboot(RB_AUTOBOOT);
|
|
break;
|
|
}
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
int i, ret, count;
|
|
struct pollfd pfd[2];
|
|
|
|
if (getpid() != 1) {
|
|
fputs("init does not have pid 1, terminating!\n", stderr);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
supervisor_init();
|
|
|
|
sigfd = sigsetup();
|
|
if (sigfd < 0)
|
|
return -1;
|
|
|
|
for (;;) {
|
|
while (supervisor_process_queues())
|
|
;
|
|
|
|
memset(pfd, 0, sizeof(pfd));
|
|
count = 0;
|
|
|
|
pfd[count].fd = sigfd;
|
|
pfd[count].events = POLLIN;
|
|
++count;
|
|
|
|
if (sockfd >= 0) {
|
|
pfd[count].fd = sockfd;
|
|
pfd[count].events = POLLIN;
|
|
++count;
|
|
}
|
|
|
|
ret = poll(pfd, count, -1);
|
|
if (ret <= 0)
|
|
continue;
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
if (pfd[i].revents & POLLIN) {
|
|
if (pfd[i].fd == sigfd)
|
|
handle_signal();
|
|
if (pfd[i].fd == sockfd)
|
|
handle_request();
|
|
}
|
|
}
|
|
}
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|