Microsoft MVP성태의 닷넷 이야기
Linux: 108. Linux와 Windows의 프로세스/스레드 ID 관리 방식 [링크 복사], [링크+제목 복사],
조회: 6097
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)
(시리즈 글이 2개 있습니다.)
스크립트: 44. 파이썬의 3가지 스레드 ID
; https://www.sysnet.pe.kr/2/0/13251

Linux: 108. Linux와 Windows의 프로세스/스레드 ID 관리 방식
; https://www.sysnet.pe.kr/2/0/13821




Linux와 Windows의 프로세스/스레드 ID 관리 방식

윈도우 기반의 개발자가 Linux로 이전하면서 혼란을 느끼는 것 중의 하나가 바로 프로세스와 스레드일 것입니다. ^^;

우선, 윈도우는 프로세스와 스레드의 관계가 주종 관계입니다.

EPROCESS_1
   |- 스레드_1 (ETHREAD) 
   |- 스레드_2 (ETHREAD)
   :    ...   (ETHREAD)
   ㄴ 스레드_N (ETHREAD)

EPROCESS_2
   |- 스레드_1 (ETHREAD) 
   |- 스레드_2 (ETHREAD)
   :    ...   (ETHREAD)
   ㄴ 스레드_N (ETHREAD)

따라서, 프로세스의 ID는 EPROCESS 구조체에 담긴 ID 필드 값이고, 스레드의 ID는 ETHREAD 구조체에 담긴 ID 필드 값으로 분리돼 있습니다.

lkd> dt _EPROCESS
nt!_EPROCESS
   +0x000 Pcb              : _KPROCESS
   +0x438 ProcessLock      : _EX_PUSH_LOCK
   +0x440 UniqueProcessId  : Ptr64 Void
   +0x448 ActiveProcessLinks : _LIST_ENTRY
   ...[생략]...
   +0x5e0 ThreadListHead   : _LIST_ENTRY
   ...[생략]...

lkd> dt _ETHREAD
nt!_ETHREAD
   +0x000 Tcb              : _KTHREAD
   +0x430 CreateTime       : _LARGE_INTEGER
   +0x438 ExitTime         : _LARGE_INTEGER
   ...[생략]...
   +0x478 Cid              : _CLIENT_ID
   ...[생략]...

lkd> dt _CLIENT_ID
nt!_CLIENT_ID
   +0x000 UniqueProcess    : Ptr64 Void
   +0x008 UniqueThread     : Ptr64 Void

하지만, 리눅스 세계에서는 프로세스와 스레드의 구분이 없고 모든 것이 "Task"입니다.

Task_1
   |- Task_2
   |- Task_3
   : ...
   ㄴ Task_N

Task_N+1
    |- Task_N+2
    |- Task_N+3
    : ...
    ㄴ Task_N+N

문제는, Task를 표현하는 task_struct 구조체 필드의 이름이 윈도우 세계의 개념과 혼동을 줄 수 있다는 점입니다.

$ grep -A 214 "struct task_struct {" vmlinux.h
struct task_struct {
        ...[생략]...
        pid_t pid;
        pid_t tgid;
        ...[생략]...
        struct key *cached_requested_key;
        char comm[16];
        ...[생략]...
};

// https://www.linkedin.com/pulse/brief-linux-process-amit-nadiger/

pid: the process ID of the process
tgid: the thread group ID of the process

pid를 "process ID"라고 설명하고 있는데요, 저 말을 윈도우 세계의 프로세스/스레드 개념으로 이해하면 안 됩니다. 즉, 리눅스에서 저 2개의 필드는 실은 다음과 같은 의미로 쓰입니다.

pid: task id (윈도우라면 스레드 ID와 유사)
tgid: task group id (윈도우라면 프로세스 ID와 유사)

실제로, 저 개념으로 보면 그나마 윈도우 운영체제의 프로세스/스레드 개념과 비슷하게 이해할 수 있습니다. 가령, task를 하나 생성하면 최초에 pid와 tgid는 동일한 값으로 나옵니다.

[신규 task 생성]
pid == 100
tgid == 100

그리고, 저 task에서 새로운 스레드(task)를 하나 생성하면 pid는 새로운 값으로 변경되지만, tgid는 부모 task의 tgid 값으로 결정됩니다.

[task에서 신규 스레드 생성]
pid == 100
tgid == 100
    pid == 101
    tgid == 100

반면, 스레드가 아닌 자식 프로세스의 개념(예를 들어, fork)으로 task를 생성하는 경우에는 pid는 언제나처럼 새로운 값으로 변경되고, tgid는 다시 그 pid의 값을 따르게 됩니다.

[task에서 신규 프로세스 생성]
pid == 100
tgid == 100
    pid == 101
    tgid == 101

그런데, 위의 필드들과 관련된 함수들을 보면 더욱더 혼란스러운 면이 있습니다.

pid_t getpid (void): task_struct의 tgid 반환 (윈도우라면 프로세스 ID와 유사)

pid_t getppid (void): task_struct의 real_parent의 tgid 반환 (윈도우라면 부모 프로세스 ID와 유사)

pid_t gettid (void): task_struct의 pid 반환 (윈도우라면 스레드 ID와 유사)

보는 바와 같이 getpid가 (pid가 아닌) tgid를 반환하고, gettid가 pid를 반환하는 식입니다. 즉, 함수 측면에서 보면 오히려 윈도우의 프로세스/스레드 개념과 유사하게 이해할 수 있습니다.




참고로, "pstree" 명령어를 사용하면 리눅스 시스템에서 프로세스/스레드의 관계를 트리 구조로 확인할 수 있습니다.

systemd─┬─ModemManager───2*[{ModemManager}]
        ├─NetworkManager───2*[{NetworkManager}]
        ├─accounts-daemon───2*[{accounts-daemon}]
...[생략]...

한 가지 유의해야 할 점은, 위에서 systemd와 ModemManager는 프로세스의 부모/자식 관계인 반면, ModemManager와 "2*[{ModemManager}]"는 같은 프로세스에서의 스레드 부모/자식 관계입니다. 이름으로 그렇게 구분하는 것은 아니고, "{", "}" 중괄호로 묶인 것이 스레드라는 의미이기 때문인데요, 스레드의 경우 앞서 표기된 "2"라는 숫자는 스레드의 개수를 의미합니다. (윈도우의 관점에서 해석하면, ModemManager 프로세스에는 Main 스레드 1개와 secondary 스레드 2개로 총 3개의 스레드가 존재하는 걸로 이해할 수 있습니다.)

마지막으로 리눅스와 윈도우 모두 스레드 ID는 시스템 내에서 유일한 값임을 보증할 수 있습니다. 단지, 리눅스의 경우에는 꾸준히 증가하는 식으로 할당이 되는 반면, 윈도우는 스레드가 종료되면 그 ID가 재사용될 수 있으므로 주의가 필요합니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/18/2024]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 31  [32]  33  34  35  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13173정성태11/27/202214382.NET Framework: 2072. 닷넷 응용 프로그램의 스레드 스택 크기 변경
13172정성태11/25/202214072.NET Framework: 2071. 닷넷에서 ESP/RSP 레지스터 값을 구하는 방법파일 다운로드1
13171정성태11/25/202213337Windows: 214. 윈도우 - 스레드 스택의 "red zone"
13170정성태11/24/202214905Windows: 213. 윈도우 - 싱글 스레드는 컨텍스트 스위칭이 없을까요?
13169정성태11/23/202216344Windows: 212. 윈도우의 Protected Process (Light) 보안 [1]파일 다운로드2
13168정성태11/22/202213359제니퍼 .NET: 31. 제니퍼 닷넷 적용 사례 (9) - DB 서비스에 부하가 걸렸다?!
13167정성태11/21/202214216.NET Framework: 2070. .NET 7 - Console.ReadKey와 리눅스의 터미널 타입
13166정성태11/20/202214493개발 환경 구성: 651. Windows 사용자 경험으로 WSL 환경에 dotnet 런타임/SDK 설치 방법
13165정성태11/18/202212800개발 환경 구성: 650. Azure - "scm" 프로세스와 엮인 서비스 모음
13164정성태11/18/202215435개발 환경 구성: 649. Azure - 비주얼 스튜디오를 이용한 AppService 원격 디버그 방법
13163정성태11/17/202215591개발 환경 구성: 648. 비주얼 스튜디오에서 안드로이드 기기 인식하는 방법
13162정성태11/15/202216851.NET Framework: 2069. .NET 7 - AOT(ahead-of-time) 컴파일 [1]
13161정성태11/14/202215282.NET Framework: 2068. C# - PublishSingleFile로 배포한 이미지의 역어셈블 가능 여부 (난독화 필요성) [4]
13160정성태11/11/202215882.NET Framework: 2067. C# - PublishSingleFile 적용 시 native/managed 모듈 통합 옵션
13159정성태11/10/202218579.NET Framework: 2066. C# - PublishSingleFile과 관련된 옵션 [3]
13158정성태11/9/202214265오류 유형: 826. Workload definition 'wasm-tools' in manifest 'microsoft.net.workload.mono.toolchain' [...] conflicts with manifest 'microsoft.net.workload.mono.toolchain.net7'
13157정성태11/8/202215322.NET Framework: 2065. C# - Mutex의 비동기 버전파일 다운로드1
13156정성태11/7/202217594.NET Framework: 2064. C# - Mutex와 Semaphore/SemaphoreSlim 차이점파일 다운로드1
13155정성태11/4/202215442디버깅 기술: 183. TCP 동시 접속 (연결이 아닌) 시도를 1개로 제한한 서버
13154정성태11/3/202216193.NET Framework: 2063. .NET 5+부터 지원되는 GC.GetGCMemoryInfo파일 다운로드1
13153정성태11/2/202217258.NET Framework: 2062. C# - 코드로 재현하는 소켓 상태(SYN_SENT, SYN_RECV)
13152정성태11/1/202216341.NET Framework: 2061. ASP.NET Core - DI로 추가한 클래스의 초기화 방법 [1]
13151정성태10/31/202215645C/C++: 161. Windows 11 환경에서 raw socket 테스트하는 방법파일 다운로드1
13150정성태10/30/202213973C/C++: 160. Visual Studio 2022로 빌드한 C++ 프로그램을 위한 다른 PC에서 실행하는 방법
13149정성태10/27/202214583오류 유형: 825. C# - CLR ETW 이벤트 수신이 GCHeapStats_V1/V2에 대해 안 되는 문제파일 다운로드1
13148정성태10/26/202215278오류 유형: 824. msbuild 에러 - error NETSDK1005: Assets file '...\project.assets.json' doesn't have a target for 'net5.0'. Ensure that restore has run and that you have included 'net5.0' in the TargetFramew
... 31  [32]  33  34  35  36  37  38  39  40  41  42  43  44  45  ...