컴파일러 설계-런타임 환경

소스 코드로서의 프로그램은 단순히 텍스트 (코드, 명령문 등)의 모음 일 뿐이며이를 활성화하려면 대상 컴퓨터에서 수행 할 작업이 필요합니다. 프로그램은 명령을 실행하기 위해 메모리 리소스가 필요합니다. 프로그램에는 런타임에 실제 메모리 위치와 매핑해야하는 프로 시저, 식별자 등에 대한 이름이 포함되어 있습니다.

런타임이란 실행중인 프로그램을 의미합니다. 런타임 환경은 시스템에서 실행되는 프로세스에 서비스를 제공하기 위해 소프트웨어 라이브러리, 환경 변수 등을 포함 할 수있는 대상 머신의 상태입니다.

런타임 지원 시스템은 주로 실행 가능한 프로그램 자체로 생성되는 패키지로 프로세스와 런타임 환경 간의 프로세스 통신을 용이하게합니다. 프로그램이 실행되는 동안 메모리 할당 및 할당 해제를 처리합니다.

활성화 트리

프로그램은 여러 절차로 결합 된 일련의 명령입니다. 프로 시저의 명령은 순차적으로 실행됩니다. 프로 시저에는 시작 및 끝 구분 기호가 있으며 그 안에있는 모든 것을 프로 시저 본문이라고합니다. 프로 시저 식별자와 그 안에있는 유한 명령어 시퀀스가 ​​프로 시저의 본문을 구성합니다.

프로 시저의 실행을 활성화라고합니다. 활성화 레코드에는 절차를 호출하는 데 필요한 모든 정보가 포함되어 있습니다. 활성화 레코드에는 다음 단위가 포함될 수 있습니다 (사용 된 소스 언어에 따라 다름).

임시 식의 임시 및 중간 값을 저장합니다.
로컬 데이터 호출 된 프로 시저의 로컬 데이터를 저장합니다.
기계 상태 프로 시저가 호출되기 전에 레지스터, 프로그램 카운터 등과 같은 기계 상태를 저장합니다.
제어 링크 호출자 프로 시저의 활성화 레코드 주소를 저장합니다.
액세스 링크 로컬 범위를 벗어난 데이터 정보를 저장합니다.
실제 매개 변수 실제 매개 변수, 즉 호출 된 프로 시저에 입력을 보내는 데 사용되는 매개 변수를 저장합니다.
반환 값 반환 값을 저장합니다.

프로 시저가 실행될 때마다 활성화 레코드가 제어 스택이라고도하는 스택에 저장됩니다. 프로 시저가 다른 프로 시저를 호출하면 호출 된 프로 시저가 실행을 완료 할 때까지 호출자의 실행이 일시 중단됩니다. 이때 호출 된 프로 시저의 활성화 레코드가 스택에 저장됩니다.

프로그램 제어가 순차적으로 흐르고 프로 시저가 호출되면 제어가 호출 된 프로 시저로 전송된다고 가정합니다. 호출 된 프로 시저가 실행되면 컨트롤을 호출자에게 반환합니다. 이러한 유형의 제어 흐름을 사용하면 일련의 활성화를 트리 형태로 쉽게 나타낼 수 있습니다.activation tree.

이 개념을 이해하기 위해 코드를 예로 들어 보겠습니다.

. . .
printf(“Enter Your Name: “);
scanf(“%s”, username);
show_data(username);
printf(“Press any key to continue…”);
. . .
int show_data(char *user)
   {
   printf(“Your name is %s”, username);
   return 0;
   }
. . .

아래는 주어진 코드의 활성화 트리입니다.

이제 프로 시저가 깊이 우선 방식으로 실행되므로 스택 할당이 프로 시저 활성화에 가장 적합한 스토리지 형식이라는 것을 이해합니다.

스토리지 할당

런타임 환경은 다음 엔티티에 대한 런타임 메모리 요구 사항을 관리합니다.

  • Code: 런타임시 변경되지 않는 프로그램의 텍스트 부분이라고합니다. 메모리 요구 사항은 컴파일 시간에 알려져 있습니다.

  • Procedures: 텍스트 부분은 정적이지만 무작위로 호출됩니다. 그렇기 때문에 스택 스토리지는 프로 시저 호출 및 활성화를 관리하는 데 사용됩니다.

  • Variables: 변수는 전역이거나 상수가 아닌 경우 런타임에만 알려져 있습니다. 힙 메모리 할당 체계는 런타임에서 변수에 대한 메모리 할당 및 할당 해제를 관리하는 데 사용됩니다.

정적 할당

이 할당 방식에서 컴파일 데이터는 메모리의 고정 된 위치에 바인딩되며 프로그램이 실행될 때 변경되지 않습니다. 메모리 요구 사항 및 저장 위치가 미리 알려져 있으므로 메모리 할당 및 할당 해제를위한 런타임 지원 패키지가 필요하지 않습니다.

스택 할당

프로 시저 호출 및 활성화는 스택 메모리 할당을 통해 관리됩니다. LIFO (last-in-first-out) 방식으로 작동하며이 할당 전략은 재귀 프로 시저 호출에 매우 유용합니다.

힙 할당

프로 시저에 로컬 인 변수는 런타임시에만 할당 및 할당 해제됩니다. 힙 할당은 변수에 메모리를 동적으로 할당하고 변수가 더 이상 필요하지 않을 때이를 되 돌리는 데 사용됩니다.

정적으로 할당 된 메모리 영역을 제외하고 스택 및 힙 메모리는 모두 동적으로 그리고 예기치 않게 늘어나거나 줄어들 수 있습니다. 따라서 시스템에 고정 된 양의 메모리를 제공 할 수 없습니다.

위의 이미지에서 볼 수 있듯이 코드의 텍스트 부분에는 고정 된 양의 메모리가 할당됩니다. 스택 및 힙 메모리는 프로그램에 할당 된 총 메모리의 극단에 정렬됩니다. 둘 다 수축하고 서로에 대해 성장합니다.

매개 변수 전달

프로 시저 간의 통신 매체를 매개 변수 전달이라고합니다. 호출 프로 시저의 변수 값은 일부 메커니즘에 의해 호출 된 프로 시저로 전송됩니다. 계속 진행하기 전에 먼저 프로그램의 값과 관련된 몇 가지 기본 용어를 살펴보십시오.

r- 값

식의 값을 r- 값이라고합니다. 단일 변수에 포함 된 값이 할당 연산자의 오른쪽에 나타나면 r- 값이됩니다. r- 값은 항상 다른 변수에 할당 될 수 있습니다.

l- 값

식이 저장되는 메모리 (주소)의 위치를 ​​해당 식의 l- 값이라고합니다. 항상 할당 연산자의 왼쪽에 나타납니다.

예를 들면 :

day = 1;
week = day * 7;
month = 1;
year = month * 12;

이 예에서 1, 7, 12와 같은 상수 값과 일, 주, 월 및 연도와 같은 변수는 모두 r- 값을 갖는다는 것을 이해합니다. 변수에만 할당 된 메모리 위치를 나타 내기 때문에 l- 값이 있습니다.

예를 들면 :

7 = x + y;

상수 7은 메모리 위치를 나타내지 않으므로 l 값 오류입니다.

공식 매개 변수

호출자 프로 시저가 전달한 정보를받는 변수를 형식 매개 변수라고합니다. 이러한 변수는 호출 된 함수의 정의에서 선언됩니다.

실제 매개 변수

값 또는 주소가 호출 된 프로 시저에 전달되는 변수를 실제 매개 변수라고합니다. 이러한 변수는 함수 호출에서 인수로 지정됩니다.

Example:

fun_one()
{
   int actual_parameter = 10;
   call fun_two(int actual_parameter);
}
   fun_two(int formal_parameter)
{
   print formal_parameter;
}

형식 매개 변수는 사용 된 매개 변수 전달 기술에 따라 실제 매개 변수의 정보를 보유합니다. 값 또는 주소 일 수 있습니다.

가치로 전달

값에 의한 전달 메커니즘에서 호출 프로시 저는 실제 매개 변수의 r 값을 전달하고 컴파일러는이를 호출 된 프로 시저의 활성화 레코드에 넣습니다. 그런 다음 형식 매개 변수는 호출 프로 시저가 전달한 값을 보유합니다. 형식 매개 변수가 보유한 값이 변경되면 실제 매개 변수에 영향을주지 않아야합니다.

참조로 통과

참조 메커니즘에 의한 전달에서 실제 매개 변수의 l- 값은 호출 된 프로 시저의 활성화 레코드에 복사됩니다. 이런 식으로 호출 된 프로시 저는 이제 실제 매개 변수의 주소 (메모리 위치)를 가지며 형식 매개 변수는 동일한 메모리 위치를 참조합니다. 따라서 형식 매개 변수가 가리키는 값이 변경되면 동일한 값을 가리켜 야하기 때문에 실제 매개 변수에 미치는 영향을 확인해야합니다.

복사-복원 통과

이 매개 변수 전달 메커니즘은 호출 된 프로 시저가 종료 될 때 실제 매개 변수가 변경된다는 점을 제외하면 '참조에 의한 전달'과 유사하게 작동합니다. 함수 호출시 실제 매개 변수 값이 호출 된 프로 시저의 활성화 레코드에 복사됩니다. 조작 된 경우 형식 매개 변수는 실제 매개 변수에 실시간 영향을 미치지 않지만 (l 값이 전달됨에 따라) 호출 된 프로 시저가 종료되면 형식 매개 변수의 l 값이 실제 매개 변수의 l 값에 복사됩니다.

Example:

int y; 
calling_procedure() 
{
   y = 10;     
   copy_restore(y); //l-value of y is passed
   printf y; //prints 99 
}
copy_restore(int x) 
{     
   x = 99; // y still has value 10 (unaffected)
   y = 0; // y is now 0 
}

이 함수가 종료되면 형식 매개 변수 x의 l- 값이 실제 매개 변수 y에 복사됩니다. 프로 시저가 종료되기 전에 y의 값이 변경 되더라도 x의 l 값이 y의 l 값에 복사되어 참조에 의한 호출처럼 동작합니다.

이름으로 전달

Algol과 같은 언어는 C 언어의 전 처리기처럼 작동하는 새로운 종류의 매개 변수 전달 메커니즘을 제공합니다. 이름 전달 메커니즘에서 호출되는 프로 시저의 이름은 실제 본문으로 대체됩니다. 이름에 의한 전달은 이제 참조에 의한 전달과 같이 실제 매개 변수에 대해 작동 할 수 있도록 프로 시저 본문의 해당 매개 변수에 대한 프로 시저 호출의 인수 표현식을 텍스트 방식으로 대체합니다.