как немедленно разбудить демона, отправив ему сигнал SIGUSR1 - PullRequest
0 голосов
/ 15 апреля 2019

Я написал программу deamon, которая копирует файлы из одной папки в другую. Мне нужно реализовать SIGUSR1, который немедленно пробуждает демона, посылая ему сигнал SIGUSR1. Я не знаю, что я сделал не так, я использую команду kill -SIGUSR1, может быть, неправильную команду? Кто-то знает, что не так с этим кодом? У меня не было никаких предупреждений после компиляции этой программы, но просто ничего не произошло

 #include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
#include <syslog.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <dirent.h>
#include <fcntl.h>
#include <signal.h>

#define _XOPEN_SOURCE ;

int recursion = 0; //1 if enabled, otherwise 0
int sleepTime = 300;
int fileLimit = 0;
int signaL = 0;
int exitSignal = 0;
int buffer = 1000;

//Returns 0 if arguments are correct otherwise returns 1
int readArguments(int number, char **argv, char *source, char *goal);
int checkFileType(struct stat file);
int copy(char *source, char *target, mode_t mask);
int copy_map(char *source, char *target, struct stat *Source);
void syncCopy(char *source, char *target);
void syncRemove(char *source, char *target);


void my_handler(int sig)
{
    syslog(LOG_INFO, "Daemon received signal SIGUSR1\n");
    signaL = 1;
}

void exitFunction(int sig)
{
    syslog(LOG_INFO, "Daemon received signal SIGUSR2\n");
    exitSignal = 1;
}
int main(int argc, char **argv)
{
    //char tables for paths
    char source[500], goal[500];
    struct stat Source, Goal;
    struct sigaction my_action, old_action;

    //checking and reading arguments
    if (readArguments(argc, argv, source, goal) == 1)
        exit(-1);

    //checking paths

    //checking  if argv[1] and argv[2] are existing paths
    if (lstat(source, &Source) != 0 || lstat(goal, &Goal) != 0) //bad result
    {
        printf("One of the paths or both dont exist\n");
        exit(-1);
    }

    if (checkFileType(Source) != 0)
    {
        printf("Source path is not path to folder");
        exit(-1);
    }

    if (checkFileType(Goal) != 0)
    {
        printf("Goal path is not path to folder");
        exit(-1);
    }

    //forking the parent process
    pid_t pid;
    // Fork off the parent process  and create new
    pid = fork();
    //if failure
    if (pid < 0)
    {
        exit(-1);
    }
    // if it is native process
    else if (pid > 0)
    {
        return 0;
    }
    //if pid==0 then it is childs process

    //now we have to umask in order to write to any files(for exmaple logs)

    umask(0);
    openlog("logFile", LOG_PID, LOG_DAEMON);
    syslog(LOG_INFO, "Deamon has just started running\n");

    pid_t sid = setsid();
    if (sid < 0)
    {
        syslog(LOG_ERR, "Error with session opening\n");
        exit(-1);
    }

    //SIGNAL SIGUSR1
    my_action.sa_handler = my_handler;
    sigfillset(&my_action.sa_mask);
    my_action.sa_flags = 0;
    if (sigaction(SIGUSR1, &my_action, &old_action) < 0)
    {
        syslog(LOG_ERR, "Error with the use of  SIGUSR1 signal\n");
        exit(-1);
    }

    //SIGNAL SIGUSR2 for exiting daemon
    my_action.sa_handler = exitFunction;
    sigfillset(&my_action.sa_mask);
    my_action.sa_flags = 0;
    if (sigaction(SIGUSR2, &my_action, &old_action) < 0)
    {
        syslog(LOG_ERR, "Error with the use of  SIGUSR2 signal\n");
        exit(-1);
    }

    while (!exitSignal)
    {
        sleep(sleepTime);
        switch (signaL)
        {
        case 0:
            syslog(LOG_INFO, "Demon started working after %ds\n", sleepTime);
            break;

        case 1:
        {
            syslog(LOG_INFO, "Demon started working after SIGUSR1 signal\n");
            signaL = 0; //Need to reeset signaL
            break;
        }
        }

        syncCopy(source, goal);
        syncRemove(source, goal);
        syslog(LOG_INFO, "Demon has just  gone to sleep");
    }

    //at the end of program we need to close log using
    syslog(LOG_INFO, "Demon has stopped\n");
    closelog();

    return 0;
}

1 Ответ

0 голосов
/ 15 апреля 2019

Используйте команду как kill -10 <pid> для SIGUSR1 и kill -12 <pid> для SIGUSR2.

 kill -l // command to know the signal number.

Также сделайте переменную signaL, exitSignal как volatile sig_atomic_t тип.

ПОЧЕМУ volatile?

, когда глобальная переменная, обновляемая в обработчике сигналов, периодически проверяется в какой-либо другой функции на предмет соответствующих действий, мы всегда должны объявлять ихиспользование атрибута volatile, чтобы не допустить выполнения компилятором оптимизаций, которые приводят к сохранению переменной в регистре.В худшем случае обновленное значение переменной (обновленное в контексте обработчика) не будет отображаться при опросе функции для переменной.

WHY sig_atomic_t?

Чтение и запись глобальных переменных может включать более одной инструкции на машинном языке, и обработчик сигнала может прерывать основную программу в середине такой последовательности команд.(Мы говорим, что доступ к переменной неатомичен.) По этой причине стандарты языка C и SUSv3 определяют целочисленный тип данных sig_atomic_t, для которого чтение и запись гарантированно будут атомарными.Таким образом, переменная глобального флага, которая используется совместно основной программой и обработчиком сигнала, должна быть объявлена ​​следующим образом:

volatile sig_atomic_t signaL;

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...