QNX/UNIX: Анатомия параллелизма
Правила наследования (и ненаследования) параметров дочернего процесса от родителя (RID, RGID и других атрибутов) жестко регламентированы, достаточно сложны (в зависимости от флагов) и могут быть уточнены в технической документации QNX. Отметим, что безусловно наследуются такие параметры, как: а) приоритет и дисциплина диспетчеризации; б) рабочий и корневой каталоги файловой системы. Не наследуются: установки таймеров процесса
,tms_utime,tms_stimeиtms_cutime, значение взведенного сигналаtms_cstime(это значение сбрасывается в ноль), файловые блокировки, блокировки и отображения памяти (shared memory), установленные родителем.SIGALRMПри успешном завершении вызов функции возвращает PID порожденного процесса. При неудаче возвращается -1 и
устанавливается:errno•
— количество байт, заданное в списке аргументов или переменных окружения и превышающееE2BIG;ARG_MAX•
— нет права поиска в каталогах префикса имени файла, или для файла не установлены права на выполнение, или файловая система по указанному пути была смонтирована с флагомEACCESS;ST_NOEXEC•
— недостаточно системных ресурсов для порождения процесса;EAGAIN•
— недопустим хотя бы один из файловых дескрипторов в массивеERADF;fd_map•
— недопустима одна из буферных областей, указанных в вызове;EFAULT•
— слишком глубокий уровень символических ссылок к файлу или глубина префиксов (каталогов) в полном пути к файлу;ELOOP•
— недостаточно ресурсов для отображения файловых дескрипторов в дочерний процесс;EMFILE•
— длина полного пути превышаетENAMETOOLONGили длина компонента имени файла и пути превышаетPATH_MAX;NAME_MAX•
— файл нулевой длины или несуществующий префиксный компонент в полном пути;ENOENT•
— файл, указанный как программа, имеет ошибочный для исполняемого файла формат;ENOEXEC•
— в системе недостаточно свободной памяти для порождения процесса;ENOMEM•
— файловая система, специфицированная полным путевым именем файла, не предназначена для выполненияENOSYS;spawn()•
— префиксные компоненты пути исполняемого файла не являются каталогами;ENOTDIRДаже из этого очень краткого обзора вызова
становятся очевидными некоторые вещи:spawn()• Эта форма универсальна (самодостаточна), она позволяет обеспечить весь спектр разнообразных форм порождения нового процесса
• Она же и самая громоздкая форма, тяжеловесная для практического кодирования, поэтому в реальных текстах в большинстве случаев вы вместо нее встретите ее конкретизации:
,spawnl(),spawnle(),spawnlp(),spawnlpe(),spawnp(),spawnv(),spawnve(),spawnvp(). Все эти формы достаточно полно описаны в [1]. Функционально они эквивалентныspawnvpe(), поэтому мы не станем на них детально останавливаться.spawn()• Хотя вызов
и упоминается в описаниях как POSIX-совместимый, в QNX он существенно расширен и модифицирован и поэтому в лучшем случае может квалифицироваться как «выполненный по мотивам» POSIX.spawn()В качестве примера приведем использованную в [4] (глава Д. Алексеева «Утилита on») форму вызова для запуска программы (с именем, заданным в строке
) на удаленном узлеcommand(например,node) сети QNET (как вы понимаете, это совершенно уникальная возможность QNX, говорить о которой в рамках POSIX-совместимости просто бессмысленно):/net/xxx
int main() {
char* command = "...", *node = "...";
// параметры запуска не используются
char* const argv[] = { NULL };
struct inheritance inh;
inh.flags = 0;
// флаг удаленного запуска
inh.flags |= SPAWN_SETND;
// дескриптор хоста
inh.nd = netmgr_strtond(node, NULL);
pid_t pid = spawnp(command, 0, NULL, &inh, argv, NULL);
...
}Использованная здесь форма
наиболее близка к базовойspawnp()и отличается лишь тем, что для поиска файла программы используется переменная системного окруженияspawn().PATHПриведем характерный пример вызова группы
:exec*()
int execl(const char* path, const char* arg0, const char* arg1, ...
const char* argn, NULL);где
— путевое имя исполняемого файла;path, …,arg0— символьные строки, доступные процессу как список аргументов. Список аргументов должен завершаться значениемargn. АргументNULLдолжен быть именем файла, ассоциированного с запускаемым процессом.arg0ПримечаниеУстоявшаяся терминология «запускаемый процесс» относительно
явно неудачна и лишь вводит в заблуждение. Здесь гораздо уместнее говорить о замещении выполнявшегося до этой точки кода новым, выполнение которого начинается с точки входа главного потока замещающего процесса.exec*()ПримечаниеЕсли вызов
выполняется из многопоточного родительского процесса, то все выполняющиеся потоки этого процесса предварительно завершаются. Никакие функции деструкторов для них не выполняются.exec*()Если вызов
успешен, управление никогда уже не возвращается в точку вызова. В случае неудачи возвращается -1 иexec*()устанавливается так же, как описано выше дляerrno.spawn()