6 프로세스 생성과 실행.

Slides:



Advertisements
Similar presentations
12장 프로세스.
Advertisements

ㅎㅎ 구조체 구조체 사용하기 함수 매개변수로서의 구조체 구조체 포인터와 레퍼런스 구조체 배열.
2장. 프로그램의 기본 구성. 2장. 프로그램의 기본 구성 2-1"Hello, World!" 들여다 보기 /* Hello.c */ #include int main(void) { printf("Hello, World! \n"); return 0;
6 레이스 컨디션.
인공지능실험실 석사 2학기 이희재 TCP/IP Socket Programming… 제 11장 프로세스간 통신 인공지능실험실 석사 2학기 이희재
Linux/UNIX Programming APUE (Process Control)
Chapter 6. 리눅스 백도어 황 선 환.
8장 프로세스.
9 파이프.
인공지능실험실 석사 2학기 김승겸 TCP/IP Socket Programming… 제 10장 멀티태스킹 기반의 서버구현 인공지능실험실 석사 2학기 김승겸
Signal & Inter-Process Communication
공유 메모리[1] 공유 메모리 공유 메모리 생성: shmget(2) 같은 메모리 공간을 두 개 이상의 프로세스가 공유하는 것
Department of Computer Engineering
Department of Computer Science and Engineering
6 프로세스 생성과 실행.
조 병 규 Software Quality Lab. 한국교통대학교
제15장 파일 입출력 문자열을 출력하는 여러가지 방법 (15-2쪽) 문자열만 처리하는 입출력 함수
제 6장. 생성자와 소멸자 학기 프로그래밍언어및실습 (C++).
UNIT 09 프로세스 제어 로봇 SW 교육원 3기.
Department of Computer Engineering
Homework #6 (1/3) 다음을 수행한 후, 화면(혹은 파일)을 출력하여 제출한다.
양방향 파이프의 활용 양방향 통신 파이프는 기본적으로 단방향이므로 양방향 통신을 위해서는 파이프를 2개 생성한다.
12장 파이프.
fork로 생성한 자식 프로세스에서 exec 함수군을 호출
Linux System Programming
Chapter 06 프로세스와 예약작업 관리 Solaris 1. 프로세스 관리
11장. 포인터 01_ 포인터의 기본 02_ 포인터와 Const.
12장 프로세스.
8장 함수 함수의 필요성 라이브러리 함수와 사용자 정의 함수 함수의 정의, 원형, 호출 배열을 함수 인자로 전달 재귀호출.
링크 파일 생성[1] 링크 하드링크 생성 : link(2) 이미 있는 파일이나 디렉토리에 접근할 수 있는 새로운 이름
Signal & Inter-Process Communication
Term Project Team Member
Department of Computer Engineering
프로세스 생성[1] 프로그램 실행 : system(3) #include <stdlib.h>
컴퓨터 프로그래밍 기초 #02 : printf(), scanf()
파이프와 exec 시스템 호출 (ls | wc)
Homework 6… 12월 2일(금) 11:59pm까지 자신의 이름과 학번을 출력해 주는 유닉스/리눅스 네트워크 소켓 서버 프로그램 과 클라이언트 프로그램 을 작성해 보세요 참고 (실습1) Hello 프로그램 helloserver.c helloclient.c 컴파일.
8 메모리 매핑.
파일 기술자 파일 기술자 현재 열려있는 파일을 구분하는 정수값 저수준 파일 입출력에서 열린 파일을 참조하는데 사용
메시지 큐[5] – test1.c 메시지 제어: msgctl(2) #include <sys/msg.h>
파일 접근권한 제어 stat 구조체의 st_mode 항목에 파일의 종류와 접근권한 정보저장 st_mode 값의 구조.
14장. 포인터와 함수에 대한 이해.
5 프로세스 정보.
Method & library.
사용자 함수 사용하기 함수 함수 정의 프로그램에서 특정한 기능을 수행하도록 만든 하나의 단위 작업
Signal & Inter-Process Communication
3장 상수 변수 기본 자료형 키워드와 식별자 상수와 변수 기본 자료형 형변환 자료형의 재정의.
Homework #6 (1/3) 다음을 수행한 후, 화면(혹은 파일)을 출력하여 제출한다.
Device Driver 임베디드 시스템 I.
24장. 파일 입출력.
쉽게 풀어쓴 C언어 Express 제14장 포인터 활용 C Express Slide 1 (of 22)
19. 함수 포인터와 void 포인터.
Homework #6 (1/3) 다음을 수행한 후, 화면(혹은 파일)을 출력하여 제출한다.
제어문 & 반복문 C스터디 2주차.
School of Electronics and Information. Kyung Hee University.
컴퓨터 프로그래밍 기초 - 10th : 포인터 및 구조체 -
Department of Computer Engineering
5장 선택제어문 if 선택문 switch-case 선택문 다양한 프로그램 작성 조건 연산자.
Fflush 사용이유 및 방법 [이유] 키보드에서 입력된 내용은 입력버퍼에 저장되었다가 Enter 키가 들어오면 프로그램으로 전달됨 이 때 입력버퍼에 있는 Enter 키도 프로그램으로 전달됨 그러므로 아래와 같은 프로그램에서 문자 하나를 입력해도 Enter키도 입력된 것으로.
Signal & Inter-Process Communication
6장 반복제어문 for 문 while 문 do while 문 기타 제어문.
Fucntion 요약.
Department of Computer Engineering
( Windows Service Application Debugging )
3. 모듈 (5장. 모듈).
실습과제 1번 생성된 파일 basic.txt를 프로젝트 폴더에서 메모장으로 열고 내용을 확인
9 파이프.
Signal & Inter-Process Communication
수업 내용 수업 목표 강의 내용 강의 계획서 교과서 및 참고도서 평가 방법 수강생의 학습 방법 제안 강의자료 사이트
Signal & Inter-Process Communication
Presentation transcript:

6 프로세스 생성과 실행

학습목표 프로세스를 생성하는 방법을 이해한다. 프로세스를 종료하는 방법을 이해한다. exec함수군으로 새로운 프로그램을 실행하는 방법을 이해한다. 프로세스를 동기화하는 방법을 이해한다.

목차 프로세스 생성 프로세스 종료함수 exec 함수군 활용 exec 함수군과 fork 함수 프로세스 동기화

프로세스 생성 (1) 프로그램 실행 : system(3) 새로운 프로그램을 실행하는 가장 간단한 방법이나 비효율적 실행할 프로그램명을 인자로 지정 #include <stdlib.h> int system(const char *string); 01 #include <stdlib.h> 02 #include <stdio.h> 03 04 int main(void) { 05 int a; 06 a = system("ps -ef | grep han > han.txt"); 07 printf("Return Value : %d\n", a); 08 09 return 0; 10 } [예제 6-1] system 함수 사용하기 ex6_1.c # ex6_1.out Return Value : 0 # cat han.txt root 736 735 0 10:31:02 pts/3 0:00 grep han root 735 734 0 10:31:02 pts/3 0:00 sh -c ps -ef | grep han> han.txt

프로세스 생성 (2) 프로세스 생성: fork(2) 새로운 프로세스를 생성 생성된 프로세스 : 자식 프로세스 fork 함수를 호출한 프로세스 : 부모 프로세스 자식 프로세스는 부모 프로세스의 메모리를 복사하여 생성 (자기 복제) RUID, EUID, RGID, EGID, 환경변수 열린 파일기술자, 시그널 처리, setuid, setgid 현재 작업 디렉토리, umask, 사용가능자원 제한 #include <sys/types.h> #include <unistd.h> pid_t fork(void);

프로세스 생성 (3) 프로세스 생성: fork(2) 자식 프로세스와 부모 프로세스와 차이점 자식 프로세스는 유일한 PID를 갖는다 자식 프로세스는 유일한 PPID를 갖는다. 부모 프로세스가 설정한 프로세스잠금, 파일 잠금, 기타 메모리 잠금은 상속 안함 자식 프로세스의 tms구조체 값은 0으로 설정 부모 프로세스와 자식 프로세스는 열린 파일을 공유하므로 읽거나 쓸 때 주의해야 한다.

fork함수의 리턴값 0은 자식 프로세스가 실행 ex6_2.c ... 06 int main(void) { 07 pid_t pid; 08 09 switch (pid = fork()) { 10 case -1 : /* fork failed */ 11 perror("fork"); 12 exit(1); 13 break; 14 case 0 : /* child process */ 15 printf("Child Process - My PID:%d, My Parent's PID:%d\n", 16 (int)getpid(), (int)getppid()); 17 break; 18 default : /* parent process */ 19 printf("Parent process - My PID:%d, My Parent's PID:%d, " "My Child's PID:%d\n", (int)getpid(), (int)getppid(), (int)pid); 21 break; 22 } 23 24 printf("End of fork\n"); 26 return 0; 27 } fork함수의 리턴값 0은 자식 프로세스가 실행 # ex6_2.out Child Process - My PID:796, My Parent's PID:795 End of fork Parent process - My PID:795, My Parent's PID:695, My Child's PID:796

프로세스 종료 함수 (1) 프로그램 종료: exit(2) 프로그램 종료시 수행할 작업 예약: atexit(2) status : 종료 상태값 프로그램 종료시 수행할 작업 예약: atexit(2) func : 종료시 수행할 작업을 지정한 함수명 프로그램 종료: _exit(2) 일반적으로 프로그램에서 직접 사용하지 않고 exit 함수 내부적으로 호출 #include <stdlib.h> void exit(int status); #include <stdlib.h> int atexit(void (*func)(void)); #include <unistd.h> void _exit(int status);

프로세스 종료 함수 (2) 프로그램 종료 함수의 일반적 종료 절차 모든 파일 기술자를 닫는다. 부모 프로세스에 종료 상태를 알린다. 자식 프로세스들에 SIGHUP 시그널을 보낸다. 부모 프로세스에 SIGCHLD 시그널을 보낸다. 프로세스간 통신에 사용한 자원을 반납한다.

[예제 6-3] exit, atexit 함수 사용하기 ex6_3.c 01 #include <stdlib.h> 02 #include <stdio.h> 03 04 void cleanup1(void) { 05 printf("Cleanup 1 is called.\n"); 06 } 07 08 void cleanup2(void) { 09 printf("Cleanup 2 is called.\n"); 10 } 11 12 int main(void) { 13 atexit(cleanup1); 14 atexit(cleanup2); 15 16 exit(0); 17 } 종료시 수행할 함수 지정 지정한 순서의 역순으로 실행(실행결과 확인) # ex6_3.out Cleanup 2 is called. Cleanup 1 is called.

exec 함수군 활용 exec 함수군 exec 함수군의 형태 6가지 path : 명령의 경로 지정 file : 실행 파일명 지정 arg#, argv : main 함수에 전달할 인자 지정 envp : main 함수에 전달할 환경변수 지정 함수의 형태에 따라 NULL 값 지정에 주의해야 한다. #include <unistd.h> int execl(const char *path, const char *arg0, ..., const char *argn,(char *)0); int execv(const char *path, char *const argv[]); int execle(const char *path, const char *arg0, ..., const char *argn, (char *)0, char *const envp[]); int execve(const char *path, char *const argv[], char *const envp[]); int execlp(const char *file, const char *arg0, ..., const char *argn, (char *)0); int execvp(const char *file, char *const argv[]);

[예제 6-4] execlp 함수 사용하기 01 #include <unistd.h> ex6_4.c 01 #include <unistd.h> 02 #include <stdlib.h> 03 #include <stdio.h> 04 05 int main(void) { 06 printf("--> Before exec function\n"); 07 08 if (execlp("ls", "ls", "-a", (char *)NULL) == -1) { 09 perror("execlp"); 10 exit(1); 11 } 12 13 printf("--> After exec function\n"); 14 15 return 0; 16 } 인자의 끝을 표시하는 NULL 포인터 첫 인자는 관례적으로 실행파일명 지정 메모리 이미지가 ‘ls’ 명령으로 바뀌어 13행은 실행안됨 # ex6_4.out --> Before exec function . ex6_1.c ex6_3.c ex6_4.out .. ex6_2.c ex6_4.c han.txt

[예제 6-5] execv 함수 사용하기 01 #include <unistd.h> ex6_5.c 01 #include <unistd.h> 02 #include <stdlib.h> 03 #include <stdio.h> 04 05 int main(void) { 06 char *argv[3]; 07 08 printf("Before exec function\n"); 09 10 argv[0] = "ls"; 11 argv[1] = "-a"; 12 argv[2] = NULL; 13 if (execv("/usr/bin/ls", argv) == -1) { 14 perror("execv"); 15 exit(1); 16 } 17 18 printf("After exec function\n"); 19 20 return 0; 21 } 첫 인자는 관례적으로 실행파일명 지정 인자의 끝을 표시하는 NULL 포인터 경로로 명령 지정 역시 실행안 됨 # ex6_5.out --> Before exec function . ex6_1.c ex6_3.c ex6_5.c han.txt .. ex6_2.c ex6_4.c ex6_5.out

[예제 6-6] execve 함수 사용하기 ... 05 int main(void) { 06 char *argv[3]; ex6_6.c ... 05 int main(void) { 06 char *argv[3]; 07 char *envp[2]; 08 09 printf("Before exec function\n"); 10 11 argv[0] = "arg.out"; 12 argv[1] = "100"; 13 argv[2] = NULL; 14 15 envp[0] = "MYENV=hanbit"; 16 envp[1] = NULL; 17 18 if (execve("./arg.out", argv, envp) == -1) { 19 perror("execve"); 20 exit(1); 21 } 22 23 printf("After exec function\n"); 24 25 return 0; 26 } 실행파일명 지정 인자의 끝을 표시하는 NULL 포인터 환경변수 설정 ex6_6_arg.c를 컴파일하여 생성 # ex6_6.out --> Before exec function --> In ex6_6_arg.c Main argc = 2 argv[0] = arg.out argv[1] = 100 MYENV=hanbit

[예제 6-6] (2) ex6_6_arg.c 파일 01 #include <stdio.h> 02 03 int main(int argc, char **argv, char **envp) { 04 int n; 05 char **env; 06 07 printf("\n--> In ex6_6_arg.c Main\n"); 08 printf("argc = %d\n", argc); 09 for (n = 0; n < argc; n++) 10 printf("argv[%d] = %s\n", n, argv[n]); 11 12 env = envp; 13 while (*env) { 14 printf("%s\n", *env); 15 env++; 16 } 17 18 return 0; 19 } 인자 값 출력 환경변수 출력

exec 함수군과 fork 함수 유닉스 운영체제에서의 새로운 프로그램 실행 fork로 생성한 자식 프로세스에서 exec 함수군을 호출 자식 프로세스의 메모리 이미지가 부모 프로세스 이미지에서 exec 함수로 호출한 새로운 명령으로 대체 자식 프로세스는 부모 프로세스와 다른 프로그램 실행 가능 부모 프로세스와 자식 프로세스가 각기 다른 작업을 수행해야 할 때 fork와 exec 함수를 함께 사용

[예제 6-7] fork와 exec 함수 사용하기 ex6_7.c ... 06 int main(void) { 07 pid_t pid; 08 09 switch (pid = fork()) { 10 case -1 : /* fork failed */ 11 perror("fork"); 12 exit(1); 13 break; 14 case 0 : /* child process */ 15 printf("--> Child Process\n"); 16 if (execlp("ls", "ls", "-a", (char *)NULL) == -1) { 17 perror("execlp"); 18 exit(1); } 20 exit(0); 21 break; 22 default : /* parent process */ 23 printf("--> Parent process - My PID:%d\n“,(int)getpid()); 24 break; 27 return 0; 28 } 자식프로세스에서 execlp 함수 실행 부모프로세스는 이 부분 실행 # ex6_7.out --> Child Process ex6_1.c ex6_3.c ex6_5.c ex6_6_arg.c ex6_7.out ex6_2.c ex6_4.c ex6_6.c ex6_7.c han.txt --> Parent process - My PID:10535

프로세스 동기화 부모 프로세스와 자식 프로세스의 종료절차 좀비 프로세스 고아 프로세스 부모 프로세스와 자식 프로세스는 순서와 상관없이 실행하고 먼저 실행을 마친 프로세스는 종료 부모 프로세스와 자식 프로세스 사이에 종료 절차가 제대로 진행되지 않으면 좀비 프로세스 발생 좀비 프로세스 실행을 종료하고 자원을 반납한 자식 프로세스의 종료 상태를 부모 프로세스가 가져가지 않으면 좀비 프로세스 발생 좀비 프로세스는 프로세스 테이블에만 존재 좀비 프로세스는 일반적인 제거 방법은 없음 방지를 위해 부모 프로세스와 자식 프로세스를 동기화 해야함. 고아 프로세스 자식 프로세스보다 부모 프로세스가 먼저 종료할 경우 자식 프로세스들은 고아 프로세스가 됨 고아 프로세스는 1번 프로세스(init)의 자식 프로세스로 등록

프로세스 동기화 함수 (1) 프로세스 동기화: wait(3) stat_loc : 상태정보를 저장할 주소 부모 프로세스가 wait 함수를 호출하기 전에 자식 프로세스가 종료하면 wait 함수는 즉시 반환 wait 함수의 반환값은 자식 프로세스의 PID 반환값이 -1이면 살아있는 자식 프로세스가 하나도 없다는 의미 #include <sys/types.h> #include <sys/wait.h> pid_t wait(int *stat_loc);

[예제 6-8] wait 함수 사용하기 ... 07 int main(void) { 08 int status; ex6_8.c ... 07 int main(void) { 08 int status; 09 pid_t pid; 11 switch (pid = fork()) { 12 case -1 : /* fork failed */ 13 perror("fork"); 14 exit(1); 15 break; 16 case 0 : /* child process */ 17 printf("--> Child Process\n"); 18 exit(2); 19 break; 20 default : /* parent process */ 21 while (wait(&status) != pid) 22 continue; 23 printf("--> Parent process\n"); 24 printf("Status: %d, %x\n", status, status); 25 printf("Child process Exit Status:%d\n”,status >> 8); 26 break; } 29 return 0; 30 } # ex6_8.out --> Child Process --> Parent process Status: 512, 200 Child process Exit Status:2 자식 프로세스의 종료를 기다림 오른쪽으로 8비트 이동해야 종료 상태값을 알 수 있음

프로세스 동기화 함수[2] 특정 자식 프로세스와 동기화: waitpid(3) pid에 지정할 수 있는 값 -1보다 작은 경우 : pid의 절댓값과 같은 프로세스 그룹ID에 속한 자식 프로세스 중 임의의 프로세스의 상태값 요청 -1인 경우 : wait 함수처럼 임의의 자식 프로세스의 상태값을 요청 0인 경우 : 함수를 호출한 프로세스와 같은 프로세스 그룹에 속한 임의의 프로세스의 상태값 요청 0보다 큰 경우 : 지정한 PID의 상태값 요청 options: waitpid 함수의 반환 조건 WCONTINUED: 수행중인 자식 프로세스의 상태값 반환 WNOHANG: pid로 지정한 자식프로세스의 상태값을 즉시 반환받을 수 없어도 이를 호출한 프로세스의 실행을 블록하지 않고 다른 작업을 수행토록 함 WNOWAIT: 상태값을 반환한 프로세스가 대기 상태에 머물 수 있도록 함 WUNTRACED: 실행을 중단한 자식 프로세스의 상태값을 반환 #include <sys/types.h> #include <sys/wait.h> pid_t waitpid(pid_t pid, int *stat_loc, int options);

[예제 6-9] waitpid 함수 사용하기 # ex6_9.out --> Child process ex6_9.c ... 07 int main(void) { 08 int status; 09 pid_t pid; 10 11 if ((pid = fork()) < 0) { /* fork failed */ 12 perror("fork"); 13 exit(1); 14 } 15 16 if (pid == 0) { /* child process */ 17 printf("--> Child process\n"); 18 sleep(3); 19 exit(3); 20 } 21 22 printf("--> Parent process\n"); 23 24 while (waitpid(pid, &status, WNOHANG) == 0) { printf("Parent still wait...\n"); 26 sleep(1); 27 } 28 29 printf("Child Exit Status : %d\n", status>>8); 30 31 return 0; 32 } # ex6_9.out --> Child process --> Parent process Parent still wait... Child Exit Status : 3 WNOHANG이므로 waitpid 함수는 블록되지 않고 25~26행 반복 실행