Форум программистов, компьютерный форум, киберфорум
C/С++ под Linux
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.81/91: Рейтинг темы: голосов - 91, средняя оценка - 4.81
 Аватар для javasc
6 / 5 / 2
Регистрация: 08.11.2010
Сообщений: 192

Нужно создать демон....как?

20.11.2010, 05:37. Показов 18596. Ответов 8
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
задача.

Создать процесс работающий от рута, постоянно работающий (это демон будет).
И создать exec файл, который при запуске НЕ от рута будет обращаться к процессу запущенному от рута и передавать этому процессу число (да просто число).

Такое возможно?
0
Лучшие ответы (1)
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
20.11.2010, 05:37
Ответы с готовыми решениями:

Демон под BSD для вычислений
Здравстуйте, обитатели форума. прошу помощи в реализации такой практически возникшей передо мной задачи: Дано: - сервер...

Демон, который логирует любые изменения в папке
Хочу создать демона который будет логировать любые изменения в заданной директории Проблема в том что я не знаю какими функциями можно...

демон для отлова RTP(вопрос о выборе библиотек и всего сопутствующего)
Вопрос конечно наверно офтоп, но так как ОС - линь, напишу сюда Нужно повешать демона на порт, который будет принимать/ожидать RTP,...

8
Эксперт С++
 Аватар для niXman
3211 / 1459 / 74
Регистрация: 09.08.2009
Сообщений: 3,441
Записей в блоге: 2
20.11.2010, 14:12
Лучший ответ Сообщение было отмечено как решение

Решение

1. Нужно вызвать функцию umask(), чтобы сбросить маску режима создания файла в 0. Маска режима создания файлов, наследуемая от запускающего процесса, может маскировать некоторые права доступа. Если предполагается, что процесс-демон будет создавать файлы, может потребоваться установка определенных битов прав доступа. Например, если демон создает файлы с правом на чтение и на запись для группы, маска режима создания файла, которая выключает любой из этих битов, воспрепятствовала бы этому.
2. Вызвать функцию fork() и завершить родительский процесс. Для чего это делается? Во-первых, если демон был запущен как обычная команда оболочки, то завершив родительский процесс, мы заставим командную оболочку думать, что команда была выполнена. Во-вторых, дочерний процесс, наследует идентификатор группы процессов от родителя, но получает свой идентификатор процесса; таким образом мы гарантируем, что дочерний процесс не будет являться лидером группы, а это необходимое условие для вызова функции setsid(), который будет произведен далее.
3. Создать новую сессию, обратившись к функции setsid(). При этом процесс становится (a) лидером новой сессии, (b) лидером новой группы процессов и (c) лишается управляющего терминала.
4. Сделать корневой каталог текущим рабочим каталогом. Текущий рабочий каталог унаследованный от родительского процесса, может находится на смонтированной файловой системе. Поскольку демон, как правило, существует все время, пока система не будет перезагружена, то в подобной ситуации, когда рабочий каталог демона находится на смонтированной файловой системе, ее невозможно будет отмонтировать.
5. Закрыть все ненужные файловые дескрипторы. Это предотвращает удержание в открытом состоянии некоторых дескрипторов, унаследованных от родительского процесса. С помощью функции getrlimit() можно определить максимально возможный номер дескриптора и закрыть все, вплоть до этого номера.
6. Некоторые демоны открывают файловые дескрипторы с номерами 0, 1, 2 на устройстве "/dev/null", таким образом любые библиотечные и пользовательские функции, которые пытаются читать из стандартного ввода или писать на стандартный вывод или вывод для ошибок, не будут оказывать никакого влияния.

Собственно, код шаблона:

main.h

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
#ifndef MAIN_H_FILE
#define MAIN_H_FILE
 
#include <unistd.h>
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <syslog.h>
#include <fcntl.h>
#include <signal.h>
#include <sys/stat.h>
#include <sys/resource.h>
 
 
/***************************************************************************/
 
struct process_signal_args {
    void(*reread_func_t)(const void*);
    void* reread_arg;
    void(*terminate_func_t)(const void*);
    void* terminate_arg;
    /**  */
    sigset_t* mask;
    volatile int running;
};
 
/***************************************************************************/
 
void demonize(const char* cmd) {
    pid_t pid;
    struct rlimit limits;
    struct sigaction sa;
 
    umask(0);
 
    if ( getrlimit(RLIMIT_NOFILE, &limits) < 0 ) {
        perror("can`t get RLIMIT_NOFILE");
        exit(EXIT_FAILURE);
    }
 
    if ( (pid = fork()) < 0 ) {
        perror("fork error");
        exit(EXIT_FAILURE);
    } else if ( 0 != pid ) {
        exit(0);
    }
 
    if ( (setsid()) == (pid_t)-1 ) {
        perror("setsig error");
        exit(EXIT_FAILURE);
    }
 
    sa.sa_handler = SIG_IGN;
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = 0;
    if ( sigaction(SIGHUP, &sa, NULL) < 0 ) {
        perror("can`t ignore SIGHUP");
        exit(EXIT_FAILURE);
    }
 
    if ( (pid = fork()) < 0 ) {
        perror("fork error");
        exit(EXIT_FAILURE);
    } else if ( 0 != pid ) {
        exit(0);
    }
 
    if ( chdir("/") < 0 ) {
        perror("can`t chdir() to /");
        exit(EXIT_FAILURE);
    }
 
    if ( limits.rlim_max == RLIM_INFINITY )
        limits.rlim_max = 1024;
 
    u_int32_t idx;
    for ( idx = 0; idx < limits.rlim_max; ++idx ) {
        close(idx);
    }
 
    int fd0 = open("/dev/null", O_RDWR);
    int fd1 = dup(0);
    int fd2 = dup(0);
 
    openlog(cmd, LOG_CONS, LOG_DAEMON);
    if ( fd0 != 0 || fd1 != 1 || fd2 != 2 ) {
        syslog(LOG_ERR, "bad file descriptors: %d, %d, %d", fd0, fd1, fd2);
        exit(EXIT_FAILURE);
    }
}
 
/***************************************************************************/
 
int already_running(const char* cmd) {
    char buff[4096] = "\0";
    sprintf(buff, "%s%s", "/var/run/", cmd);
    /**  */
    int fd = open(buff, O_RDWR|O_CREAT, S_IRUSR|S_IWUSR|S_IRGRP|S_IROTH);
    if ( fd < 0 ) {
        perror("can`t open PID file");
        exit(EXIT_FAILURE);
    }
    if ( lockf(fd, F_TLOCK, 0) ) {
        if ( errno == EACCES || errno == EAGAIN ) {
            close(fd);
            return 1;
        }
        perror("can`t lock PID file");
        exit(EXIT_FAILURE);
    }
 
    ftruncate(fd, 0);
    sprintf(buff, "%d", (int)getpid());
    write(fd, buff, strlen(buff));
 
    return 0;
}
 
/***************************************************************************/
 
void* process_signal_thr(void* arg) {
    struct process_signal_args* opt = (struct process_signal_args*)arg;
    while ( opt->running ) {
        int signo = 0;
        if ( 0 != sigwait(opt->mask, &signo) ) {
            syslog(LOG_ERR, "error in sigwait: %s", strerror(errno));
            exit(EXIT_FAILURE);
        }
        switch ( signo ) {
            case SIGHUP:
                syslog(LOG_INFO, "reread configuration");
                if ( opt->reread_func_t ) {
                    opt->reread_func_t(opt->reread_arg);
                }
                break;
            case SIGTERM:
                syslog(LOG_INFO, "receive signal SIGTERM. exiting...");
                if ( opt->terminate_func_t ) {
                    opt->terminate_func_t(opt->terminate_arg);
                }
                opt->running = 0;
                break;
            default:
                syslog(LOG_INFO, "unspecified signal '%d' received. ignored.", signo);
                break;
        }
    }
    return NULL;
}
 
/***************************************************************************/
 
#endif // MAIN_H_FILE
main.cpp

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
#include <pthread.h>
#include "main.h"
 
volatile int runflag = 1;
/***************************************************************************/
 
void reread_func(const void* ptr) {
    printf("reread_func() called(%s)", (const char*)ptr);
}
 
void terminate_func(const void* ptr) {
    printf("terminate_func() called(%s)", (const char*)ptr);
    runflag = 0;
}
 
/***************************************************************************/
/** прежняя main() функция */
int main2(int argc, char** argv) {
    int idx = 0;
    for ( ; runflag; ++idx ) {
        syslog(LOG_INFO, "ping %d", idx);
        sleep(1);
    }
    return 0;
}
 
/***************************************************************************/
 
int main(int argc, char** argv) {
    sigset_t mask;
    struct sigaction sa;
    pthread_t tid;
    struct process_signal_args thread_args;
    memset(&thread_args, 0, sizeof(struct process_signal_args));
 
    const char* cmd = (NULL == strrchr(argv[0], '/')) ? argv[0] : strrchr(argv[0], '/')+1;
    demonize(cmd);
 
    if ( already_running(cmd) ) {
        perror("daemon already running");
        exit(EXIT_FAILURE);
    }
 
    sa.sa_handler = SIG_DFL;
    sigemptyset(&sa.sa_mask);
    sa.sa_flags = 0;
    if ( sigaction(SIGHUP, &sa, NULL) < 0 ) {
        perror("sigaction error");
        exit(EXIT_FAILURE);
    }
 
    sigfillset(&mask);
    if ( 0 != pthread_sigmask(SIG_BLOCK, &mask, NULL) ) {
        perror("pthread_sigmask error");
        exit(EXIT_FAILURE);
    }
 
    thread_args.mask = &mask;
    thread_args.running = 1;
    /**  */
    thread_args.reread_func_t = reread_func;
    thread_args.reread_arg = "reread_arg";
    /**  */
    thread_args.terminate_func_t = terminate_func;
    thread_args.terminate_arg = "terminate_arg";
    /**  */
    if ( pthread_create(&tid, NULL, process_signal_thr, &thread_args) ) {
        perror("can`t create thread");
        exit(EXIT_FAILURE);
    }
 
    /* тут код программы */
    syslog(LOG_INFO, "process \"%s\" running...", cmd);
    return main2(argc, argv);
}
 
/***************************************************************************/
отсюда.
8
2 / 2 / 0
Регистрация: 10.05.2012
Сообщений: 45
14.06.2012, 18:30
Это, конечно, полезный пример. Но это исходный код самого демона. А вот клиент? Передавать по сокету? Или какими-то другими методами?
0
Эксперт С++
 Аватар для niXman
3211 / 1459 / 74
Регистрация: 09.08.2009
Сообщений: 3,441
Записей в блоге: 2
14.06.2012, 18:35
Цитата Сообщение от Lucky_spirit Посмотреть сообщение
А вот клиент?
какой клиент?

Цитата Сообщение от Lucky_spirit Посмотреть сообщение
Передавать по сокету?
что передавать?

Цитата Сообщение от Lucky_spirit Посмотреть сообщение
Или какими-то другими методами?
о чем речь?
0
2 / 2 / 0
Регистрация: 10.05.2012
Сообщений: 45
15.06.2012, 11:52
Цитата Сообщение от niXman Посмотреть сообщение
1. Нужно вызвать функцию umask()...отсюда.
Простите, у Вас в этом примере ошибке:
Code
1
2
/home/user/Qt/cyberDaemon/main.cpp:62: ошибка: invalid conversion from 'const void*' to 'void*' [-fpermissive]
/home/user/Qt/cyberDaemon/main.cpp:65: ошибка: invalid conversion from 'const void*' to 'void*' [-fpermissive]
2
Эксперт С++
 Аватар для niXman
3211 / 1459 / 74
Регистрация: 09.08.2009
Сообщений: 3,441
Записей в блоге: 2
15.06.2012, 11:58
спасибо!
1
0 / 0 / 0
Регистрация: 09.08.2012
Сообщений: 14
25.08.2012, 16:39
Цитата Сообщение от niXman Посмотреть сообщение
спасибо!
юниксман не поможешь мне раборатся в процессе создания демона ?? если есть время напиши в скайп gideon799
0
Эксперт С++
 Аватар для niXman
3211 / 1459 / 74
Регистрация: 09.08.2009
Сообщений: 3,441
Записей в блоге: 2
25.08.2012, 16:41
Цитата Сообщение от Gideon Посмотреть сообщение
не поможешь мне раборатся в процессе создания демона ?
зависит от того, что подразумевается под словом "помощь". если "сделай за меня", то только за деньги.

Цитата Сообщение от Gideon Посмотреть сообщение
напиши в скайп
не использую IM месенджеры.
да и не за чем, это же форум. создай тему с интересующим тебя вопросом.
0
310 / 233 / 15
Регистрация: 01.07.2011
Сообщений: 812
Записей в блоге: 1
25.08.2012, 18:48
Цитата Сообщение от javasc Посмотреть сообщение
передавать этому процессу число
это можно разными способами. называются они общим словом IPC (Inter-Process Communication) - межпроцессное взаимодействие. Если "по-тупому" и по-быстрому:
1. Файл:
клиент - создать файл в /tmp
демон - прочесть файл из /tmp

2. Именованный канал (FIFO pipe):
создается с помощью функции mkfifo().
man для bash
Bash
1
man mkfifo
man для Си
Bash
1
man 3 mkfifo
то есть один процесс (клиент) в него пишет,
другой процесс (демон) читает.

3. Через очереди сообщений.
не помню как делается, если очень надо - завтра могу в книге найти. (а вообще можно и гуглануть).

4. Псевдофайловая система tmpfs (типа как proc). Файловая система в оперативной памяти.
Демон - монтирует tmpfs куда-нибудь и периодически читает из него файл.
Клиент - пишет в этот файл.
про tmpfs можно почитать в википедии.

5. Через сокеты.
Демон - создает сокет и слушает.
Клиент - пишет в него число.
Bash
1
man socket
Добавлено через 2 минуты

Не по теме:

Цитата Сообщение от niXman Посмотреть сообщение
создай тему с интересующим тебя вопросом.
Тема: "Никсман, сделай для меня демона, как у тебя на аватаре" :)

0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
25.08.2012, 18:48
Помогаю со студенческими работами здесь

Создать демон, проверяющий факт модификации файлов
Помогите закончить программу под линукс Нужно написать демон под линукс который бы проверял заданий каталог каждые 3 мин. на придет...

Как написать демон
Решила попробывать написать простой демон. Нашла пример, но он не компелируется. Поскажите что делать. #include &lt;sys/types.h&gt; ...

Запустить сокет-сервер как демон
расталкуйте плиз как запустить сокет сервер на сервере? что надо использовать Cron запустить фаил как демона или что ещё??...

Нужен демон. Не знаю как написать
Привет, народ. Я думаю многие сталкивались с потребностью создания фонового процесса? Так вот, написал сценарий синхронизации ЦРМки и...

Как посмотреть и запустить с параметром демон?
Всем доброй ночи друзья! Вопрос может показаться глупым, но я с линуксом на Вы. Есть программа в дебиане, запускается автоматически при...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
9
Ответ Создать тему
Новые блоги и статьи
Отправка уведомления на почту при изменении наименования справочника
Maks 24.03.2026
Программная отправка письма электронной почты на примере изменения наименования справочника "Склады" в конфигурации БП3. Перед реализацией необходимо выполнить настройку системной учетной записи. . .
модель ЗдравоСохранения 5. Меньше увольнений- больше дохода!
anaschu 24.03.2026
Теперь система здравосохранения уменьшает количество увольнений. 9TO2GP2bpX4 a42b81fb172ffc12ca589c7898261ccb/ https:/ / rutube. ru/ video/ a42b81fb172ffc12ca589c7898261ccb/ Слева синяя линия -. . .
Midnight Chicago Blues
kumehtar 24.03.2026
Такой Midnight Chicago Blues, знаешь?. . Когда вечерние улицы становятся ночными, а ты не можешь уснуть. Ты идёшь в любимый старый бар, и бармен наливает тебе виски. Ты смотришь на пролетающие. . .
Контроль уникальности заводского номера - вариант №2
Maks 24.03.2026
В отличие от предыдущего варианта добавлено прерывание циклов, также добавлены новые переменные для сохранения контекста ошибки перед прерыванием цикла: Процедура ПередЗаписью(Отказ, РежимЗаписи,. . .
SDL3 для Desktop (MinGW): Вывод текста со шрифтом TTF с помощью библиотеки SDL3_ttf на Си и C++
8Observer8 24.03.2026
Содержание блога Финальные проекты на Си и на C++: finish-text-sdl3-c. zip finish-text-sdl3-cpp. zip
Жизнь в неопределённости
kumehtar 23.03.2026
Жизнь — это постоянное существование в неопределённости. Например, даже если у тебя есть список дел, невозможно дойти до точки, где всё окончательно завершено и больше ничего не осталось. В принципе,. . .
Модель здравоСохранения: работники работают быстрее после её введения.
anaschu 23.03.2026
geJalZw1fLo Корпорация до введения программа здравоохранения имела много невыполненных работниками заданий, после введения программы количество заданий выросло. Но на выплатах по больничным это. . .
Контроль уникальности заводского номера - вариант №1
Maks 23.03.2026
Алгоритм контроля уникальности заводского (или серийного) номера на примере документа выдачи шин для спецтехники с табличной частью в конфигурации КА2. Данные берутся из регистра сведений, по. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru