스택메모리
각 함수에서 사용하는 지역 변수, 매개 변수 등을 임시적으로 저장하는 공간이다.
스택 메모리의 크기는 프로그램 빌드 시에 결정되며 위치는 실행 시에 결정된다.
위의 간단한 예제로 설명하자면 main 함수의 int a, int b 그리고 add 함수로 넘겨주는 매개변수 a, b 모두 지역, 매개변수 임으로 스택메모리에 값이 저장된다.
C언어에서는 New라는 키워드가 없는데 이는 기본적으로 변수를 선언하면 변수가 스택메모리에 할당되기 때문이다.
C언어 기본 자료형 변수(char, int, float)는 스택 메모리에 할당되며 기본 자료형을 함수 매개변수로 전달하면 스택에 복사본을 만든다(값복사, 값형)
반대로 다른언어에서 처럼 malloc을 쓰면 Heap 영역에 할당된다. Heap 영역에 메모리를 할당하면 운영체제가 프로그래머가 요구한 메모리 사이즈를 검색하고 반환하며 크기가 가변적이라 메모리에 구멍이 숭숭뚫린다.
스택 메모리 조금 더 자세히
1. 스택은 큰주소에서 작은 주소로 쌓인다
2. EBP(Extended Base Pointer) : 현재 스택 프레임의 첫 주소(Base 주소)
3. ESP(Extened Stack Pointer) : 현재 스택 포인터
4. 스택 프레임(stack frame) : 각 함수가 사용하는 스택 메모리의 범위
스택 메모리 + 어셈블리어
왼쪽은 어셈블리어 오른쪽은 그에 대응되는 스택 메모리의 그림이다.
1. esp를 ebp로 Move해라! 그러니깐 같은 것을 가르켜라
2. esp를 14h(16진수) 즉 십진수 20만큼 sub(빼라) -> ESP 영역을 20만큼 확장해라 한 블럭이 4바이트니 5칸 밑으로 ESP를 내려라
3. 주소 ebp - 4 즉 0x0073FE68에 0을 넣어라. ebp - 8 주소에 1을 넣어라, ebp -0Ch 주소에 2를 넣어라
4-1. eax에 ebp-0Ch의 값을 넣고, ecx에 ebp-8의 값을 넣어라(eax, ecx는 CPU 레지스터 영역임)
4-2. 아까 담은 ecx의 값을 현재 스택 포인터인 esp에 넣어라 그리고 esp+4에는 eax를 넣어라
방금 4줄의 어셈블리어가 add 함수에 매개변수를 전달하는 방식이다
5. 00011040이라는 주소를 Call한다(add 함수) 그리고 다시 돌아올 주소 0001102D를 ESP-14h 주소에 add한다.
6. 호출자 함수의 EBP를 PUSH 한다. 호출된 add 함수가 끝나고 다시 main함수로 돌아가야하니깐
7. EBP를 ESP로 move한다
8. ESP를 0c(12) 만큼 늘림으로서 add 함수의 스택 메모리 영역 확보
9-1. eax, ecx, edx 에 2,1,1을 담고 dex에 ebp+0Ch 값을 더한다 1+2 = 3
9-2. edx의 값을 ebp-4에 할당한다 즉(res 변수)
10-1. edx에 ebp-4를 할당(의미없음)
10-2. eax의 값(2)를 ebp -8에 할당
10-3, edx의 값을 eax에 할당(의미없음)
10-4, ecx의 값(1)을 ebp-0c에 할당
11. esp에 0c를 add 함으로서 add 함수의 스택 영역을 해제한다. 여기서 중요한점은 3,2,1과 같은 값을 지우지 않는다는 점이다. 나중에 다른 함수가 해당 주소를 다시 읽었을때 3,2,1 같은 쓰레기 데이터가 남아있는 이유가 이때문이다.
12. 0x0073FE6C의 주소로 ebp를 이동하면서 pop
13. return 명령어 실행 : 0x0001102D 의 주소로 return (add 함수 종료)
14-1. esp, 14h 만큼 add : esp가 ebp를 가르키게함 (메모리 축소)
14-2. ebp를 pop
14-3. return 후 종료
'C' 카테고리의 다른 글
C언어 구조체 (0) | 2019.12.18 |
---|---|
C언어 포인터 배열 (0) | 2019.12.13 |
C 포인터 const (0) | 2019.12.12 |
C언어 배열과 포인터 차이점 (0) | 2019.12.12 |
C언어 Char, int 정리 (0) | 2019.12.05 |