कंपाइलर डिज़ाइन - रन-टाइम पर्यावरण
एक स्रोत कोड के रूप में एक कार्यक्रम केवल पाठ (कोड, बयान आदि) का एक संग्रह है और इसे जीवित बनाने के लिए, इसे लक्ष्य मशीन पर किए जाने वाले कार्यों की आवश्यकता होती है। निर्देशों को निष्पादित करने के लिए एक प्रोग्राम को मेमोरी संसाधनों की आवश्यकता होती है। एक प्रोग्राम में प्रक्रियाओं, पहचानकर्ताओं आदि के नाम होते हैं, जिन्हें रनटाइम पर वास्तविक मेमोरी लोकेशन के साथ मैपिंग की आवश्यकता होती है।
रनटाइम से, हमारा मतलब निष्पादन में एक कार्यक्रम है। रनटाइम वातावरण लक्ष्य मशीन की एक स्थिति है, जिसमें सिस्टम में चल रही प्रक्रियाओं को सेवाएं प्रदान करने के लिए सॉफ्टवेयर लाइब्रेरी, पर्यावरण चर आदि शामिल हो सकते हैं।
रनटाइम सपोर्ट सिस्टम एक पैकेज है, जो ज्यादातर निष्पादन योग्य कार्यक्रम के साथ ही उत्पन्न होता है और प्रक्रिया और रनटाइम वातावरण के बीच प्रक्रिया संचार की सुविधा प्रदान करता है। यह मेमोरी एलोकेशन और डी-आवंटन का ध्यान रखता है, जबकि प्रोग्राम को निष्पादित किया जा रहा है।
सक्रियण पेड़
एक कार्यक्रम कई प्रक्रियाओं में संयुक्त निर्देशों का एक अनुक्रम है। एक प्रक्रिया में निर्देश क्रमिक रूप से निष्पादित किए जाते हैं। एक प्रक्रिया में एक शुरुआत और एक अंतिम सीमांकक होता है और इसके अंदर मौजूद हर चीज को प्रक्रिया का शरीर कहा जाता है। प्रक्रिया पहचानकर्ता और इसके अंदर परिमित निर्देशों का क्रम प्रक्रिया का शरीर बनाता है।
एक प्रक्रिया के निष्पादन को इसकी सक्रियता कहा जाता है। एक सक्रियण रिकॉर्ड में एक प्रक्रिया को कॉल करने के लिए आवश्यक सभी आवश्यक जानकारी होती है। एक सक्रियण रिकॉर्ड में निम्नलिखित इकाइयाँ हो सकती हैं (उपयोग की गई स्रोत भाषा के आधार पर)।
temporaries | एक अभिव्यक्ति के अस्थायी और मध्यवर्ती मूल्यों को संग्रहीत करता है। |
स्थानीय डेटा | कहा प्रक्रिया के स्थानीय डेटा संग्रहीत करता है। |
मशीन की स्थिति | प्रक्रिया से पहले स्टोर मशीन की स्थिति जैसे रजिस्टर, प्रोग्राम काउंटर आदि। |
नियंत्रण लिंक | कॉलर प्रक्रिया के सक्रियण रिकॉर्ड के पते को संग्रहीत करता है। |
पहुंच लिंक | डेटा की जानकारी संग्रहीत करता है जो स्थानीय दायरे से बाहर है। |
वास्तविक पैरामीटर | स्टोर वास्तविक पैरामीटर, अर्थात्, पैरामीटर जो इनपुट को प्रक्रिया को भेजने के लिए उपयोग किया जाता है। |
प्रतिलाभ की मात्रा | स्टोर रिटर्न मान। |
जब भी किसी प्रक्रिया को निष्पादित किया जाता है, उसका सक्रियण रिकॉर्ड स्टैक पर संग्रहीत होता है, जिसे नियंत्रण स्टैक के रूप में भी जाना जाता है। जब कोई प्रक्रिया किसी अन्य प्रक्रिया को कॉल करती है, तो कॉल करने वाले का निष्पादन तब तक निलंबित रहता है जब तक कि प्रक्रिया समाप्त नहीं हो जाती। इस समय, कॉल की गई प्रक्रिया का सक्रियण रिकॉर्ड स्टैक पर संग्रहीत होता है।
हम मानते हैं कि प्रोग्राम नियंत्रण क्रमबद्ध तरीके से बहता है और जब एक प्रक्रिया कहलाती है, तो इसका नियंत्रण नामक प्रक्रिया को स्थानांतरित कर दिया जाता है। जब एक प्रक्रिया को निष्पादित किया जाता है, तो यह नियंत्रण को कॉल करने वाले को वापस कर देता है। इस प्रकार के नियंत्रण प्रवाह को पेड़ के रूप में सक्रियणों की एक श्रृंखला का प्रतिनिधित्व करना आसान हो जाता है, जिसे के रूप में जाना जाता है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) विधि में काम करता है और यह आवंटन रणनीति पुनरावर्ती प्रक्रिया कॉल के लिए बहुत उपयोगी है।
ढेर आवंटन
एक प्रक्रिया में स्थानीय चर को केवल रनटाइम पर ही आबंटित और डी-आबंटित किया जाता है। हीप आवंटन का उपयोग डायनामिक रूप से वेरिएबल्स को मेमोरी आवंटित करने के लिए किया जाता है और यह तब दावा करता है जब वेरिएबल की आवश्यकता नहीं होती है।
स्टैटिक और आवंटित मेमोरी क्षेत्र को छोड़कर, स्टैक और हीप मेमोरी दोनों गतिशील और अप्रत्याशित रूप से सिकुड़ और बढ़ सकते हैं। इसलिए, उन्हें सिस्टम में निश्चित मात्रा में मेमोरी प्रदान नहीं की जा सकती है।
जैसा कि ऊपर की छवि में दिखाया गया है, कोड का पाठ भाग एक निश्चित मात्रा में मेमोरी आवंटित किया जाता है। कार्यक्रम के लिए आवंटित कुल मेमोरी के चरम पर स्टैक और हीप मेमोरी की व्यवस्था की जाती है। दोनों एक दूसरे के खिलाफ सिकुड़ते और बढ़ते हैं।
पैरामीटर पासिंग
प्रक्रियाओं के बीच संचार माध्यम को पैरामीटर पासिंग के रूप में जाना जाता है। एक कॉलिंग प्रक्रिया से चर के मूल्यों को कुछ तंत्र द्वारा बुलाया प्रक्रिया में स्थानांतरित किया जाता है। आगे बढ़ने से पहले, पहले एक कार्यक्रम में मूल्यों से संबंधित कुछ बुनियादी शब्दावली के माध्यम से जाना।
आर-मूल्य
किसी अभिव्यक्ति के मूल्य को उसका r- मूल्य कहा जाता है। एकल चर में निहित मूल्य भी आर-मान बन जाता है यदि यह असाइनमेंट ऑपरेटर के दाईं ओर दिखाई देता है। r- मान हमेशा किसी अन्य चर को सौंपा जा सकता है।
एल मूल्य
स्मृति का स्थान (पता) जहां एक अभिव्यक्ति संग्रहीत होती है, उस अभिव्यक्ति के एल-मूल्य के रूप में जाना जाता है। यह हमेशा एक असाइनमेंट ऑपरेटर के बाएं हाथ में दिखाई देता है।
उदाहरण के लिए:
day = 1;
week = day * 7;
month = 1;
year = month * 12;
इस उदाहरण से, हम समझते हैं कि लगातार मान जैसे 1, 7, 12 और चर जैसे दिन, सप्ताह, महीना और वर्ष, सभी में r-मान हैं। केवल चर में ही मान हैं, क्योंकि वे उनके द्वारा सौंपी गई मेमोरी लोकेशन का भी प्रतिनिधित्व करते हैं।
उदाहरण के लिए:
7 = x + y;
एक L- मान त्रुटि है, क्योंकि निरंतर 7 किसी भी स्मृति स्थान का प्रतिनिधित्व नहीं करता है।
औपचारिक पैरामीटर
कॉल प्रक्रिया द्वारा पारित जानकारी लेने वाले चर को औपचारिक पैरामीटर कहा जाता है। इन चर को फ़ंक्शन की परिभाषा में घोषित किया जाता है।
वास्तविक पैरामीटर
चर, जिनके मूल्य या पते को प्रक्रिया में पारित किया जा रहा है, उन्हें वास्तविक पैरामीटर कहा जाता है। इन चर को तर्कों के रूप में फ़ंक्शन कॉल में निर्दिष्ट किया जाता है।
Example:
fun_one()
{
int actual_parameter = 10;
call fun_two(int actual_parameter);
}
fun_two(int formal_parameter)
{
print formal_parameter;
}
औपचारिक पैरामीटर वास्तविक पैरामीटर की जानकारी रखते हैं, जो इस्तेमाल की जाने वाली तकनीक के पैरामीटर पर निर्भर करता है। यह एक मान या एक पता हो सकता है।
मान से पास करें
वैल्यू मैकेनिज़्म द्वारा पास होने पर, कॉलिंग प्रक्रिया वास्तविक मापदंडों के आर-वैल्यू से गुजरती है और कंपाइलर उस प्रक्रिया के सक्रियण रिकॉर्ड में डाल देता है। औपचारिक पैरामीटर तब कॉलिंग प्रक्रिया द्वारा पारित मूल्यों को पकड़ते हैं। यदि औपचारिक मापदंडों द्वारा आयोजित मूल्यों को बदल दिया जाता है, तो इसका वास्तविक मापदंडों पर कोई प्रभाव नहीं होना चाहिए।
संदर्भ द्वारा पास करें
संदर्भ तंत्र द्वारा पास में, वास्तविक पैरामीटर के एल-मूल्य को तथाकथित प्रक्रिया के सक्रियण रिकॉर्ड में कॉपी किया जाता है। इस तरह, अब प्रक्रिया को वास्तविक पैरामीटर का पता (मेमोरी स्थान) और औपचारिक पैरामीटर समान मेमोरी स्थान को संदर्भित करता है। इसलिए, यदि औपचारिक पैरामीटर द्वारा इंगित मूल्य को बदल दिया जाता है, तो प्रभाव वास्तविक पैरामीटर पर देखा जाना चाहिए, क्योंकि उन्हें भी उसी मूल्य को इंगित करना चाहिए।
कॉपी-रीस्टोर करके पास करें
यह पैरामीटर पासिंग मैकेनिज्म 'पास-बाय-रेफरेंस' के समान काम करता है, सिवाय इसके कि वास्तविक पैरामीटर में बदलाव तब किया जाता है जब कॉल की प्रक्रिया समाप्त होती है। फ़ंक्शन कॉल पर, वास्तविक पैरामीटर के मानों को प्रक्रिया के सक्रियण रिकॉर्ड में कॉपी किया जाता है। औपचारिक मापदंडों का यदि हेरफेर किया जाता है तो वास्तविक मापदंडों पर कोई वास्तविक समय प्रभाव नहीं होता है (जैसा कि एल-मान पारित किया जाता है), लेकिन जब तथाकथित प्रक्रिया समाप्त होती है, तो औपचारिक मापदंडों के एल-मूल्यों को वास्तविक मापदंडों के एल-मूल्यों पर कॉपी किया जाता है।
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 का एल-मान वास्तविक पैरामीटर y पर कॉपी किया जाता है। यहां तक कि अगर प्रक्रिया समाप्त होने से पहले y का मूल्य बदल दिया जाता है, तो x का एल-मूल्य y के एल-मूल्य पर कॉपी किया जाता है जिससे यह संदर्भ द्वारा कॉल की तरह व्यवहार करता है।
नाम से पास
अल्गोल जैसी भाषाएं एक नए तरह का पैरामीटर पासिंग मैकेनिज्म प्रदान करती हैं जो सी भाषा में प्रीप्रोसेसर की तरह काम करता है। नाम तंत्र द्वारा पास में, प्रक्रिया के नाम को उसके वास्तविक निकाय द्वारा प्रतिस्थापित किया जाता है। पास-बाय-नेम पाठ प्रक्रिया के शरीर में संबंधित मापदंडों के लिए कॉल प्रक्रिया में तर्क भावों को वैकल्पिक रूप से प्रतिस्थापित करता है ताकि यह अब वास्तविक मापदंडों पर काम कर सके, बहुत कुछ पास-बाय-रेफरेंस की तरह।