본 페이지에서는 함수 호출 규약에 대하여 정리하고자 한다.
먼저 윈도우 프로그램의 정석 마이크로 소프트 문서 페이지의 개요를 이해하기 위해 번역하고,
좀더 구체적인 개념에 대해 정리한다.
마이크로 소프트 문서 번역
출처 - https://docs.microsoft.com/en-us/cpp/cpp/calling-conventions?view=vs-2019
Calling Conventions
Visual C/C++ 컴파일러는 내부와 외부 함수을 호출하기 위한 여러가지 다른 규약을 제공한다. 이 차이점을 이해하는 것은 구독자의 프로그램을 디버깅 할 수 있도록 도와주거나 어셈블리 언어 루틴을 포함하는 구독자의 코드를 링크할 수 있도록 도와준다.
이 주제에서 토픽들은 함수 호출 규약의 차이점을 설명한다, 아규먼트들이 어떻게 입력되고 반환값들은 함수에 의해 어떻게 반환되는지 등에 대한 것들. 토픽은 구독자가 자신의 프롤로그 코드와 에필로그 코드를 작성할 수 있게 하는 좀더 향상된 특성인 naked 함수 호출에 대해 설명한다.
x64 프로세서를 위한 함수 호출 규약에 대한 정보는 여기를 참조하라.
# 이 세션의 토픽들
1) 아규먼트 입력과 네이밍 규약 ( __cdecl, __stdcall, __fastcall, and others )
2) 함수 호출 예제: Function Prototype and Call
3) 커스터머 프롤로그/에필로그 코드를 호출하는 naked 함수 사용하기
5) 예전 함수 호출 규약
아규먼트 입력과 네이밍 규약
MS C++ 컴파일러는 구독자가 함수와 호출자 간의 아규먼트 입력과 반환값을 위한 규약을 정하도록 한다. - 생략( - 구독자는 ESI, EDI, EBX 레지스터에 저장하고 복구하기 위한 프롤로그와 에필로그 코드를 생성한다.
구독자 함수의 프롤로크와 에필로그 코드를 정의하는 방법은 Naked Function Calls을 참조하라. x64 플랫폼을 대상으로 하는 기본 함수 호출 규약에 대한 정보는 x64 Calling Convntions를 참고하라. ARM 플랫폼을 대상으로 하는 함수 호출 규약 이슈에 대한 정보는 Common Visual C++ ARM Migration Issues를 참고하라.
Visual c/C++ 컴파일러가 제공하는 함수 호출 규약은 다음과 같다.
* __cdecl
- C/C++의 기본 호출 규약
- 스택을 정리하는 주체는 Caller
- 가변 인수 함수를 지원한다. ( ex) printf() )
* __sdtcall
- Win32 API에서 사용
- 스택을 정리하는 주체는 Callee
- 가변 인수 함수를 지원하지 않는다.
Naked Function Calls
naked 속성으로 선언된 함수는 인라인 어셈블러를 사용한 구독자의 프롤로그/에필로그 시퀀스를 작성할 수 있는 프롤로그와 에필로그 코드 없이 발생된다. naked 함수는 향상된 특성으로 제공된다. naked 함수는 C/C++ 이외의 컨텍스트로부터 호출되는 함수를 선언할 수 있으므로 파라미터가 어디에 있는지 혹은 어떤 레지스터가 보존되는지에 대한 다양한 가정을 할 수 있다. 예를 들어, 인터럽트 핸들러와 같은 루틴을 포함한다. 이 특성은 가상 디바이스 드라이버의 작업자에게 유용한다.
- naked 속성으로 선언된 함수에 대해 컴파일러는 프롤로그와 에필로그 코드 없이 코드를 생성한다. 구독자는 인라인 어셈블러 코드를 사용하여 자신의 프롤로그와 에필로그 시퀀스를 작성하기 위해 이 특성을 이용할 수 있다. naked 함수는 특히 가상 디바이스 드라이버를 작성하는데 유용하다. 이것은 x86와 ARM에서 가능하고 x64는 이용할 수 없다.
자료 정리
에 달린 댓글들 중에서 흥미로운 몇 가지를 링크한다.
- 시계방향 / 나선 규칙, Clockwise/Spiral Rule :
- 함수원형과 나선 규칙에 따른 설명
※ Prolog / Epilog Code ? ( soen.kr )
- 함수가 호출될 때 스택에는 함수로 전달되는 인수, 복귀 번지, 지역 변수 등의 정보들이 저장된다. 스택에 저장되는 함수의 호출 정보를 스택 프레임이라고 한다.
- 함수가 호출되는 과정( 어셈블리어에서 실행되는 시퀀스 )에서 함수 실행전 실행 준비 단계를 프롤로그 코드, 실행 후 정리 단계를 에필로그 코드라고 한다.
※ Static Link, Dynamic Link
댓글
댓글 쓰기