LINUX.ORG.RU

Ограничение времени выполнения процесса


0

0

Добрый всем день.

У меня возникла такая задача:

Несоздать из одного процесса запустить второй с ограничением по времени выполнения второго.

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

Не могли мне подсказать, каким образом я могу организовать указанное ожидание? А еще было бы очень здорово, если бы я мог указывать время, на протяжении которого второй процесс находился в рабочем состоянии.

Извините, опечатка: Необходимо из одного процесса запустить второй с ограничением по времени выполнения второго.

smokeless
() автор топика
Ответ на: комментарий от smokeless

Как сделать "Как только созданный процесс завершился" я не знаю, но можно вот как (псевдокод):

pid = fork();
if( pid == 0 )
do child job
else if( pid > 0 )
do
++i
sleep(1)
wpid = waitpid()
if( wpid = pid )
chid is terminated, exit
endif
while( i < 10 )
kill( pid, SIGTERM )
endif

man fork, man waitpid, man sleep, man kill

yz
()

Устанавливаешь обработчик SIGCHLD со снятым флагом SA_RESTART, после fork'а делаешь sleep.
Если получил SIGCHLD, то sleep прервётся с ошибкой EINTR и выполнение пойдёт дальше.
Если sleep завершился успешно (т.е. потомок завис), то убиваешь потомка и идёшь дальше.

execve
()
Ответ на: комментарий от smokeless

pid_t pid;
int status;

if((pid = fork()) == 0)
{
   /* мы в дочернем процессе */
}
else
{
   sleep(сколько_нужно);
   if((pid = waitpid(-1, &status, WNOHANG)) == 0)
   {
       if(kill(pid, 9) == -1)
       {
          /* Обрабатываем  errno */
       }
   }
}

BreadFan ★★
()

$ apt-get source timeout
$ cat timeout.c

/*++
/* NAME
/*	timeout 1
/* SUMMARY
/*	run command with bounded time
/* SYNOPSIS
/*	\fBtimeout\fR [-\fIsignal\fR] \fItime\fR \fIcommand\fR ...
/* DESCRIPTION
/*	\fBtimeout\fR executes a command and imposes an elapsed time limit.
/*	The command is run in a separate POSIX process group so that the
/*	right thing happens with commands that spawn child processes.
/*
/*	Arguments:
/* .IP \fI-signal\fR
/*	Specify an optional signal to send to the controlled process.
/*	By default, \fBtimeout\fR sends SIGKILL, which cannot be caught
/*	or ignored.
/* .IP \fItime\fR
/*	The elapsed time limit after which the command is terminated.
/* .IP \fIcommand\fR
/*	The command to be executed.
/* DIAGNOSTICS
/*	The command exit status is the exit status of the command
/*	(status 1 in case of a usage error).
/* AUTHOR(S)
/*	Wietse Venema
/*	This program is part of SATAN.
/*--*/

/* System libraries. */

#include <sys/types.h>
#include <signal.h>
#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>

extern int optind;

/* Application-specific. */

#define perrorexit(s) { perror(s); exit(1); }

static int kill_signal = SIGKILL;
static char *progname;
static char *commandname;

static void usage()
{
    fprintf(stderr, "usage: %s [-signal] time command...\n", progname);
    exit(1);
}

static void terminate(sig)
int     sig;
{
    signal(kill_signal, SIG_DFL);
    fprintf(stderr, "Timeout: aborting command ``%s'' with signal %d\n",
	    commandname, kill_signal);
    kill(0, kill_signal);
}

int     main(argc, argv)
int     argc;
char  **argv;
{
    int     time_to_run;
    pid_t   pid;
    pid_t   child_pid;
    int     status;

    progname = argv[0];

    /*
     * Parse JCL.
     */
    while (--argc && *++argv && **argv == '-')
	if ((kill_signal = atoi(*argv + 1)) <= 0)
	    usage();

    if (argc < 2 || (time_to_run = atoi(argv[0])) <= 0)
	usage();

    commandname = argv[1];

    /*
     * Run the command and its watchdog in a separate process group so that
     * both can be killed off with one signal.
     */
    setsid();
    switch (child_pid = fork()) {
    case -1:					/* error */
	perrorexit("timeout: fork");
    case 00:					/* run controlled command */
	execvp(argv[1], argv + 1);
	perrorexit(argv[1]);
    default:					/* become watchdog */
	(void) signal(SIGHUP, terminate);
	(void) signal(SIGINT, terminate);
	(void) signal(SIGQUIT, terminate);
	(void) signal(SIGTERM, terminate);
	(void) signal(SIGALRM, terminate);
	alarm(time_to_run);
	while ((pid = wait(&status)) != -1 && pid != child_pid)
	     /* void */ ;
	return (pid == child_pid ? status : -1);
    }
}

sdio ★★★★★
()

Исходники timeout тебе в помощь:

zoidberg:~$ apt-cache show timeout
Package: timeout
Priority: optional
Section: admin
Installed-Size: 52
Maintainer: Andrew Stribblehill <ads@debian.org>
Architecture: i386
Source: tct
Version: 1.11-6.1
Replaces: tct (<= 1.07-2), netatalk (<< 1.5pre6-7)
Depends: libc6 (>= 2.3.2.ds1-4)
Conflicts: tct (<= 1.07-2), netatalk (<< 1.5pre6-7)
Filename: pool/main/t/tct/timeout_1.11-6.1_i386.deb
Size: 6230
MD5sum: 63b0f65009f46ef24c52842c64504f39
Description: Run a command with a time limit.
 timeout executes a command and imposes an elapsed time limit. When the
 time limit is reached, timeout sends a predefined signal to the target
 process.

Shmuma
()
Вы не можете добавлять комментарии в эту тему. Тема перемещена в архив.