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

ЖАНРЫ

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

Троан Эрик В.

Шрифт:

430: exit(0);

431: } else if (!strcmp(newJob.progs[0].argv[0], "pwd")) {

432: len = 50;

433: buf = malloc(len);

434: while (!getcwd(buf, len) && errno == ERANGE) {

435: len += 50;

436: buf = realloc(buf, len);

437: }

438: printf("%s\n", buf);

439: free(buf);

440: return 0;

441: } else if (!strcmp(newJob.progs[0].argv[0], "cd")) {

442: if (!newJob.progs[0].argv[1] == 1)

443: newdir == getenv("HOME");

444: else

445: newdir = newJob.progs[0].argv[1];

446: if (chdir(newdir))

447: printf("сбой
при смене текущего каталога: %s\n",

448: strerror(errno));

449: return 0;

450: } else if (!strcmp(newJob.progs[0].argv[0], "jobs")) {

451: for (job = jobList->head; job; job = job->next) {

452: if (job->runningProgs == job->stoppedProgs)

453: statusString = "Остановлено";

454: else

455: statusString = "Выполняется";

456:

457: printf(JOB_STATUS_FORMAT, job->jobId, statusString,

458: job->text);

459: }

460: return 0;

461: } else if (!strcmp(newJob.progs[0].argv[0], "fg") ||

462: !strcmp(newJob.progs[0].argv[0], "bg")) {

463: if (!newJob.progs[0].argv[1] || newJob.progs[0].argv[2]) {

464: fprintf(stderr,

465: "%s: ожидался в точности один аргумент\n",

466: newJob.progs[0].argv[0]);

467: return 1;

468: }

469:

470: if (sscanf(newJob.progs[0].argv[1], "%%%d", &jobNum) != 1) {

471: fprintf(stderr, "%s: неверный аргумент '%s'\n",

472: newJob.progs[0].argv[0],

473: newJob.progs[0].argv[1]);

474: return 1;

475: }

476:

477: for (job = jobList->head; job; job = job->next)

478: if (job->jobId == jobNum) break;

479:

480: if (!job) {

481: fprintf(stderr, "%s: неизвестное задание %d\n",

482: newJob.progs[0].argv[0], jobNum);

483: return 1;

484: }

485:

486: if (*new Job. progs[0].argv[0] == 'f') {

487: /* Переводим задание на передний план */

488:

489: if (tcsetpgrp(0, job->pgrp))

490: perror("tcsetpgrp");

491: jobList->fg = job;

492: }

493:

494: /*
Повторяем запуск процессов в задании */

495: for (i = 0; i<job->numProgs; i++)

496: job->progs[i].isStopped = 0;

497:

498: kill(-job->pgrp, SIGCONT);

499:

500: job->stoppedProgs = 0;

501:

502: return 0;

503: }

504:

505: nextin = 0, nextout = 1;

506: for (i = 0; i < newJob.numProgs; i++) {

507: if ((i + 1) < newJob.numProgs) {

508: pipe(pipefds);

509: nextout = pipefds[1];

510: } else {

511: nextout = 1;

512: }

513:

514: pipe(controlfds);

515:

516: if (!(newJob.progs[i].pid = fork)) {

517: signal(SIGTTOU, SIG_DFL);

518:

519: close(controlfds[1]);

520: /* при чтении будет возвращен 0, когда записывающая сторона закрыта */

521: read(controlfds[0], &len, 1);

522: close(controlfds[0]);

523:

524: if (nextin != 0) {

525: dup2(nextin, 0);

526: close(nextin);

527: }

528:

529: if (nextout != 1) {

530: dup2(nextout, 1);

531: close(nextout);

532: }

533:

534: /* явные переадресации подменяют каналы */

535: setupRedirections(newJob.progs + i);

536:

537: execvp(newJob.progs[i].argv[0], newJob.progs[i].argv);

538: fprintf(stderr, "сбой exec для %s: %s\n",

539: newJob.progs[i].argv[0],

540: strerror(errno));

541: exit(1);

542: }

543:

544: /* помещаем дочерний процесс в группу процессов, лидером в которой

545: является первый процесс в этом канале */

546: setpgid(newJob.progs[i].pid, newJob.progs[0].pid);

547:

548: /* закрываем канал управления, чтобы продолжить работу дочернего процесса */

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