Чтение онлайн

ЖАНРЫ

Разработка приложений в среде Linux. Второе издание

Троан Эрик В.

Шрифт:

10.7.1. Запуск внешних программ с помощью ladsh

Вот первая (и самая простая) версия

ladsh
, называемая
ladsh1
.

1: /*ladsh1.c*/

2:

3: #include <ctype.h>

4: #include <errno.h>

5: #include <fcntl.h>

6: #include <signal.h>

7: #include <stdio.h>

8: #include <stdlib.h>

9: #include <string.h>

 10: #include <sys/ioctl.h>

 11: #include <sys/wait.h>

 12: #include <unistd.h>

 13:

 14: #define MAX_COMMAND_LEN 250 /*
максимальная длина отдельной

 15: командной строки */

 16: #define JOB_STATUS_FORMAT "[%d]%-22s%.40s\n"

 17:

 18: struct jobSet {

 19: struct job *head; /* заголовок списка запущенных заданий */

 20: struct job *fg; /* текущее задание переднего плана */

 21: };

 22:

 23: struct childProgram {

 24: pid_t Pid; /* 0 на выходе */

 25: char **argv; /* имя программы с аргументами */

 26: };

 27:

 28: struct job {

 29: int job Id; /* номер задания */

 30: int numProgs; /* общее кол-во программ в задании */

 31: int runningProgs; /* кол-во работающих программ */

 32: char *text; /* имя задания */

 33: char *cmdBuf; /* буфер различных argv */

 34: pid_t pgrp; /* идентификатор группы процессов задания */

 35: struct childProgram *progs; /* массив программ в задании */

 36: struct job *next; /* для слежения за фоновыми программами */

 37: };

 38:

 39: void freeJob(struct job *cmd) {

 40: int i;

 41:

 42: for (i=0; i<cmd->numProgs; i++) {

 43: free (cmd->progs[i].argv);

 44: }

 45: free(cmd->progs);

 46: if (cmd->text) free(cmd->text);

 47: free(cmd->cmdBuf);

 48: }

 49:

 50: int getCommand(FILE *source, char *command) {

 51: if (source == stdin) {

 52: printf("#");

 53: fflush(stdout);

 54: }

 55:

 56: if (!fgets(command, MAX_COMMAND_LEN, source)) {

 57: if (source==stdin) printf("\n");

 58: return 1;

 59: }

 60:

 61: /*
удалить завершающий перевод строки */

 62: command[strlen(command) - 1] = '\0';

 63:

 64: return 0;

 65: }

 66:

 67: /* Возвратить cmd->numProgs как 0, если нет никаких команд (то есть пустая

 68: строка). Если найдена правильная команда, commandPtr устанавливается в

 69: указатель на начало следующей команды (если исходная команда имеет более

 70: одного задания, ассоциированного с ней) или NULL, если

 71: больше нет команд.*/

 72: int parseCommand(char **commandPtr, struct job *job, int *isBg) {

 73: char *command;

 74: char *returnCommand = NULL;

 75: char *src, *buf;

 76: int argc = 0;

 77: int done = 0;

 78: int argvAlloced;

 79: char quote = '\0';

 80: int count;

 81: struct childProgram *prog;

 82:

 83: /* Пропустить ведущие пробелы */

 84: while(**commandPtr && isspace(**commandPtr)) (*commandPtr)++;

 85:

 86: /* здесь обрабатываются пустые строки и ведущие символы '#' */

 87: if (!**commandPtr || (**commandPtr=='#')) {

 88: job->numProgs = 0;

 89: *commandPtr = NULL;

 90: return 0;

 91: }

 92:

 93: *isBg = 0;

 94: job->numProgs = 1;

 95: job->progs = malloc(sizeof(*job->progs));

 96:

 97: /* Мы устанавливаем элементы argv в указатели внутри строки.

 98: Память освобождается freeJob.

 99:

100: Получение чистой памяти позволяет далее иметь дело с

101: NULL-завершающимися вещами и делает все остальное немного

102: яснее (к тому же, это добавляет эффективности) */

103: job->cmdBuf = command = calloc(1, strlen(*commandPtr) + 1);

104: job->text = NULL;

105:

106: prog = job->progs;

107:

108: argvAlloced = 5;

109: prog->argv = malloc(sizeof(*prog->argv) * argvAlloced);

Поделиться с друзьями: