Форум программистов, компьютерный форум, киберфорум
C/С++ под Linux
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.56/18: Рейтинг темы: голосов - 18, средняя оценка - 4.56
27 / 25 / 5
Регистрация: 22.04.2010
Сообщений: 772

Fork не создал дочерний процесс

16.03.2012, 09:44. Показов 3570. Ответов 11
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Доброго дня!
Зашел в тупик, переписываю кусок кода...

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
pid_t wpid=0;
wpid=fork();
switch(wpid)
{
    case(-1):
    { 
         ....
         //Error
         goto exit;
    }
    case (0):
    {
         //Child process
        //запуск приложения  через execv()
        break;
    }
    default:
    {
         if(wpid)
         {
               switch(d=waitpid(wpid,&status,WNAHANG))
               {
                       case(-1):{goto exit;}
                       case(0): {break;}
                       default:
                       {
                              goto exit;
                       }
               }
         }
         else
         { 
                printf("Application is not running\n");
          }
          break;
    }
}
wpid всегда больше нуля. т.е. это пид родительского процесса похоже...

1. wpid нулем не становится... т.е. не создается дочерний процесс...
2. попытался разобраться от чего это происходит, т.е. посмотреть переменную errno...
поставил дважды
C
1
printf("Errno: %s\n",strerrno(errno));
перед fork() и после...
надпись одна и тажа:
Errno: Cannot assign requested address...

Приложение очень маленькое... т.е. с памятью не должно быть никаких проблем...
0
Лучшие ответы (1)
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
16.03.2012, 09:44
Ответы с готовыми решениями:

Передача информации в дочерний процесс
Имеется следующая ситуация: после старта программа с помощью fork создает дочерний процесс, который занимается прослушиванием udp...

Приостановить на 1 секунду дочерний процесс
Помогите написать код, похожая программа внизу. *** Приостановить на 1 секунду дочерний процесс. Выполнить в дочернем процессе один из...

Передать параметр в дочерний процесс
Создаю дочерний процесс . В него нужно передать число , которое водится в родительском процессе . Вот код : main.c ...

11
1259 / 650 / 44
Регистрация: 06.02.2011
Сообщений: 1,654
16.03.2012, 10:12
Цитата Сообщение от sitesv Посмотреть сообщение
wpid всегда больше нуля. т.е. это пид родительского процесса похоже...
1. wpid нулем не становится... т.е. не создается дочерний процесс...
Так не бывает. либо -1, либо ребенок создан.
Попробуйте выложить кусок кода на котором можно воспроизвести ситуацию.
0
27 / 25 / 5
Регистрация: 22.04.2010
Сообщений: 772
16.03.2012, 10:40  [ТС]
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
static int execute_program(int argc, char *argv[], pid_t *pid)
{
    int r;
    pid_t wpid = 0;
    int d, status;
    wpid = fork();
    sleep(1);
    switch (wpid)
    {
        case (-1):
        {
            // Parent: error to create child process
            perror("Error: fork()");
            goto exit;
        }
        case (0):
        {
            // Child process
            printf("application PID is %d\n", getpid());
            printf("monitor PID is %d\n", getppid());
            if (chmod(argv[0], S_IRUSR | S_IWUSR | S_IXUSR) < 0)
            {
                perror("Error: chmod()");
                printf("argv[0] is %s\n", argv[0]);
                goto exit;
            }
            // Execute program
            if (execv(argv[0], argv) < 0)
            {
                perror("Error: exec()");
                printf("argv[0] is %s\n", argv[0]);
                printf("argv[1] is %s\n", argv[1]);
                printf("argv[2] is %s\n", argv[2]);
                goto exit;
            }
            break;
        }
        default:
        {
            // Parent process
            sleep(1);
            if (wpid)
            {
                switch (d = waitpid(wpid, &status, WNOHANG))
                {
                    case -1:
                        perror("Error: waitpid()");
                        goto exit;
 
                    case 0:
                        printf("application is running\n");
                        break;
 
                    default:
                        printf("application is not running\n");
                        goto exit;
                }
            }
            else
                printf("application is not running\n");
            break;
        }
    }
    *pid = wpid;
 
exit:
    return (r);
}
И что удивительно... перед вызовом этой функции я работаю с сокетами... ну стандартный вызов функций socket, bind, listen...
Bind() не срабатывает с первого раза, я делаю while, пока bind не вернет значение =0.
Перед первым вызовом bind Errno возвращает Success, после первого вызова: Cannot assign requested address, НО и после того, как bind сработал, и мы вывалились из while, errno все же вернуло Cannot assign requested address. Что за чудеса такие?
0
1259 / 650 / 44
Регистрация: 06.02.2011
Сообщений: 1,654
16.03.2012, 11:33
За исключением не понятных sleep(), "сахарных" скобок, отсутствия _exit() в случае ошибки execv() и пожалуй (тут Вам видней почему так) waitpid() кода оно должно работать. Покрайней мере форкаться точно будет...
Bind() не срабатывает с первого раза, я делаю while, пока bind не вернет значение =0.
Перед первым вызовом bind Errno возвращает Success, после первого вызова: Cannot assign requested address, НО и после того, как bind сработал, и мы вывалились из while, errno все же вернуло Cannot assign requested address. Что за чудеса такие?
bind() в цикле это мягко говоря жжесть man setsockopt про SO_REUSEADDR
errno - Если функция успешно выполнена, то errno _не_ изменяется. man любой стандарт языка C.
0
27 / 25 / 5
Регистрация: 22.04.2010
Сообщений: 772
16.03.2012, 12:05  [ТС]
g_u_e_s_t, "exit:" присутствует в конце фукнции, sleep вреда не несет.
Что за сахарные скобки?
У меня в отладчике вываливается все время у switch по default варианту.
После waitpid тоже попадает в default.

Там я делаю проверку состояния процесса с пидом: проверяю WIFEXITED(status), WIFSIGNALED(status)... последнее срабатывает... т.е. процесс убит сигналом WTERMSIG(status).

почему так...

т.к. значение errno не изменилось, то fork() получается выполнился успешно...
0
1259 / 650 / 44
Регистрация: 06.02.2011
Сообщений: 1,654
16.03.2012, 12:25
Цитата Сообщение от sitesv Посмотреть сообщение
g_u_e_s_t, "exit:" присутствует в конце фукнции
В коде, который Вы показали стоит break и соответственно return(), нужен именно _exit() - ведь ребенка который не смог сделать execv() надо убить.

Цитата Сообщение от sitesv Посмотреть сообщение
Что за сахарные скобки?
Не нужные языковые конструкции типа switch (){case 0: {...}...}
Цитата Сообщение от sitesv Посмотреть сообщение
sleep вреда не несет.
Не согласен, но не суть.

Цитата Сообщение от sitesv Посмотреть сообщение
У меня в отладчике вываливается все время у switch по default варианту.
После waitpid тоже попадает в default.
Отладчик (покрайней мере gdb)не следит за ребенком если его не заставлять делать это.

Цитата Сообщение от sitesv Посмотреть сообщение
После waitpid тоже попадает в default.
А запускаемый код должен работать долго??? Честно говоря глядя на Ваш sleep() и WNOHANG я решил, что предполагается завершение в течении 1й секунды. Кто, кого, почему и зачем убил сигналом - мне не видно

Цитата Сообщение от sitesv Посмотреть сообщение
к. значение errno не изменилось, то fork() получается выполнился успешно...
Не так. fork() выполнился успешно потому, что вернул не -1.
0
27 / 25 / 5
Регистрация: 22.04.2010
Сообщений: 772
16.03.2012, 13:20  [ТС]
g_u_e_s_t, спасибо за Ваши ответы!
но проблему еще не решил...

В выложенном коде не указал, но в switch(...waitpid...) в варианте default я вызываю функцию проверки состояния процесса:

C
1
2
3
4
5
6
7
8
9
10
11
12
static void display_program_status(pid_t pid, int status)
{
if (pid!=0)
{
   if (WIFEXITED(status))
      printf("status= %d\n",WEXITSTATUS(status));
   else if (WIFSIGNALED(status))
      printf("killed by signal= %d\n",WTERMSIG(status));
   else
      printf("unknown status\n");
}
}
Вот printf("killed by signal= %d\n",WTERMSIG(status)); и срабатывает...

То что запускается через execv должно работать долго...
Этот кусок кода является так сказать монитором загрузки стороннего приложения...
0
1259 / 650 / 44
Регистрация: 06.02.2011
Сообщений: 1,654
16.03.2012, 13:35
Цитата Сообщение от sitesv Посмотреть сообщение
В выложенном коде не указал, но в switch(...waitpid...) в варианте default я вызываю функцию проверки состояния процесса:
Приведенный кусок не на что не влияет...

Смотрите сами/Показывайте сюда что именно и как Вы запускаете.
Я не проверял но уверен что:
C
1
2
char *args={"./foo.sh", "1", "2", NULL};
execute_program(0, args, &pid);
с вменяемым содержимым foo.sh отработает нормально.
0
27 / 25 / 5
Регистрация: 22.04.2010
Сообщений: 772
16.03.2012, 14:15  [ТС]
g_u_e_s_t,
до запуска стороннего приложения процесс не доходит... он видимо раньше уничтожается... по каким еще неясным причинам. главная задача child'a - запустить программку, которую я через сокет переслал...
программно перед запуском этой программки делается chmod...
вручную попытался запустить через консоль эту программку, но chmod над ней не выполнялся, т.е. child не выполнился...

Проделам все тоже самое через консоль - программка запускается...

Что заметил! :

после команды wpid = fork()
wpid = 2901
написал в консоле ps aux
высветился в списке этот pid
параметры процесса:
PID=2901
CPU=0.0%
MEM=0%
VSZ = 0
RSS=0
TTY=pts/2
STAT=Z+
START=08:58
TIME=0:00
COMMAND=[mon] <defunct>


очень интересно что за [mon] <defunct>
и почему Z+!

и через некоторое время этот процесс пропадает... все хорошо, но он не выполнил ничего из своего предназначения...

перед chmod сделал создание пустого файла... после запуска fork() ничего не создалось...
0
1259 / 650 / 44
Регистрация: 06.02.2011
Сообщений: 1,654
16.03.2012, 16:18
Лучший ответ Сообщение было отмечено Памирыч как решение

Решение

Вы сами гадаете на кофейной гуще и пытаетесь заставить это делать меня...
Хорошо давайте гадать
Например в execute_program() попадает оригинальный (т.е. с которыми была запущенна программа) argv.
Если не угадал, то может выложите куда-нибудь весь код?

Добавлено через 1 час 48 минут
Ваш код с пастебина с небольшими правками, работает...
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
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <fcntl.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
 
static int execute_program(int, char *[], pid_t *);
static void display_program_status(pid_t, int);
 
int
main(void)
{
        char *mass_names[5]={"myapp_cpp.elf",'\0',"00000","00000", NULL}; //массив объявленных аргументов
        int mass_argc = sizeof(mass_names);
        pid_t pid;
 
        execute_program(mass_argc,mass_names,&pid);  //Из программы "Монитор" я дергаю эту функцию для запуска присланного приложения
        return (0);
}
 
/* Я не понимаю, что по Вашей задумке должна возвращать эта ф-я. по этому не правил. */
static int execute_program(int argc, char *argv[], pid_t *pid)
{
        int r;
        pid_t wpid;
        int d, status;
 
        // Set execute bit for a program
        if (chmod(argv[0], S_IRUSR | S_IWUSR | S_IXUSR) < 0)
        {
                perror("Error: chmod()");
                printf("argv[0] is %s\n", argv[0]);
                goto exit;
        }
        switch ((wpid = fork()))
        {
                case (-1):
                        // Parent: error to create child process
                        perror("Error: fork()");
                        goto exit;
                case (0):
                        // Child process
                        printf("application PID is %d\n", getpid());
                        printf("monitor PID is %d\n", getppid());
 
                        // Execute program
                        if (execv(argv[0], argv) < 0)
                        {
                                perror("Error: exec()");
                                printf("argv[0] is %s\n", argv[0]);
                                printf("argv[1] is %s\n", argv[1]);
                                printf("argv[2] is %s\n", argv[2]);
                                _exit(1);
                        }
                default:
                        // Parent process
                        sleep(1);
                        /*
                         * Для чего тут NOHANG???
                         * Всего кода я не видел, но в таком виде это бессмысленно...
                         */
                        switch (d = waitpid(wpid, &status, WNOHANG))
                        {
                                case -1:
                                        perror("Error: waitpid()");
                                        goto exit;
 
                                case 0:
                                        printf("application is running\n");
                                        break;
 
                                default:
                                        printf("HCX DAQ application is not running\n");
                                        display_program_status(wpid, status);
                                        goto exit;
                        }
        }
        *pid = wpid;
 
exit:
        return (r);
}
 
static void display_program_status(pid_t pid, int status)
{
        if (pid != 0)
        {
                printf("Process %d: ", pid);
                if (WIFEXITED(status))
                        printf("exited, status = %d\n", WEXITSTATUS(status));
                else if (WIFSIGNALED(status))
                        printf("killed by signal %d\n", WTERMSIG(status));
                else
                        printf("unknown status %d\n", status);
        }
}
0
27 / 25 / 5
Регистрация: 22.04.2010
Сообщений: 772
16.03.2012, 16:27  [ТС]
отходил по работе)
опередили меня) хотел отправить вам готовый вариант...
сейчас буду пробовать Ваш)

но в своем случае, случилось что один раз данный вариант запустился, а все остальные - нет...
1
27 / 25 / 5
Регистрация: 22.04.2010
Сообщений: 772
19.03.2012, 09:24  [ТС]
Нашел в чем у меня была проблема.
Вычитал следующее:
"If you have set a breakpoint in any code which the child then executes, the child will get a SIGTRAP signal which (unless it catches the signal) will cause it to terminate."

http://www.delorie.com/gnu/docs/gdb/gdb_26.html

_g_u_e_s_t_, ОГРОМНОЕ СПАСИБО!!
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
19.03.2012, 09:24
Помогаю со студенческими работами здесь

Наследует ли дочерний процесс стандартный ввод и вывод?
Объясните, пожалуйста один момент. имеем следующий кусок кода. pid = fork(); if (pid == 0) { ...

Дочерний процесс os.fork() и функция input()
Я только начал осваивать python и заранее извиняюсь, если мой вопрос покажется кому то глупым. Я поставил себе цель написать некоторую...

Запущен ли процесс? или Завершен ли дочерний процесс моей программы?
Моя программа выполняет execute('cmd', '/C ...'). На момент execute-а других экземпляров cmd.exe нет. Мне хочется узнать, когда окошко cmd...

Родительский и дочерний процесс: процесс не переходит обратно к родителю
Здравствуйте знаю что тема уже была, но все же! создаю элементарную программу, пока просто 1 дочерний процесс, но после запуска родителя...

Есть процесс A и дочерний процесс B, который находит хэндл родительского процесса A. И этот хэндл в разных случаях РАЗНЫЙ! Почему?
Друзья! Вот код A.exe, суть которого просто стать родительским для B.exe и висеть в системе. #include &lt;stdio.h&gt; #include...


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

Или воспользуйтесь поиском по форуму:
12
Ответ Создать тему
Новые блоги и статьи
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