Download presentation
Presentation is loading. Please wait.
1
프로세스 생성[1] 프로그램 실행 : system(3) #include <stdlib.h>
새로운 프로그램을 실행하는 가장 간단한 방법이나 비효율적이므로 남용하지 말 것 실행할 프로그램명을 인자로 지정 #include <stdlib.h> int system(const char *string); 01 #include <stdlib.h> 02 #include <stdio.h> 03 04 int main(void) { int a; a = system("ps -ef | grep han > han.txt"); printf("Return Value : %d\n", a); 08 return 0; 10 } [예제 6-1] system 함수 사용하기 (test1.c) # ex6_1.out Return Value : 0 # cat han.txt root :31:02 pts/3 0:00 grep han root :31:02 pts/3 0:00 sh -c ps -ef | grep han> han.txt
2
프로세스 생성[2] 프로세스 생성: fork(2) #include <sys/types.h>
새로운 프로세스를 생성 : 자식 프로세스 fork 함수를 호출한 프로세스 : 부모 프로세스 자식 프로세스는 부모 프로세스의 메모리를 복사 RUID, EUID, RGID, EGID, 환경변수 열린 파일기술자, 시그널 처리, setuid, setgid 현재 작업 디렉토리, umask, 사용가능자원 제한 부모 프로세스와 다른 점 자식 프로세스는 유일한 PID를 갖는다 자식 프로세스는 유일한 PPID를 갖는다. 부모 프로세스가 설정한 프로세스잠금, 파일 잠금, 기타 메모리 잠금은 상속 안함 자식 프로세스의 tms구조체 값은 0으로 설정 부모 프로세스와 자식 프로세스는 열린 파일을 공유하므로 읽거나 쓸 때 주의해야 한다. #include <sys/types.h> #include <unistd.h> pid_t fork(void);
3
[예제 6-2] fork 함수 사용하기 (test2.c)
... 06 int main(void) { pid_t pid; 08 switch (pid = fork()) { case -1 : /* fork failed */ perror("fork"); exit(1); break; case 0 : /* child process */ printf("Child Process - My PID:%d, My Parent's PID:%d\n", (int)getpid(), (int)getppid()); break; default : /* parent process */ printf("Parent process - My PID:%d, My Parent's PID:%d, " "My Child's PID:%d\n", (int)getpid(), (int)getppid(), (int)pid); break; } 23 printf("End of fork\n"); 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
4
프로그램 종료시 수행할 작업 예약: atexit(2)
프로세스 종료 함수[1] 프로그램 종료: exit(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);
5
프로세스 종료 함수[2] 프로그램 종료 함수의 일반적 종료 절차 모든 파일 기술자를 닫는다.
부모 프로세스에 종료 상태를 알린다. 자식 프로세스들에 SIGHUP 시그널을 보낸다. 부모 프로세스에 SIGCHLD 시그널을 보낸다. 프로세스간 통신에 사용한 자원을 반납한다.
6
좀비 프로세스 (zombie process)
참고자료 좀비 프로세스 (zombie process) 부모 프로세스가 wait를 수행하지 않고 있는 상태에서 자식이 종료 자식 프로세스의 종료를 부모 프로세스가 처리해주지 않으면 자식 프로세스는 좀비 프로세스가 된다. 좀비 프로세스는 CPU, Memory 등의 자원을 사용하지 않으나, 커널의 작업 리스트에는 존재한다. 고아 프로세스 (orphan process) 하나 이상의 자식 프로세스가 수행되고 있는 상태에서 부모가 먼저 종료 부모 프로세스가 수행 중인 자식 프로세스를 기다리지 않고 먼저 종료 init 프로세스 좀비와 고아 프로세스의 관리는 결국 시스템의 init 프로세스로 넘겨진다. init 프로세스가 새로운 부모가 된다.
7
실습 부모는 “I’m the parent, child has pid %d”, 자식은 “I’m the child\n”를 출력하는 프로그램을 작성하라 두 개의 부 프로세스를 생성하고, 생성된 각각의 부 프로세스들은 그 자 신의 부 프로세스를 하나씩 또 생성하는 프로그램을 작성하라. 각 fork가 끝난 다음, 각 프로세스들은 자신과 부모프로세스들의 식별번호를 출력 한다 부모는 ‘A’에서 ‘Z’까지 대문자를 자식은 소문자를 표준 출력하는 프로그 램을 작성하라
8
[예제 6-3] exit, atexit 함수 사용하기
ex6_3.c 01 #include <stdlib.h> 02 #include <stdio.h> 03 04 void cleanup1(void) { printf("Cleanup 1 is called.\n"); 06 } 07 08 void cleanup2(void) { printf("Cleanup 2 is called.\n"); 10 } 11 12 int main(void) { atexit(cleanup1); atexit(cleanup2); 15 exit(0); 17 } 종료시 수행할 함수 지정 지정한 순서의 역순으로 실행(실행결과 확인) # ex6_3.out Cleanup 2 is called. Cleanup 1 is called.
9
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, int execvp(const char *file, char *const argv[]);
10
[예제 6-4] execlp 함수 사용하기 (test3.c)
01 #include <unistd.h> 02 #include <stdlib.h> 03 #include <stdio.h> 04 05 int main(void) { printf("--> Before exec function\n"); 07 if (execlp("ls", "ls", "-a", (char *)NULL) == -1) { perror("execlp"); exit(1); } 12 printf("--> After exec function\n"); 14 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
11
[예제 6-5] execv 함수 사용하기(test4.c)
01 #include <unistd.h> 02 #include <stdlib.h> 03 #include <stdio.h> 04 05 int main(void) { char *argv[3]; 07 printf("Before exec function\n"); 09 argv[0] = "ls"; argv[1] = "-a"; argv[2] = NULL; if (execv("/usr/bin/ls", argv) == -1) { perror("execv"); exit(1); } 17 printf("After exec function\n"); 19 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
12
[예제 6-6] execve 함수 사용하기(test5.c)
... 05 int main(void) { char *argv[3]; char *envp[2]; 08 printf("Before exec function\n"); 10 argv[0] = "arg.out"; argv[1] = "100"; argv[2] = NULL; 14 envp[0] = "MYENV=hanbit"; envp[1] = NULL; 17 if (execve("./arg.out", argv, envp) == -1) { perror("execve"); exit(1); } 22 printf("After exec function\n"); 24 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
13
[예제 6-6] (2) arg.c 파일 (arg.out)
01 #include <stdio.h> 02 03 int main(int argc, char **argv, char **envp) { int n; char **env; 06 printf("\n--> In ex6_6_arg.c Main\n"); printf("argc = %d\n", argc); for (n = 0; n < argc; n++) printf("argv[%d] = %s\n", n, argv[n]); 11 env = envp; while (*env) { printf("%s\n", *env); env++; } 17 return 0; 19 } 인자 값 출력 환경변수 출력
14
실습 “반복문을 사용해 1~10000까지 더하면서 결과값을 반복적으로 출력하 는 프로그램을 작성하고, 이 프로그램의 실행시간을 측정하는” 프로그램 을 exec를 사용하여 수행하라
Similar presentations