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

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

20.11.2010, 05:37. Показов 18529. Ответов 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
Ответ Создать тему
Новые блоги и статьи
Thinkpad X220 Tablet — это лучший бюджетный ноутбук для учёбы, точка.
Programma_Boinc 23.12.2025
Thinkpad X220 Tablet — это лучший бюджетный ноутбук для учёбы, точка. Рецензия / Мнение Это мой обзор планшета X220 с точки зрения школьника. Недавно я решила попытаться уменьшить свой. . .
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Как объединить две одинаковые БД Access с разными данными
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
Ломающие изменения в C#.NStar Alpha
Etyuhibosecyu 20.11.2025
Уже можно не только тестировать, но и пользоваться C#. NStar - писать оконные приложения, содержащие надписи, кнопки, текстовые поля и даже изображения, например, моя игра "Три в ряд" написана на этом. . .
Мысли в слух
kumehtar 18.11.2025
Кстати, совсем недавно имел разговор на тему медитаций с людьми. И обнаружил, что они вообще не понимают что такое медитация и зачем она нужна. Самые базовые вещи. Для них это - когда просто люди. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru