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

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

16.03.2012, 09:44. Показов 3603. Ответов 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
Ответ Создать тему
Новые блоги и статьи
SDL3 для Web (WebAssembly): Идентификация объектов на Box2D v3 - использование userData и событий коллизий
8Observer8 02.03.2026
Содержание блога Финальная демка в браузере. Итоговый код: finish-collision-events-sdl3-c. zip https:/ / www. cyberforum. ru/ blog_attachment. php?attachmentid=11680&amp;d=1772460536 Одним из. . .
Реалии
Hrethgir 01.03.2026
Нет, я не закончил до сих пор симулятор. Эта задача сложнее. Не получилось уйти в плавсостав, но оно и к лучшему, возможно. Точнее получалось - но сварщиком в палубную команду, а это значит, в моём. . .
Ритм жизни
kumehtar 27.02.2026
Иногда приходится жить в ритме, где дел становится всё больше, а вовлечения в происходящее — всё меньше. Плотный график не даёт вниманию закрепиться ни на одном событии. Утро начинается с быстрых,. . .
SDL3 для Web (WebAssembly): Сборка библиотек: SDL3, Box2D, FreeType, SDL3_ttf, SDL3_mixer и SDL3_image из исходников с помощью CMake и Emscripten
8Observer8 27.02.2026
Недавно вышла версия 3. 4. 2 библиотеки SDL3. На странице официальной релиза доступны исходники, готовые DLL (для x86, x64, arm64), а также библиотеки для разработки под Android, MinGW и Visual Studio. . . .
SDL3 для Web (WebAssembly): Реализация движения на Box2D v3 - трение и коллизии с повёрнутыми стенами
8Observer8 20.02.2026
Содержание блога Box2D позволяет легко создать главного героя, который не проходит сквозь стены и перемещается с заданным трением о препятствия, которые можно располагать под углом, как верхнее. . .
Конвертировать закладки radiotray-ng в m3u-плейлист
damix 19.02.2026
Это можно сделать скриптом для PowerShell. Использование . \СonvertRadiotrayToM3U. ps1 <path_to_bookmarks. json> Рядом с файлом bookmarks. json появится файл bookmarks. m3u с результатом. # Check if. . .
Семь CDC на одном интерфейсе: 5 U[S]ARTов, 1 CAN и 1 SSI
Eddy_Em 18.02.2026
Постепенно допиливаю свою "многоинтерфейсную плату". Выглядит вот так: https:/ / www. cyberforum. ru/ blog_attachment. php?attachmentid=11617&stc=1&d=1771445347 Основана на STM32F303RBT6. На борту пять. . .
Камера Toupcam IUA500KMA
Eddy_Em 12.02.2026
Т. к. у всяких "хикроботов" слишком уж мелкий пиксель, для подсмотра в ESPriF они вообще плохо годятся: уже 14 величину можно рассмотреть еле-еле лишь на экспозициях под 3 секунды (а то и больше),. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru