विधानसभा - त्वरित गाइड

असेंबली लैंग्वेज क्या है?

प्रत्येक व्यक्तिगत कंप्यूटर में एक माइक्रोप्रोसेसर होता है जो कंप्यूटर की अंकगणितीय, तार्किक और नियंत्रण गतिविधियों का प्रबंधन करता है।

प्रोसेसर के प्रत्येक परिवार के पास विभिन्न कार्यों को संभालने के लिए निर्देशों का अपना सेट है जैसे कीबोर्ड से इनपुट प्राप्त करना, स्क्रीन पर जानकारी प्रदर्शित करना और विभिन्न अन्य कार्य करना। निर्देशों के इन सेट को 'मशीन भाषा निर्देश' कहा जाता है।

एक प्रोसेसर केवल मशीन भाषा निर्देशों को समझता है, जो कि 1 और 0 के तार हैं। हालाँकि, सॉफ़्टवेयर विकास में उपयोग करने के लिए मशीन भाषा बहुत अस्पष्ट और जटिल है। तो, निम्न-स्तरीय असेंबली भाषा प्रोसेसर के एक विशिष्ट परिवार के लिए डिज़ाइन की गई है जो प्रतीकात्मक कोड में विभिन्न निर्देशों और एक अधिक समझने योग्य रूप का प्रतिनिधित्व करता है।

असेंबली लैंग्वेज के फायदे

असेंबली लैंग्वेज की समझ होना किसी को भी जागरूक करता है -

  • ओएस, प्रोसेसर और BIOS के साथ प्रोग्राम इंटरफ़ेस कैसे;
  • मेमोरी और अन्य बाहरी उपकरणों में डेटा का प्रतिनिधित्व कैसे किया जाता है;
  • प्रोसेसर कैसे पहुंचता है और निर्देश को कार्यान्वित करता है;
  • डेटा तक पहुंच और प्रक्रिया कैसे निर्देश देती है;
  • कैसे एक कार्यक्रम बाहरी उपकरणों तक पहुँचता है।

विधानसभा भाषा का उपयोग करने के अन्य फायदे हैं -

  • इसके लिए कम मेमोरी और निष्पादन समय की आवश्यकता होती है;

  • यह आसान तरीके से हार्डवेयर-विशिष्ट जटिल नौकरियों की अनुमति देता है;

  • यह समय-महत्वपूर्ण नौकरियों के लिए उपयुक्त है;

  • यह रुकावट सेवा दिनचर्या और अन्य स्मृति निवासी कार्यक्रम लिखने के लिए सबसे उपयुक्त है।

पीसी हार्डवेयर की बुनियादी विशेषताएं

पीसी के मुख्य आंतरिक हार्डवेयर में प्रोसेसर, मेमोरी और रजिस्टर होते हैं। रजिस्टर्स प्रोसेसर कंपोनेंट्स होते हैं जो डेटा और एड्रेस को होल्ड करते हैं। किसी प्रोग्राम को निष्पादित करने के लिए, सिस्टम बाहरी डिवाइस से आंतरिक मेमोरी में इसे कॉपी करता है। प्रोसेसर प्रोग्राम निर्देशों को निष्पादित करता है।

कंप्यूटर स्टोरेज की मूलभूत इकाई थोड़ी है; यह ON (1) या OFF (0) हो सकता है और 8 संबंधित बिट्स का एक समूह अधिकांश आधुनिक कंप्यूटरों पर एक बाइट बनाता है।

तो, एक बाइट में बिट्स की संख्या को विषम बनाने के लिए समता बिट का उपयोग किया जाता है। यदि समता सम है, तो सिस्टम मानता है कि समता त्रुटि (हालांकि दुर्लभ) थी, जो हार्डवेयर की खराबी या विद्युत गड़बड़ी के कारण हो सकती है।

प्रोसेसर निम्नलिखित डेटा आकारों का समर्थन करता है -

  • वर्ड: एक 2-बाइट डेटा आइटम
  • डबलवर्ड: एक 4-बाइट (32 बिट) डेटा आइटम
  • क्वाडवर्ड: एक 8-बाइट (64 बिट) डेटा आइटम
  • अनुच्छेद: एक 16-बाइट (128 बिट) क्षेत्र
  • किलोबाइट: 1024 बाइट्स
  • मेगाबाइट: 1,048,576 बाइट्स

बाइनरी नंबर सिस्टम

प्रत्येक संख्या प्रणाली स्थितीय अंकन का उपयोग करती है, अर्थात, प्रत्येक स्थिति जिसमें एक अंक लिखा जाता है, का एक अलग स्थितीय मूल्य होता है। प्रत्येक स्थिति आधार की शक्ति है, जो बाइनरी नंबर सिस्टम के लिए 2 है, और ये शक्तियां 0 से शुरू होती हैं और 1 से बढ़ती हैं।

निम्न तालिका 8-बिट बाइनरी संख्या के लिए स्थितीय मान दिखाती है, जहाँ सभी बिट्स चालू हैं।

बिट मूल्य 1 1 1 1 1 1 1 1
बेस 2 की शक्ति के रूप में स्थिति मूल्य 128 64 32 16 8 4 2 1
बिट संख्या 7 6 5 4 3 2 1 0

एक द्विआधारी संख्या का मूल्य 1 बिट्स की उपस्थिति और उनकी स्थिति के मूल्य पर आधारित है। तो, दिए गए बाइनरी नंबर का मान है -

1 + 2 + 4 + 8 +16 + 32 + 64 + 128 = 255

जो कि 2 8 - 1 के समान है।

हेक्साडेसिमल नंबर सिस्टम

हेक्साडेसिमल संख्या प्रणाली आधार 16 का उपयोग करती है। इस प्रणाली के अंक 0 से 15. तक होते हैं। कन्वेंशन द्वारा, F के माध्यम से अक्षर A का उपयोग दशमलव मानों 10 के 15 के अनुसार हेक्साडेसिमल अंकों का प्रतिनिधित्व करने के लिए किया जाता है।

कंप्यूटिंग में हेक्साडेसिमल संख्याओं का उपयोग लंबी बाइनरी अभ्यावेदन को संक्षिप्त करने के लिए किया जाता है। मूल रूप से, हेक्साडेसिमल संख्या प्रणाली प्रत्येक बाइट को आधे में विभाजित करके और प्रत्येक आधे-बाइट के मूल्य को व्यक्त करके एक बाइनरी डेटा का प्रतिनिधित्व करती है। निम्न तालिका दशमलव, बाइनरी और हेक्साडेसिमल समकक्ष प्रदान करती है -

दशमलव संख्या बाइनरी प्रतिनिधित्व हेक्साडेसिमल प्रतिनिधित्व
0 0 0
1 1 1
2 10 2
3 1 1 3
4 100 4
5 101 5
6 110 6
7 111 7
8 1000 8
9 1001 9
10 1010
1 1 1011
12 1100 सी
13 1101
14 1110
15 1111 एफ

बाइनरी नंबर को उसके हेक्साडेसिमल समतुल्य में परिवर्तित करने के लिए, इसे 4 लगातार समूहों के समूहों में तोड़ें, दाईं ओर से शुरू करके, और उन समूहों को हेक्साडेसिमल संख्या के संबंधित अंकों पर लिखें।

Example - बाइनरी नंबर 1000 1100 1101 0001 हेक्साडेसिमल के बराबर है - 8CD1

हेक्साडेसिमल संख्या को बाइनरी में बदलने के लिए, बस प्रत्येक हेक्साडेसिमल अंक को उसके 4-अंकीय बाइनरी समकक्ष में लिखें।

Example - हेक्साडेसिमल संख्या FAD8 बाइनरी के बराबर है - 1111 1010 1101 1000

द्विआधारी अंकगणित

निम्न तालिका बाइनरी जोड़ के लिए चार सरल नियमों को दर्शाती है -

(मैं) (Ii) (Iii) (Iv)
1
0 1 1 1
+0 +0 +1 +1
= 0 = 1 = 10 = 11

नियम (iii) और (iv) अगली बाईं स्थिति में 1-बिट का कैरी दिखाते हैं।

Example

दशमलव बायनरी
60 00111100
42 00101010
102 01100110

एक नकारात्मक बाइनरी मूल्य में व्यक्त किया गया है two's complement notation। इस नियम के अनुसार, एक द्विआधारी संख्या को अपने नकारात्मक मूल्य में बदलने के लिए अपने बिट मूल्यों को उल्टा करना और 1 जोड़ना है

Example

संख्या 53 00110101
बिट्स उल्टा 11001010
1 जोड़ें 0000000 1
क्रमांक -53 11001011

एक मान को दूसरे से घटाने के लिए, संख्या को घटाकर दो के पूरक प्रारूप में परिवर्तित करें और संख्याओं को जोड़ें

Example

53 से 42 घटाएं

संख्या 53 00110101
संख्या 42 00101010
42 के बिट्स को उल्टा करें 11010101
1 जोड़ें 0000000 1
संख्या -42 11010110
53 - 42 = 11 00001011

पिछले 1 बिट का अतिप्रवाह खो गया है।

मेमोरी में डेटा को संबोधित करना

वह प्रक्रिया जिसके माध्यम से प्रोसेसर निर्देशों के निष्पादन को नियंत्रित करता है, को संदर्भित किया जाता है fetch-decode-execute cycle या execution cycle। इसमें तीन निरंतर चरण होते हैं -

  • स्मृति से निर्देश प्राप्त करना
  • निर्देश को डिकोड करना या पहचानना
  • निर्देश को निष्पादित करना

प्रोसेसर एक बार में एक या अधिक बाइट्स मेमोरी तक पहुँच सकता है। आइए एक हेक्साडेसिमल संख्या 0725H पर विचार करें। इस संख्या में दो बाइट्स की आवश्यकता होगी। हाई-ऑर्डर बाइट या सबसे महत्वपूर्ण बाइट 07 है और लो-ऑर्डर बाइट 25 है।

प्रोसेसर रिवर्स-बाइट अनुक्रम में डेटा संग्रहीत करता है, अर्थात, कम-ऑर्डर बाइट को कम मेमोरी पते और उच्च-ऑर्डर बाइट में उच्च मेमोरी पते में संग्रहीत किया जाता है। इसलिए, यदि प्रोसेसर रजिस्टर से मेमोरी में 0725H मान लाता है, तो यह पहले 25 से कम मेमोरी एड्रेस और 07 से अगले मेमोरी एड्रेस पर ट्रांसफर कर देगा।

x: मेमोरी पता

जब प्रोसेसर मेमोरी से रजिस्टर करने के लिए संख्यात्मक डेटा प्राप्त करता है, तो यह फिर से बाइट्स को उलट देता है। स्मृति पते दो प्रकार के होते हैं -

  • पूर्ण पता - विशिष्ट स्थान का एक सीधा संदर्भ।

  • खंड पता (या ऑफ़सेट) - ऑफ़सेट मान के साथ एक मेमोरी सेगमेंट का पता शुरू करना।

स्थानीय पर्यावरण सेटअप

असेंबली भाषा इंस्ट्रक्टर सेट और प्रोसेसर के आर्किटेक्चर पर निर्भर है। इस ट्यूटोरियल में, हम पेंटियम जैसे इंटेल -32 प्रोसेसर पर ध्यान केंद्रित करते हैं। इस ट्यूटोरियल का अनुसरण करने के लिए, आपको आवश्यकता होगी -

  • एक IBM PC या कोई समतुल्य संगत कंप्यूटर
  • लिनक्स ऑपरेटिंग सिस्टम की एक प्रति
  • NASM कोडांतरक कार्यक्रम की एक प्रति

कई अच्छे कोडांतरक कार्यक्रम हैं, जैसे -

  • Microsoft असेंबलर (MASM)
  • बोरलैंड टर्बो असेंबलर (TASM)
  • GNU असेंबलर (GAS)

हम NASM कोडांतरक का उपयोग करेंगे, जैसा कि यह है -

  • नि: शुल्क। आप इसे विभिन्न वेब स्रोतों से डाउनलोड कर सकते हैं।
  • अच्छी तरह से प्रलेखित और आपको नेट पर बहुत सारी जानकारी मिलेगी।
  • लिनक्स और विंडोज दोनों पर इस्तेमाल किया जा सकता है।

एनएएसएम स्थापित कर रहा है

यदि आप लिनक्स स्थापित करते समय "डेवलपमेंट टूल्स" का चयन करते हैं, तो आपको लिनक्स ऑपरेटिंग सिस्टम के साथ एनएएसएम स्थापित हो सकता है और आपको इसे अलग से डाउनलोड और इंस्टॉल करने की आवश्यकता नहीं है। यह जाँचने के लिए कि क्या आपके पास पहले से NASM स्थापित है, निम्नलिखित कदम उठाएँ -

  • लिनक्स टर्मिनल खोलें।

  • प्रकार whereis nasm और ENTER दबाएँ।

  • यदि यह पहले से स्थापित है, तो nasm , / usr / bin / nasm जैसी एक रेखा दिखाई देती है। अन्यथा, आप बस nasm देखेंगे : तो आपको एनएएसएम स्थापित करने की आवश्यकता है।

एनएएसएम स्थापित करने के लिए, निम्नलिखित कदम उठाएं -

  • चेक netwide कोडांतरक (NASM) नवीनतम संस्करण के लिए वेबसाइट।

  • लिनक्स स्रोत संग्रह डाउनलोड करें nasm-X.XX.ta.gz, जहां संग्रह X.XXमें NASM संस्करण संख्या है।

  • एक निर्देशिका में संग्रह को अनपैक करें जो एक उपनिर्देशिका बनाता है nasm-X. XX

  • cd to nasm-X.XXऔर टाइप करें./configure। यह शेल स्क्रिप्ट मेकफाइल्स का उपयोग करने और उसके अनुसार सेट करने के लिए सबसे अच्छा सी कंपाइलर मिलेगा।

  • प्रकार make nasm और ndisasm बायनेरिज़ का निर्माण करना।

  • प्रकार make install nasm और ndisasm को / usr / लोकल / बिन में और मैन पेजों को स्थापित करने के लिए।

यह आपके सिस्टम पर एनएएसएम स्थापित करना चाहिए। वैकल्पिक रूप से, आप फेडोरा लिनक्स के लिए RPM वितरण का उपयोग कर सकते हैं। यह संस्करण स्थापित करने के लिए सरल है, बस RPM फ़ाइल पर डबल-क्लिक करें।

एक विधानसभा कार्यक्रम को तीन वर्गों में विभाजित किया जा सकता है -

  • data अनुभाग,

  • bss अनुभाग, और

  • text अनुभाग।

डेटा धारा

dataखंड का उपयोग प्रारंभिक डेटा या स्थिरांक घोषित करने के लिए किया जाता है। यह डेटा रनटाइम पर नहीं बदलता है। आप इस खंड में विभिन्न स्थिर मूल्यों, फ़ाइल नाम या बफर आकार, आदि की घोषणा कर सकते हैं।

डेटा अनुभाग घोषित करने का सिंटैक्स है -

section.data

बीएसएस धारा

bssखंड का उपयोग चर घोषित करने के लिए किया जाता है। Bss सेक्शन घोषित करने का सिंटैक्स है -

section.bss

पाठ अनुभाग

textअनुभाग का उपयोग वास्तविक कोड रखने के लिए किया जाता है। यह खंड घोषणा के साथ शुरू होना चाहिएglobal _start, जो कर्नेल को बताता है जहां प्रोग्राम का निष्पादन शुरू होता है।

पाठ अनुभाग घोषित करने के लिए वाक्य रचना है -

section.text
   global _start
_start:

टिप्पणियाँ

असेंबली भाषा की टिप्पणी एक अर्धविराम (;) से शुरू होती है। इसमें रिक्त सहित कोई भी मुद्रण योग्य वर्ण हो सकता है। यह अपने आप ही एक लाइन पर दिखाई दे सकता है, जैसे -

; This program displays a message on screen

या, एक निर्देश के साथ एक ही लाइन पर, जैसे -

add eax, ebx     ; adds ebx to eax

विधानसभा भाषा विवरण

असेंबली भाषा के कार्यक्रमों में तीन प्रकार के कथन होते हैं -

  • निष्पादन योग्य निर्देश या निर्देश,
  • असेंबलर निर्देश या छद्म ऑप्स, और
  • Macros.

executable instructions या केवल instructionsप्रोसेसर को बताएं कि क्या करना है। प्रत्येक निर्देश में एक होता हैoperation code(Opcode)। प्रत्येक निष्पादन योग्य निर्देश एक मशीन भाषा निर्देश उत्पन्न करता है।

assembler directives या pseudo-opsअसेंबलर को विधानसभा प्रक्रिया के विभिन्न पहलुओं के बारे में बताएं। ये गैर-निष्पादन योग्य हैं और मशीन भाषा निर्देश उत्पन्न नहीं करते हैं।

Macros मूल रूप से एक पाठ प्रतिस्थापन तंत्र हैं।

असेंबली लैंग्वेज स्टेटमेंट्स का सिंटैक्स

असेंबली भाषा के कथन प्रति पंक्ति एक कथन में दर्ज किए जाते हैं। प्रत्येक कथन निम्न प्रारूप का अनुसरण करता है -

[label]   mnemonic   [operands]   [;comment]

वर्ग कोष्ठक के क्षेत्र वैकल्पिक हैं। एक बुनियादी निर्देश के दो भाग होते हैं, पहला एक निर्देश (या mnemonic) का नाम है, जिसे निष्पादित किया जाना है, और दूसरा ऑपरेंड या कमांड के पैरामीटर हैं।

विधानसभा भाषा के विशिष्ट विवरणों के कुछ उदाहरण निम्नलिखित हैं -

INC COUNT        ; Increment the memory variable COUNT

MOV TOTAL, 48    ; Transfer the value 48 in the 
                 ; memory variable TOTAL
					  
ADD AH, BH       ; Add the content of the 
                 ; BH register into the AH register
					  
AND MASK1, 128   ; Perform AND operation on the 
                 ; variable MASK1 and 128
					  
ADD MARKS, 10    ; Add 10 to the variable MARKS
MOV AL, 10       ; Transfer the value 10 to the AL register

हेलो वर्ल्ड प्रोग्राम इन असेंबली

निम्नलिखित विधानसभा भाषा कोड स्क्रीन पर स्ट्रिंग 'हैलो वर्ल्ड' प्रदर्शित करता है -

section	.text
   global _start     ;must be declared for linker (ld)
	
_start:	            ;tells linker entry point
   mov	edx,len     ;message length
   mov	ecx,msg     ;message to write
   mov	ebx,1       ;file descriptor (stdout)
   mov	eax,4       ;system call number (sys_write)
   int	0x80        ;call kernel
	
   mov	eax,1       ;system call number (sys_exit)
   int	0x80        ;call kernel

section	.data
msg db 'Hello, world!', 0xa  ;string to be printed
len equ $ - msg     ;length of the string

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello, world!

NASM में असेंबली प्रोग्राम को संकलित करना और जोड़ना

सुनिश्चित करें कि आपने का रास्ता तय कर लिया है nasm तथा ldआपके पैथ वातावरण चर में बायनेरिज़। अब, उपरोक्त कार्यक्रम को संकलित करने और जोड़ने के लिए निम्नलिखित कदम उठाएँ -

  • एक पाठ संपादक का उपयोग करके उपरोक्त कोड टाइप करें और इसे hello.asm के रूप में सहेजें।

  • सुनिश्चित करें कि आप उसी निर्देशिका में हैं जहाँ आपने बचाया था hello.asm

  • प्रोग्राम को इकट्ठा करने के लिए, टाइप करें nasm -f elf hello.asm

  • यदि कोई त्रुटि है, तो आपको इस स्तर पर इसके बारे में संकेत दिया जाएगा। अन्यथा, आपके प्रोग्राम नाम की एक ऑब्जेक्ट फ़ाइलhello.o उत्पन्न होगा।

  • ऑब्जेक्ट फ़ाइल को लिंक करने के लिए और हैलो नामक एक निष्पादन योग्य फ़ाइल बनाएं ld -m elf_i386 -s -o hello hello.o

  • टाइप करके कार्यक्रम निष्पादित करें ./hello

यदि आपने सब कुछ सही ढंग से किया है, तो यह 'हैलो, दुनिया!' स्क्रीन पर।

हम पहले से ही एक विधानसभा कार्यक्रम के तीन वर्गों पर चर्चा कर चुके हैं। ये खंड विभिन्न स्मृति खंडों का भी प्रतिनिधित्व करते हैं।

दिलचस्प बात यह है कि यदि आप खंड कीवर्ड को खंड के साथ बदलते हैं, तो आपको वही परिणाम मिलेगा। निम्नलिखित कोड का प्रयास करें -

segment .text	   ;code segment
   global _start    ;must be declared for linker 
	
_start:	           ;tell linker entry point
   mov edx,len	   ;message length
   mov ecx,msg     ;message to write
   mov ebx,1	   ;file descriptor (stdout)
   mov eax,4	   ;system call number (sys_write)
   int 0x80	   ;call kernel

   mov eax,1       ;system call number (sys_exit)
   int 0x80	   ;call kernel

segment .data      ;data segment
msg	db 'Hello, world!',0xa   ;our dear string
len	equ	$ - msg          ;length of our dear string

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello, world!

मेमोरी सेगमेंट

एक खंडित स्मृति मॉडल, खंड रजिस्टरों में स्थित पॉइंटर्स द्वारा संदर्भित स्वतंत्र खंडों के समूह में सिस्टम मेमोरी को विभाजित करता है। प्रत्येक खंड का उपयोग एक विशिष्ट प्रकार के डेटा को रखने के लिए किया जाता है। एक खंड का उपयोग निर्देश कोड रखने के लिए किया जाता है, एक अन्य खंड डेटा तत्वों को संग्रहीत करता है, और एक तीसरा खंड कार्यक्रम को स्टैक रखता है।

उपरोक्त चर्चा के आलोक में, हम विभिन्न स्मृति खंडों को निर्दिष्ट कर सकते हैं -

  • Data segment - इसके द्वारा दर्शाया गया है .data अनुभाग और द .bss। .Data अनुभाग का उपयोग मेमोरी क्षेत्र को घोषित करने के लिए किया जाता है, जहां डेटा तत्वों को प्रोग्राम के लिए संग्रहीत किया जाता है। डेटा तत्वों के घोषित होने के बाद इस खंड का विस्तार नहीं किया जा सकता है, और यह पूरे कार्यक्रम में स्थिर रहता है।

    .Bs वर्ग भी एक स्थिर मेमोरी सेक्शन है जिसमें प्रोग्राम के लिए बाद में घोषित किए जाने वाले डेटा के बफ़र्स शामिल होते हैं। यह बफ़र मेमोरी शून्य-भरी है।

  • Code segment - इसके द्वारा दर्शाया गया है .textअनुभाग। यह स्मृति में एक क्षेत्र को परिभाषित करता है जो अनुदेश कोड को संग्रहीत करता है। यह भी एक निश्चित क्षेत्र है।

  • Stack - इस सेगमेंट में प्रोग्राम के भीतर फंक्शन्स और प्रक्रियाओं के लिए डेटा वैल्यूज़ शामिल हैं।

प्रोसेसर ऑपरेशन में ज्यादातर प्रोसेसिंग डेटा शामिल होता है। इस डेटा को मेमोरी में स्टोर किया जा सकता है और इसे वहां से एक्सेस किया जा सकता है। हालाँकि, डेटा को पढ़ने और मेमोरी में डेटा स्टोर करने से प्रोसेसर धीमा हो जाता है, क्योंकि इसमें कंट्रोल बस में और मेमोरी स्टोरेज यूनिट में डेटा अनुरोध भेजने और उसी चैनल के माध्यम से डेटा प्राप्त करने की जटिल प्रक्रियाएँ शामिल होती हैं।

प्रोसेसर संचालन को गति देने के लिए, प्रोसेसर में कुछ आंतरिक मेमोरी स्टोरेज स्थान शामिल हैं, जिन्हें कहा जाता है registers

रजिस्टर स्मृति तक पहुँचने के लिए बिना प्रसंस्करण के लिए डेटा तत्वों को संग्रहीत करता है। प्रोसेसर चिप में सीमित संख्या में रजिस्टरों का निर्माण किया जाता है।

प्रोसेसर रजिस्टर

IA-32 आर्किटेक्चर में दस 32-बिट और छह 16-बिट प्रोसेसर रजिस्टर हैं। रजिस्टर को तीन श्रेणियों में बांटा गया है -

  • सामान्य रजिस्टर,
  • नियंत्रण रजिस्टर, और
  • खंड रजिस्टर।

सामान्य रजिस्टरों को निम्नलिखित समूहों में विभाजित किया गया है -

  • डेटा रजिस्टर,
  • सूचक रजिस्टर, और
  • सूचकांक रजिस्टर।

डेटा रजिस्टर

चार 32-बिट डेटा रजिस्टर का उपयोग अंकगणित, तार्किक और अन्य कार्यों के लिए किया जाता है। इन 32-बिट रजिस्टरों का उपयोग तीन तरीकों से किया जा सकता है -

  • पूर्ण 32-बिट डेटा रजिस्टर के रूप में: EAX, EBX, ECX, EDX।

  • 32-बिट रजिस्टरों के निचले हिस्सों का उपयोग चार 16-बिट डेटा रजिस्टरों के रूप में किया जा सकता है: AX, BX, CX और DX।

  • उपर्युक्त चार 16-बिट रजिस्टरों के निचले और उच्चतर हिस्सों को आठ 8-बिट डेटा रजिस्टर के रूप में इस्तेमाल किया जा सकता है: एएच, एएल, बीएच, बीएल, सीएच, सीएल, डीएच और डीएल।

इन डेटा रजिस्टरों में से कुछ अंकगणितीय संचालन में विशिष्ट उपयोग हैं।

AX is the primary accumulator; इसका उपयोग इनपुट / आउटपुट और अधिकांश अंकगणितीय निर्देशों में किया जाता है। उदाहरण के लिए, गुणन ऑपरेशन में, एक ऑपरेंड को EAX या AX या AL रजिस्टर में संग्रहीत किया जाता है, जो ऑपरेंड के आकार के अनुसार होता है।

BX is known as the base register, क्योंकि इसका उपयोग अनुक्रमित संबोधन में किया जा सकता है।

CX is known as the count register, ECX के रूप में, सीएक्स रजिस्टर्स लूप काउंट को पुनरावृत्त संचालन में संग्रहीत करते हैं।

DX is known as the data register। इसका उपयोग इनपुट / आउटपुट संचालन में भी किया जाता है। इसका उपयोग एक्सएक्स रजिस्टर के साथ-साथ डीएक्स के साथ बड़े पैमाने पर बहुगुणित और विभाजित संचालन के लिए भी किया जाता है।

सूचक रजिस्टर

पॉइंटर रजिस्टर 32-बिट ईआईपी, ईएसपी और ईबीपी रजिस्टर और संबंधित 16-बिट सही भाग आईपी, एसपी और बीपी हैं। सूचक रजिस्टर की तीन श्रेणियां हैं -

  • Instruction Pointer (IP)- 16-बिट आईपी रजिस्टर निष्पादित होने वाले अगले निर्देश के ऑफसेट पते को संग्रहीत करता है। सीएस रजिस्टर के साथ आईपी (सीएस के रूप में: आईपी) कोड सेगमेंट में वर्तमान निर्देश का पूरा पता देता है।

  • Stack Pointer (SP)- 16-बिट एसपी रजिस्टर प्रोग्राम स्टैक के भीतर ऑफसेट मूल्य प्रदान करता है। SS रजिस्टर के साथ SP (SS: SP) प्रोग्राम स्टैक के भीतर डेटा या एड्रेस की वर्तमान स्थिति को संदर्भित करता है।

  • Base Pointer (BP)- 16-बिट बीपी रजिस्टर मुख्य रूप से सबरूटीन को पारित पैरामीटर चर को संदर्भित करने में मदद करता है। एसएस रजिस्टर में पता पैरामीटर के स्थान को प्राप्त करने के लिए बीपी में ऑफसेट के साथ संयुक्त है। बीपी को विशेष पते के लिए आधार रजिस्टर के रूप में डीआई और एसआई के साथ भी जोड़ा जा सकता है।

सूचकांक रजिस्टर

32-बिट इंडेक्स रजिस्टर, ईएसआई और ईडीआई, और उनके 16-बिट दाहिने हिस्से। SI और DI, का उपयोग अनुक्रमित पते के लिए किया जाता है और कभी-कभी इसके अलावा और घटाव में उपयोग किया जाता है। सूचकांक सूचक के दो सेट हैं -

  • Source Index (SI) - यह स्ट्रिंग संचालन के लिए स्रोत सूचकांक के रूप में उपयोग किया जाता है।

  • Destination Index (DI) - इसे स्ट्रिंग ऑपरेशंस के लिए डेस्टिनेशन इंडेक्स के रूप में इस्तेमाल किया जाता है।

नियंत्रण रजिस्टर

32-बिट इंस्ट्रक्शन पॉइंटर रजिस्टर और 32-बिट फ्लैग रजिस्टर को संयुक्त रूप से कंट्रोल रजिस्टर माना जाता है।

कई निर्देशों में तुलना और गणितीय गणना शामिल है और झंडे की स्थिति को बदलते हैं और कुछ अन्य सशर्त निर्देश नियंत्रण प्रवाह को अन्य स्थान पर ले जाने के लिए इन स्थिति झंडे के मूल्य का परीक्षण करते हैं।

सामान्य ध्वज बिट्स हैं:

  • Overflow Flag (OF) - यह एक हस्ताक्षरित अंकगणितीय ऑपरेशन के बाद डेटा के उच्च-क्रम बिट (सबसे बाईं ओर) के अतिप्रवाह को इंगित करता है।

  • Direction Flag (DF)- यह स्ट्रिंग डेटा को ले जाने या तुलना करने के लिए बाईं या दाईं दिशा निर्धारित करता है। जब DF मान 0 होता है, तो स्ट्रिंग ऑपरेशन बाएँ-से-दाएँ दिशा लेता है और जब मान 1 पर सेट होता है, तो स्ट्रिंग ऑपरेशन दाएँ-से-बाएँ दिशा लेता है।

  • Interrupt Flag (IF)- यह निर्धारित करता है कि क्या बाहरी अवरोध जैसे कीबोर्ड प्रविष्टि, आदि को अनदेखा या संसाधित किया जाना है। जब मान 0 होता है तो यह बाहरी व्यवधान को निष्क्रिय कर देता है और 1 पर सेट होने पर व्यवधान को सक्षम बनाता है।

  • Trap Flag (TF)- यह एकल-चरण मोड में प्रोसेसर के संचालन को स्थापित करने की अनुमति देता है। DEBUG प्रोग्राम का उपयोग हमने ट्रैप फ्लैग सेट किया है, इसलिए हम एक समय में एक निर्देश के निष्पादन के माध्यम से कदम बढ़ा सकते हैं।

  • Sign Flag (SF)- यह एक अंकगणितीय ऑपरेशन के परिणाम का संकेत दिखाता है। यह ध्वज अंकगणितीय ऑपरेशन के बाद डेटा आइटम के संकेत के अनुसार सेट किया गया है। संकेत सबसे बाईं ओर के उच्च-क्रम द्वारा दर्शाया गया है। एक सकारात्मक परिणाम SF के मान को 0 पर और नकारात्मक परिणाम को 1 पर सेट करता है।

  • Zero Flag (ZF)- यह एक अंकगणित या तुलना ऑपरेशन के परिणाम को इंगित करता है। एक नॉनज़रो परिणाम शून्य ध्वज को 0 पर साफ़ करता है, और एक शून्य परिणाम इसे 1 पर सेट करता है।

  • Auxiliary Carry Flag (AF)- इसमें अंकगणितीय ऑपरेशन के बाद बिट 3 से बिट 4 तक कैरी होता है; विशेष अंकगणित के लिए उपयोग किया जाता है। AF तब सेट होता है जब 1-बाइट अंकगणितीय ऑपरेशन के कारण बिट 3 से बिट 4 में कैरी होता है।

  • Parity Flag (PF)- यह अंकगणितीय ऑपरेशन से प्राप्त परिणाम में कुल 1-बिट्स को इंगित करता है। 1-बिट्स की एक समान संख्या समता ध्वज को 0 पर साफ़ करती है और 1-बिट्स की विषम संख्या समता ध्वज को 1 पर सेट करती है।

  • Carry Flag (CF)- इसमें अंकगणित ऑपरेशन के बाद एक उच्च-क्रम बिट (सबसे बाएं) से 0 या 1 का कैरी होता है। यह अंतिम बिट के शिफ्ट या रोटेट ऑपरेशन की सामग्री को भी संग्रहीत करता है ।

निम्न तालिका 16-बिट फ़्लैग्स रजिस्टर में ध्वज बिट्स की स्थिति को इंगित करती है:

झंडा: हे मैं टी रों जेड पी सी
बिट नं: 15 14 13 12 1 1 10 9 8 7 6 5 4 3 2 1 0

खंड रजिस्टर

सेगमेंट डेटा, कोड और स्टैक युक्त प्रोग्राम में परिभाषित विशिष्ट क्षेत्र हैं। तीन मुख्य खंड हैं -

  • Code Segment- इसमें सभी निर्देशों को निष्पादित किया जाना है। 16-बिट कोड सेगमेंट रजिस्टर या CS रजिस्टर कोड सेगमेंट के शुरुआती पते को संग्रहीत करता है।

  • Data Segment- इसमें डेटा, स्थिरांक और कार्य क्षेत्र शामिल हैं। एक 16-बिट डेटा सेगमेंट रजिस्टर या डीएस रजिस्टर डेटा सेगमेंट के शुरुआती पते को स्टोर करता है।

  • Stack Segment- इसमें प्रक्रियाओं या सबरूटीन्स के डेटा और रिटर्न एड्रेस शामिल हैं। इसे 'स्टैक' डेटा संरचना के रूप में कार्यान्वित किया जाता है। स्टैक सेगमेंट रजिस्टर या एसएस रजिस्टर स्टैक के शुरुआती पते को संग्रहीत करता है।

डीएस, सीएस और एसएस रजिस्टर के अलावा, अन्य अतिरिक्त खंड रजिस्टर हैं - ईएस (अतिरिक्त खंड), एफएस और जीएस, जो डेटा संग्रहीत करने के लिए अतिरिक्त खंड प्रदान करते हैं।

असेंबली प्रोग्रामिंग में, मेमोरी स्थानों तक पहुंचने के लिए एक प्रोग्राम की आवश्यकता होती है। एक खंड के भीतर सभी मेमोरी स्थान खंड के शुरुआती पते के सापेक्ष हैं। एक खंड 16 या हेक्साडेसिमल द्वारा समान रूप से विभाज्य एक पते में शुरू होता है। इसलिए, ऐसे सभी मेमोरी पतों में सबसे सही हेक्स अंक 0 है, जो आमतौर पर सेगमेंट रजिस्टरों में संग्रहीत नहीं किया जाता है।

सेगमेंट रजिस्टर एक सेगमेंट के शुरुआती पतों को संग्रहीत करता है। एक खंड के भीतर डेटा या निर्देश का सटीक स्थान प्राप्त करने के लिए, एक ऑफसेट मूल्य (या विस्थापन) की आवश्यकता होती है। किसी सेगमेंट में किसी मेमोरी लोकेशन को रेफर करने के लिए, प्रोसेसर सेगमेंट रजिस्टर में सेगमेंट एड्रेस को लोकेशन की ऑफसेट वैल्यू के साथ जोड़ देता है।

उदाहरण

असेंबली प्रोग्रामिंग में रजिस्टरों के उपयोग को समझने के लिए निम्नलिखित सरल कार्यक्रम को देखें। यह कार्यक्रम एक साधारण संदेश के साथ स्क्रीन पर 9 सितारों को प्रदर्शित करता है -

section	.text
   global _start	 ;must be declared for linker (gcc)
	
_start:	         ;tell linker entry point
   mov	edx,len  ;message length
   mov	ecx,msg  ;message to write
   mov	ebx,1    ;file descriptor (stdout)
   mov	eax,4    ;system call number (sys_write)
   int	0x80     ;call kernel
	
   mov	edx,9    ;message length
   mov	ecx,s2   ;message to write
   mov	ebx,1    ;file descriptor (stdout)
   mov	eax,4    ;system call number (sys_write)
   int	0x80     ;call kernel
	
   mov	eax,1    ;system call number (sys_exit)
   int	0x80     ;call kernel
	
section	.data
msg db 'Displaying 9 stars',0xa ;a message
len equ $ - msg  ;length of message
s2 times 9 db '*'

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Displaying 9 stars
*********

सिस्टम कॉल यूजर स्पेस और कर्नेल स्पेस के बीच इंटरफेस के लिए एपीआई हैं। हमने सिस्टम कॉल का उपयोग पहले ही कर लिया है। sys_write और sys_exit, क्रमशः स्क्रीन में लिखने और कार्यक्रम से बाहर निकलने के लिए।

लिनक्स सिस्टम कॉल

आप अपने असेंबली प्रोग्राम में लिनक्स सिस्टम कॉल का उपयोग कर सकते हैं। अपने प्रोग्राम में लिनक्स सिस्टम कॉल का उपयोग करने के लिए आपको निम्नलिखित कदम उठाने होंगे -

  • EAX रजिस्टर में सिस्टम कॉल नंबर डालें।
  • रजिस्टर EBB, ECX, आदि में सिस्टम कॉल के तर्कों को स्टोर करें।
  • प्रासंगिक व्यवधान (80h) को बुलाओ।
  • परिणाम आमतौर पर EAX रजिस्टर में वापस आ जाता है।

छह रजिस्टर हैं जो सिस्टम कॉल के तर्कों का उपयोग करते हैं। ये EBX, ECX, EDX, ESI, EDI और EBP हैं। ये रजिस्टर EBX रजिस्टर से शुरू होकर लगातार तर्क देते हैं। यदि छह से अधिक तर्क हैं, तो पहले तर्क का मेमोरी स्थान EBX रजिस्टर में संग्रहीत किया जाता है।

निम्नलिखित कोड स्निपेट सिस्टम कॉल sys_exit का उपयोग दिखाता है -

mov	eax,1		; system call number (sys_exit)
int	0x80		; call kernel

निम्नलिखित कोड स्निपेट सिस्टम कॉल sys_write का उपयोग दिखाता है -

mov	edx,4		; message length
mov	ecx,msg		; message to write
mov	ebx,1		; file descriptor (stdout)
mov	eax,4		; system call number (sys_write)
int	0x80		; call kernel

सभी syscalls में सूचीबद्ध हैं /usr/include/asm/unistd.h , एक साथ उनकी संख्या (मूल्य EAX में डालने के लिए इससे पहले कि आप पूर्णांक 80h कहते हैं) के साथ।

निम्न तालिका इस ट्यूटोरियल में प्रयुक्त कुछ सिस्टम कॉल को दिखाती है -

% eax नाम % EBX % ECX % EDX % ESX % ईडीआई
1 sys_exit पूर्णांक - - - -
2 sys_fork संरचना pt_regs - - - -
3 sys_read अहस्ताक्षरित int चार * size_t - -
4 sys_write अहस्ताक्षरित int कास्ट चार * size_t - -
5 sys_open कास्ट चार * पूर्णांक पूर्णांक - -
6 sys_close अहस्ताक्षरित int - - - -

उदाहरण

निम्नलिखित उदाहरण कीबोर्ड से एक नंबर पढ़ता है और इसे स्क्रीन पर प्रदर्शित करता है -

section .data                           ;Data segment
   userMsg db 'Please enter a number: ' ;Ask the user to enter a number
   lenUserMsg equ $-userMsg             ;The length of the message
   dispMsg db 'You have entered: '
   lenDispMsg equ $-dispMsg                 

section .bss           ;Uninitialized data
   num resb 5
	
section .text          ;Code Segment
   global _start
	
_start:                ;User prompt
   mov eax, 4
   mov ebx, 1
   mov ecx, userMsg
   mov edx, lenUserMsg
   int 80h

   ;Read and store the user input
   mov eax, 3
   mov ebx, 2
   mov ecx, num  
   mov edx, 5          ;5 bytes (numeric, 1 for sign) of that information
   int 80h
	
   ;Output the message 'The entered number is: '
   mov eax, 4
   mov ebx, 1
   mov ecx, dispMsg
   mov edx, lenDispMsg
   int 80h  

   ;Output the number entered
   mov eax, 4
   mov ebx, 1
   mov ecx, num
   mov edx, 5
   int 80h  
    
   ; Exit code
   mov eax, 1
   mov ebx, 0
   int 80h

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Please enter a number:
1234  
You have entered:1234

अधिकांश असेंबली भाषा निर्देशों के लिए ऑपरेंड को संसाधित करने की आवश्यकता होती है। एक ऑपरेंड पता स्थान प्रदान करता है, जहां संसाधित होने वाला डेटा संग्रहीत होता है। कुछ निर्देशों के लिए एक ऑपरेंड की आवश्यकता नहीं होती है, जबकि कुछ अन्य निर्देशों के लिए एक, दो या तीन ऑपरेंड की आवश्यकता हो सकती है।

जब एक निर्देश में दो ऑपरेंड की आवश्यकता होती है, तो पहला ऑपरेंड आमतौर पर गंतव्य होता है, जिसमें एक रजिस्टर या मेमोरी लोकेशन में डेटा होता है और दूसरा ऑपरेंड स्रोत होता है। स्रोत में या तो दिया जाने वाला डेटा (तत्काल पता) या डेटा का पता (रजिस्टर या मेमोरी में) होता है। आमतौर पर, ऑपरेशन के बाद स्रोत डेटा अनछुए रहते हैं।

संबोधन के तीन मूल तरीके हैं -

  • पता दर्ज करना
  • तत्काल संबोधन
  • स्मृति संबोधन

पंजीकरण पता

इस एड्रेसिंग मोड में, एक रजिस्टर में ऑपरेंड होता है। निर्देश के आधार पर, रजिस्टर पहला ऑपरेंड, दूसरा ऑपरेंड या दोनों हो सकता है।

उदाहरण के लिए,

MOV DX, TAX_RATE   ; Register in first operand
MOV COUNT, CX	   ; Register in second operand
MOV EAX, EBX	   ; Both the operands are in registers

चूंकि रजिस्टरों के बीच डाटा प्रोसेसिंग में मेमोरी शामिल नहीं है, यह डेटा का सबसे तेज प्रसंस्करण प्रदान करता है।

तत्काल संबोधन

एक तत्काल ऑपरेंड में एक निरंतर मूल्य या एक अभिव्यक्ति है। जब दो ऑपरेंड के साथ एक निर्देश तत्काल एड्रेसिंग का उपयोग करता है, तो पहला ऑपरेंड एक रजिस्टर या मेमोरी लोकेशन हो सकता है, और दूसरा ऑपरेंड एक तत्काल स्थिरांक है। पहला ऑपरेंड डेटा की लंबाई को परिभाषित करता है।

उदाहरण के लिए,

BYTE_VALUE  DB  150    ; A byte value is defined
WORD_VALUE  DW  300    ; A word value is defined
ADD  BYTE_VALUE, 65    ; An immediate operand 65 is added
MOV  AX, 45H           ; Immediate constant 45H is transferred to AX

डायरेक्ट मेमोरी एड्रेसिंग

जब ऑपरेशन्स को मेमोरी एड्रेसिंग मोड में निर्दिष्ट किया जाता है, तो मुख्य मेमोरी, जो आमतौर पर डेटा सेगमेंट में होती है, तक सीधी पहुंच की आवश्यकता होती है। डेटा को धीमा करने के परिणामों को संबोधित करने का यह तरीका है। स्मृति में डेटा की सटीक स्थिति का पता लगाने के लिए, हमें खंड प्रारंभ पते की आवश्यकता होती है, जो आमतौर पर डीएस रजिस्टर और एक ऑफसेट मूल्य में पाया जाता है। इस ऑफसेट मान को भी कहा जाता हैeffective address

प्रत्यक्ष संबोधित मोड में, ऑफसेट मूल्य को सीधे निर्देश के भाग के रूप में निर्दिष्ट किया जाता है, आमतौर पर चर नाम से इंगित किया जाता है। कोडांतरक ऑफसेट मान की गणना करता है और एक प्रतीक तालिका रखता है, जो प्रोग्राम में उपयोग किए जाने वाले सभी चर के ऑफसेट मूल्यों को संग्रहीत करता है।

डायरेक्ट मेमोरी एड्रेसिंग में, ऑपरेंड में से एक मेमोरी लोकेशन को संदर्भित करता है और दूसरा ऑपरेंड एक रजिस्टर को संदर्भित करता है।

उदाहरण के लिए,

ADD	BYTE_VALUE, DL	; Adds the register in the memory location
MOV	BX, WORD_VALUE	; Operand from the memory is added to register

डायरेक्ट-ऑफसेट एड्रेसिंग

यह एड्रेसिंग मोड किसी पते को संशोधित करने के लिए अंकगणितीय ऑपरेटरों का उपयोग करता है। उदाहरण के लिए, डेटा की तालिकाओं को परिभाषित करने वाली निम्नलिखित परिभाषाओं को देखें -

BYTE_TABLE DB  14, 15, 22, 45      ; Tables of bytes
WORD_TABLE DW  134, 345, 564, 123  ; Tables of words

निम्नलिखित ऑपरेशन मेमोरी में तालिकाओं से डेटा को रजिस्टरों में एक्सेस करते हैं -

MOV CL, BYTE_TABLE[2]	; Gets the 3rd element of the BYTE_TABLE
MOV CL, BYTE_TABLE + 2	; Gets the 3rd element of the BYTE_TABLE
MOV CX, WORD_TABLE[3]	; Gets the 4th element of the WORD_TABLE
MOV CX, WORD_TABLE + 3	; Gets the 4th element of the WORD_TABLE

अप्रत्यक्ष मेमोरी एड्रेसिंग

यह एड्रेसिंग मोड कंप्यूटर की सेगमेंट की क्षमता : ऑफ़सेट एड्रेसिंग का उपयोग करता है। आमतौर पर, बेस रेबीज ईबीएक्स, ईबीपी (या बीएक्स, बीपी) और इंडेक्स रजिस्टरों (डीआई, एसआई) को मेमोरी संदर्भ के लिए वर्ग कोष्ठक के भीतर कोडित किया जाता है, इस उद्देश्य के लिए उपयोग किया जाता है।

अप्रत्यक्ष रूप से संबोधित किया जाता है, जिसका उपयोग चर जैसे कई तत्वों, सरणियों के लिए किया जाता है। EBX रजिस्टर में सरणी का पता शुरू करते हुए स्टोर किया जाता है।

निम्न कोड स्निपेट दिखाता है कि चर के विभिन्न तत्वों को कैसे एक्सेस किया जाए।

MY_TABLE TIMES 10 DW 0  ; Allocates 10 words (2 bytes) each initialized to 0
MOV EBX, [MY_TABLE]     ; Effective Address of MY_TABLE in EBX
MOV [EBX], 110          ; MY_TABLE[0] = 110
ADD EBX, 2              ; EBX = EBX +2
MOV [EBX], 123          ; MY_TABLE[1] = 123

MOV निर्देश

हमने पहले ही MOV निर्देश का उपयोग किया है जो एक भंडारण स्थान से दूसरे में डेटा स्थानांतरित करने के लिए उपयोग किया जाता है। एमओवी निर्देश दो ऑपरेंड लेता है।

वाक्य - विन्यास

MOV निर्देश का सिंटैक्स है -

MOV  destination, source

MOV निर्देश में निम्नलिखित पाँच रूपों में से एक हो सकता है -

MOV  register, register
MOV  register, immediate
MOV  memory, immediate
MOV  register, memory
MOV  memory, register

कृपया ध्यान दें कि -

  • MOV ऑपरेशन में दोनों ऑपरेंड एक ही आकार के होने चाहिए
  • स्रोत ऑपरेंड का मान अपरिवर्तित रहता है

MOV निर्देश कई बार अस्पष्टता का कारण बनता है। उदाहरण के लिए, कथनों को देखें -

MOV  EBX, [MY_TABLE]  ; Effective Address of MY_TABLE in EBX
MOV  [EBX], 110	      ; MY_TABLE[0] = 110

यह स्पष्ट नहीं है कि आप एक बाइट समकक्ष या संख्या 110 के बराबर शब्द को स्थानांतरित करना चाहते हैं। ऐसे मामलों में, बुद्धिमान का उपयोग करना बुद्धिमानी है type specifier

निम्नलिखित तालिका सामान्य प्रकार के कुछ विशेष प्रकार दिखाती है -

स्पेसिफायर टाइप करें बाइट्स को संबोधित किया
बाइट 1
शब्द 2
DWORD 4
QWORD 8
TBYTE 10

उदाहरण

निम्नलिखित कार्यक्रम ऊपर चर्चा की गई कुछ अवधारणाओं को दिखाता है। यह मेमोरी के डेटा सेक्शन में एक नाम 'ज़ारा अली' को संग्रहीत करता है, फिर इसके मूल्य को दूसरे नाम 'नूहा अली' में प्रोग्रामेटिक रूप से बदलता है और दोनों नामों को प्रदर्शित करता है।

section	.text
   global _start     ;must be declared for linker (ld)
_start:             ;tell linker entry point
	
   ;writing the name 'Zara Ali'
   mov	edx,9       ;message length
   mov	ecx, name   ;message to write
   mov	ebx,1       ;file descriptor (stdout)
   mov	eax,4       ;system call number (sys_write)
   int	0x80        ;call kernel
	
   mov	[name],  dword 'Nuha'    ; Changed the name to Nuha Ali
	
   ;writing the name 'Nuha Ali'
   mov	edx,8       ;message length
   mov	ecx,name    ;message to write
   mov	ebx,1       ;file descriptor (stdout)
   mov	eax,4       ;system call number (sys_write)
   int	0x80        ;call kernel
	
   mov	eax,1       ;system call number (sys_exit)
   int	0x80        ;call kernel

section	.data
name db 'Zara Ali '

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Zara Ali Nuha Ali

NASM विभिन्न प्रदान करता है define directivesचर के लिए भंडारण स्थान के भंडारण के लिए। स्टोरेज स्पेस के आवंटन के लिए परिभाषित असेम्बलर निर्देश का उपयोग किया जाता है। इसे आरक्षित करने के साथ-साथ एक या अधिक बाइट्स को इनिशियलाइज़ करने के लिए भी इस्तेमाल किया जा सकता है।

आरंभिक डेटा के लिए भंडारण स्थान आवंटित करना

आरंभिक डेटा के लिए भंडारण आवंटन स्टेटमेंट का सिंटैक्स है -

[variable-name]    define-directive    initial-value   [,initial-value]...

जहां, चर-नाम प्रत्येक भंडारण स्थान के लिए पहचानकर्ता है। कोडांतरक डेटा सेगमेंट में परिभाषित प्रत्येक चर नाम के लिए एक ऑफसेट मूल्य को जोड़ता है।

परिभाषित निर्देश के पांच मूल रूप हैं -

आदेश उद्देश्य स्टोरेज की जगह
डाटाबेस बाइट को परिभाषित करें 1 बाइट आवंटित करता है
DW शब्द को परिभाषित करें 2 बाइट्स आवंटित करता है
डीडी डबलवर्ड को परिभाषित करें 4 बाइट्स आवंटित करता है
डीक्यू चतुर्भुज को परिभाषित करें 8 बाइट्स आवंटित करता है
डीटी दस बाइट को परिभाषित करें 10 बाइट्स आवंटित करता है

परिभाषित निर्देशों का उपयोग करने के कुछ उदाहरण निम्नलिखित हैं -

choice		DB	'y'
number		DW	12345
neg_number	DW	-12345
big_number	DQ	123456789
real_number1	DD	1.234
real_number2	DQ	123.456

कृपया ध्यान दें कि -

  • चरित्र के प्रत्येक बाइट को हेक्साडेसिमल में इसके ASCII मान के रूप में संग्रहीत किया जाता है।

  • प्रत्येक दशमलव मान स्वचालित रूप से अपने 16-बिट बाइनरी समकक्ष में परिवर्तित हो जाता है और एक हेक्साडेसिमल नंबर के रूप में संग्रहीत होता है।

  • प्रोसेसर छोटे-एंडियन बाइट ऑर्डरिंग का उपयोग करता है।

  • नकारात्मक संख्याएं इसके 2 के पूरक प्रतिनिधित्व में परिवर्तित हो जाती हैं।

  • छोटी और लंबी फ्लोटिंग-पॉइंट संख्याओं को क्रमशः 32 या 64 बिट्स का उपयोग करके दर्शाया गया है।

निम्नलिखित कार्यक्रम परिभाषित निर्देश का उपयोग दर्शाता है -

section .text
   global _start          ;must be declared for linker (gcc)
	
_start:                   ;tell linker entry point
   mov	edx,1		  ;message length
   mov	ecx,choice        ;message to write
   mov	ebx,1		  ;file descriptor (stdout)
   mov	eax,4		  ;system call number (sys_write)
   int	0x80		  ;call kernel

   mov	eax,1		  ;system call number (sys_exit)
   int	0x80		  ;call kernel

section .data
choice DB 'y'

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

y

Uninitialized डेटा के लिए संग्रहण स्थान आवंटित करना

आरक्षित निर्देशों का उपयोग असिंचित डेटा के लिए स्थान को आरक्षित करने के लिए किया जाता है। रिज़र्व निर्देश एक एकल ऑपरेंड लेते हैं जो आरक्षित होने के लिए अंतरिक्ष की इकाइयों की संख्या को निर्दिष्ट करता है। प्रत्येक परिभाषित निर्देश में एक संबंधित आरक्षित निर्देश होता है।

रिजर्व निर्देश के पांच मूल रूप हैं -

आदेश उद्देश्य
RESB एक बाइट रिजर्व करें
RESW एक शब्द आरक्षित करें
RESD एक डबलवर्ड आरक्षित करें
ResQ एक चतुर्भुज आरक्षित करें
आराम दस बाइट आरक्षित करें

एकाधिक परिभाषाएँ

आप एक प्रोग्राम में कई डेटा डेफिनेशन स्टेटमेंट रख सकते हैं। उदाहरण के लिए -

choice	  DB 	'Y' 		 ;ASCII of y = 79H
number1	  DW 	12345 	 ;12345D = 3039H
number2    DD  12345679  ;123456789D = 75BCD15H

कोडांतरक कई चर परिभाषाओं के लिए सन्निहित स्मृति आवंटित करता है।

एकाधिक आरंभ

TIMES निर्देश एक ही मूल्य के लिए कई आरंभीकरण की अनुमति देता है। उदाहरण के लिए, आकार 9 के नाम के एक सरणी को परिभाषित किया जा सकता है और निम्नलिखित कथन का उपयोग करके शून्य पर आरम्भ किया जा सकता है -

marks  TIMES  9  DW  0

TIMES निर्देश सरणियों और तालिकाओं को परिभाषित करने में उपयोगी है। निम्नलिखित कार्यक्रम स्क्रीन पर 9 तारांकन प्रदर्शित करता है -

section	.text
   global _start        ;must be declared for linker (ld)
	
_start:                 ;tell linker entry point
   mov	edx,9		;message length
   mov	ecx, stars	;message to write
   mov	ebx,1		;file descriptor (stdout)
   mov	eax,4		;system call number (sys_write)
   int	0x80		;call kernel

   mov	eax,1		;system call number (sys_exit)
   int	0x80		;call kernel

section	.data
stars   times 9 db '*'

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

*********

NASM द्वारा प्रदान किए गए कई निर्देश हैं जो स्थिरांक को परिभाषित करते हैं। हमने पिछले अध्यायों में EQU निर्देश का उपयोग किया है। हम विशेष रूप से तीन निर्देशों पर चर्चा करेंगे -

  • EQU
  • %assign
  • %define

एके निर्देश

EQUनिर्देशकों का उपयोग स्थिरांक को परिभाषित करने के लिए किया जाता है। EQU निर्देश का वाक्य विन्यास इस प्रकार है -

CONSTANT_NAME EQU expression

उदाहरण के लिए,

TOTAL_STUDENTS equ 50

फिर आप अपने कोड में इस निरंतर मूल्य का उपयोग कर सकते हैं, जैसे -

mov  ecx,  TOTAL_STUDENTS 
cmp  eax,  TOTAL_STUDENTS

EQU स्टेटमेंट का ऑपरेंड एक अभिव्यक्ति हो सकता है -

LENGTH equ 20
WIDTH  equ 10
AREA   equ length * width

कोड खंड के ऊपर 200 के रूप में क्षेत्र को परिभाषित करेगा।

उदाहरण

निम्न उदाहरण EQU निर्देश के उपयोग को दर्शाता है -

SYS_EXIT  equ 1
SYS_WRITE equ 4
STDIN     equ 0
STDOUT    equ 1
section	 .text
   global _start    ;must be declared for using gcc
	
_start:             ;tell linker entry point
   mov eax, SYS_WRITE         
   mov ebx, STDOUT         
   mov ecx, msg1         
   mov edx, len1 
   int 0x80                
	
   mov eax, SYS_WRITE         
   mov ebx, STDOUT         
   mov ecx, msg2         
   mov edx, len2 
   int 0x80 
	
   mov eax, SYS_WRITE         
   mov ebx, STDOUT         
   mov ecx, msg3         
   mov edx, len3 
   int 0x80
   
   mov eax,SYS_EXIT    ;system call number (sys_exit)
   int 0x80            ;call kernel

section	 .data
msg1 db	'Hello, programmers!',0xA,0xD 	
len1 equ $ - msg1			

msg2 db 'Welcome to the world of,', 0xA,0xD 
len2 equ $ - msg2 msg3 db 'Linux assembly programming! ' len3 equ $- msg3

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello, programmers!
Welcome to the world of,
Linux assembly programming!

द% ववच रत कदए

%assignEQU निर्देश जैसे संख्यात्मक स्थिरांक को परिभाषित करने के लिए निर्देश का उपयोग किया जा सकता है। यह निर्देश पुनर्वितरण की अनुमति देता है। उदाहरण के लिए, आप निरंतर कुल को परिभाषित कर सकते हैं -

%assign TOTAL 10

बाद में कोड में, आप इसे फिर से परिभाषित कर सकते हैं -

%assign  TOTAL  20

यह निर्देश केस-संवेदी है।

% निर्देशक को परिभाषित करता है

%defineनिर्देश संख्यात्मक और स्ट्रिंग स्थिरांक दोनों को परिभाषित करने की अनुमति देता है। यह निर्देश सी में #define के समान है। उदाहरण के लिए, आप निरंतर PTR को परिभाषित कर सकते हैं -

%define PTR [EBP+4]

उपरोक्त कोड PTR को [EBP + 4] से बदल देता है ।

यह निर्देश पुनर्वितरण की अनुमति भी देता है और यह केस-संवेदी है।

आईएनसी निर्देश

INC निर्देश का उपयोग किसी एक ऑपरेंड को बढ़ाने के लिए किया जाता है। यह एक एकल ऑपरेंड पर काम करता है जो या तो रजिस्टर में या मेमोरी में हो सकता है।

वाक्य - विन्यास

INC निर्देश में निम्नलिखित सिंटैक्स है -

INC destination

ऑपरेंड गंतव्य 8-बिट, 16-बिट या 32-बिट ऑपरेंड हो सकता है।

उदाहरण

INC EBX	     ; Increments 32-bit register
INC DL       ; Increments 8-bit register
INC [count]  ; Increments the count variable

डीईसी निर्देश

डीईसी निर्देश का उपयोग एक ऑपरेटर द्वारा एक को कम करने के लिए किया जाता है। यह एक एकल ऑपरेंड पर काम करता है जो या तो रजिस्टर में या मेमोरी में हो सकता है।

वाक्य - विन्यास

DEC निर्देश में निम्नलिखित सिंटैक्स है -

DEC destination

ऑपरेंड गंतव्य 8-बिट, 16-बिट या 32-बिट ऑपरेंड हो सकता है।

उदाहरण

segment .data
   count dw  0
   value db  15
	
segment .text
   inc [count]
   dec [value]
	
   mov ebx, count
   inc word [ebx]
	
   mov esi, value
   dec byte [esi]

ADD और SUB निर्देश

ADD और SUB निर्देशों का उपयोग बाइट, वर्ड और डबलवर्ड साइज में बाइनरी डेटा के सरल जोड़ / घटाव को जोड़ने के लिए किया जाता है, अर्थात, क्रमशः 8-बिट, 16-बिट या 32-बिट ऑपरेंड को जोड़ने या घटाने के लिए।

वाक्य - विन्यास

ADD और SUB निर्देशों के निम्नलिखित सिंटैक्स हैं -

ADD/SUB	destination, source

ADD / SUB निर्देश के बीच जगह ले सकता है -

  • रजिस्टर करने के लिए रजिस्टर करें
  • रजिस्टर करने के लिए मेमोरी
  • मेमोरी में रजिस्टर करें
  • निरंतर डेटा पर रजिस्टर करें
  • निरंतर डेटा के लिए मेमोरी

हालाँकि, अन्य निर्देशों की तरह, मेमोरी-टू-मेमोरी ऑपरेशन ADD / SUB निर्देशों का उपयोग करना संभव नहीं है। एक ADD या SUB ऑपरेशन ओवरफ़्लो और फ़्लैग को सेट या साफ़ करता है।

उदाहरण

निम्नलिखित उदाहरण उपयोगकर्ता से दो अंक पूछेगा, क्रमशः EAX और EBX रजिस्टर में अंकों को संग्रहीत करें, मूल्यों को जोड़ें, परिणाम को मेमोरी स्थान ' रेस ' में संग्रहीत करें और अंत में परिणाम प्रदर्शित करें।

SYS_EXIT  equ 1
SYS_READ  equ 3
SYS_WRITE equ 4
STDIN     equ 0
STDOUT    equ 1

segment .data 

   msg1 db "Enter a digit ", 0xA,0xD 
   len1 equ $- msg1 msg2 db "Please enter a second digit", 0xA,0xD len2 equ $- msg2 

   msg3 db "The sum is: "
   len3 equ $- msg3

segment .bss

   num1 resb 2 
   num2 resb 2 
   res resb 1    

section	.text
   global _start    ;must be declared for using gcc
	
_start:             ;tell linker entry point
   mov eax, SYS_WRITE         
   mov ebx, STDOUT         
   mov ecx, msg1         
   mov edx, len1 
   int 0x80                

   mov eax, SYS_READ 
   mov ebx, STDIN  
   mov ecx, num1 
   mov edx, 2
   int 0x80            

   mov eax, SYS_WRITE        
   mov ebx, STDOUT         
   mov ecx, msg2          
   mov edx, len2         
   int 0x80

   mov eax, SYS_READ  
   mov ebx, STDIN  
   mov ecx, num2 
   mov edx, 2
   int 0x80        

   mov eax, SYS_WRITE         
   mov ebx, STDOUT         
   mov ecx, msg3          
   mov edx, len3         
   int 0x80

   ; moving the first number to eax register and second number to ebx
   ; and subtracting ascii '0' to convert it into a decimal number
	
   mov eax, [num1]
   sub eax, '0'
	
   mov ebx, [num2]
   sub ebx, '0'

   ; add eax and ebx
   add eax, ebx
   ; add '0' to to convert the sum from decimal to ASCII
   add eax, '0'

   ; storing the sum in memory location res
   mov [res], eax

   ; print the sum 
   mov eax, SYS_WRITE        
   mov ebx, STDOUT
   mov ecx, res         
   mov edx, 1        
   int 0x80

exit:    
   
   mov eax, SYS_EXIT   
   xor ebx, ebx 
   int 0x80

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Enter a digit:
3
Please enter a second digit:
4
The sum is:
7

The program with hardcoded variables −

section	.text
   global _start    ;must be declared for using gcc
	
_start:             ;tell linker entry point
   mov	eax,'3'
   sub     eax, '0'
	
   mov 	ebx, '4'
   sub     ebx, '0'
   add 	eax, ebx
   add	eax, '0'
	
   mov 	[sum], eax
   mov	ecx,msg	
   mov	edx, len
   mov	ebx,1	;file descriptor (stdout)
   mov	eax,4	;system call number (sys_write)
   int	0x80	;call kernel
	
   mov	ecx,sum
   mov	edx, 1
   mov	ebx,1	;file descriptor (stdout)
   mov	eax,4	;system call number (sys_write)
   int	0x80	;call kernel
	
   mov	eax,1	;system call number (sys_exit)
   int	0x80	;call kernel
	
section .data
   msg db "The sum is:", 0xA,0xD 
   len equ $ - msg   
   segment .bss
   sum resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The sum is:
7

MUL / IMUL निर्देश

बाइनरी डेटा को गुणा करने के दो निर्देश हैं। MUL (Multiply) निर्देश अहस्ताक्षरित डेटा को संभालता है और IMUL (Integer Multiply) हस्ताक्षरित डेटा को संभालता है। दोनों निर्देश कैरी और ओवरफ्लो ध्वज को प्रभावित करते हैं।

वाक्य - विन्यास

MUL / IMUL निर्देशों का सिंटैक्स निम्नानुसार है -

MUL/IMUL multiplier

दोनों मामलों में गुणक एक संचायक में होगा, गुणक के आकार और गुणक के आधार पर और उत्पन्न उत्पाद को भी दो रजिस्टरों में भंडार के आकार के आधार पर संग्रहीत किया जाता है। निम्नलिखित अनुभाग तीन अलग-अलग मामलों के साथ MUL के निर्देशों की व्याख्या करता है -

अनु क्रमांक। परिदृश्यों
1

When two bytes are multiplied −

गुणक एएल रजिस्टर में है, और गुणक मेमोरी में या किसी अन्य रजिस्टर में एक बाइट है। उत्पाद कुल्हाड़ी में है। उत्पाद के उच्च-क्रम 8 बिट्स एएच में संग्रहीत हैं और निम्न-क्रम 8 बिट्स एएल में संग्रहीत हैं।

2

When two one-word values are multiplied −

गुणक एक्सएक्स रजिस्टर में होना चाहिए, और गुणक स्मृति या किसी अन्य रजिस्टर में एक शब्द है। उदाहरण के लिए, MUL DX जैसे निर्देश के लिए, आपको DX में गुणक और AX में गुणक को संग्रहीत करना होगा।

परिणामी उत्पाद एक डबलवर्ड है, जिसे दो रजिस्टरों की आवश्यकता होगी। उच्च-क्रम (सबसे बाएं) भाग को DX में संग्रहीत किया जाता है और निचले क्रम (सबसे दाएं) भाग को AX में संग्रहीत किया जाता है।

3

When two doubleword values are multiplied −

जब दो दोहरे मूल्यों को गुणा किया जाता है, तो गुणक EAX में होना चाहिए और गुणक मेमोरी में या किसी अन्य रजिस्टर में संग्रहीत एक डबलवर्ड मूल्य है। जेनरेट किया गया उत्पाद EDX में संग्रहीत होता है: EAX रजिस्टर, यानी, उच्च क्रम 32 बिट्स EDX रजिस्टर में संग्रहीत हो जाते हैं और निम्न क्रम 32-बिट्स EAX रजिस्टर में संग्रहीत होते हैं।

उदाहरण

MOV AL, 10
MOV DL, 25
MUL DL
...
MOV DL, 0FFH	; DL= -1
MOV AL, 0BEH	; AL = -66
IMUL DL

उदाहरण

निम्न उदाहरण 3 को 2 से गुणा करता है, और परिणाम प्रदर्शित करता है -

section	.text
   global _start    ;must be declared for using gcc
	
_start:             ;tell linker entry point

   mov	al,'3'
   sub     al, '0'
	
   mov 	bl, '2'
   sub     bl, '0'
   mul 	bl
   add	al, '0'
	
   mov 	[res], al
   mov	ecx,msg	
   mov	edx, len
   mov	ebx,1	;file descriptor (stdout)
   mov	eax,4	;system call number (sys_write)
   int	0x80	;call kernel
	
   mov	ecx,res
   mov	edx, 1
   mov	ebx,1	;file descriptor (stdout)
   mov	eax,4	;system call number (sys_write)
   int	0x80	;call kernel
	
   mov	eax,1	;system call number (sys_exit)
   int	0x80	;call kernel

section .data
msg db "The result is:", 0xA,0xD 
len equ $- msg   
segment .bss
res resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The result is:
6

DIV / IDIV निर्देश

डिवीजन ऑपरेशन दो तत्वों को उत्पन्न करता है - ए quotient और एक remainder। गुणन के मामले में, अतिप्रवाह नहीं होता है क्योंकि उत्पाद को रखने के लिए डबल-लंबाई रजिस्टरों का उपयोग किया जाता है। हालांकि, विभाजन के मामले में, अतिप्रवाह हो सकता है। यदि अतिप्रवाह होता है तो प्रोसेसर एक अवरोध उत्पन्न करता है।

DIV (डिवाइड) निर्देश का उपयोग अहस्ताक्षरित डेटा के लिए किया जाता है और IDIV (इंटेगर डिवाइड) का उपयोग हस्ताक्षरित डेटा के लिए किया जाता है।

वाक्य - विन्यास

DIV / IDIV निर्देश के लिए प्रारूप -

DIV/IDIV	divisor

लाभांश एक संचायक में है। दोनों निर्देश 8-बिट, 16-बिट या 32-बिट ऑपरेंड के साथ काम कर सकते हैं। ऑपरेशन सभी छह स्थिति ध्वज को प्रभावित करता है। निम्नलिखित अनुभाग विभिन्न ऑपरेंड आकार के साथ विभाजन के तीन मामलों की व्याख्या करता है -

अनु क्रमांक। परिदृश्यों
1

When the divisor is 1 byte −

लाभांश को AX रजिस्टर (16 बिट्स) में माना जाता है। विभाजन के बाद, भागफल AL रजिस्टर में चला जाता है और शेष AH रजिस्टर में चला जाता है।

2

When the divisor is 1 word −

लाभांश को 32 बिट लंबा और DX में माना जाता है: AX रजिस्टर। उच्च-क्रम वाले 16 बिट्स DX में हैं और लो-ऑर्डर 16 बिट्स AX में हैं। विभाजन के बाद, 16-बिट भागफल AX रजिस्टर में जाता है और 16-बिट शेष DX रजिस्टर में जाता है।

3

When the divisor is doubleword −

लाभांश 64 बिट लंबा और EDX में माना जाता है: EAX रजिस्टर। उच्च-क्रम वाले 32 बिट्स EDX में हैं और निम्न-क्रम वाले 32 बिट्स EAX में हैं। विभाजन के बाद, 32-बिट भागफल EAX रजिस्टर में जाता है और 32-बिट शेष EDX रजिस्टर में जाता है।

उदाहरण

निम्न उदाहरण 2 के साथ 8 को विभाजित करता है dividend 8 में संग्रहित है 16-bit AX register और यह divisor 2 में संग्रहित है 8-bit BL register

section	.text
   global _start    ;must be declared for using gcc
	
_start:             ;tell linker entry point
   mov	ax,'8'
   sub     ax, '0'
	
   mov 	bl, '2'
   sub     bl, '0'
   div 	bl
   add	ax, '0'
	
   mov 	[res], ax
   mov	ecx,msg	
   mov	edx, len
   mov	ebx,1	;file descriptor (stdout)
   mov	eax,4	;system call number (sys_write)
   int	0x80	;call kernel
	
   mov	ecx,res
   mov	edx, 1
   mov	ebx,1	;file descriptor (stdout)
   mov	eax,4	;system call number (sys_write)
   int	0x80	;call kernel
	
   mov	eax,1	;system call number (sys_exit)
   int	0x80	;call kernel
	
section .data
msg db "The result is:", 0xA,0xD 
len equ $- msg   
segment .bss
res resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The result is:
4

प्रोसेसर निर्देश सेट निर्देश और प्रदान करता है, या, XOR, परीक्षण, और बूलियन तर्क नहीं, जो कार्यक्रम की आवश्यकता के अनुसार बिट्स का परीक्षण, सेट और क्लीयर करता है।

इन निर्देशों का प्रारूप -

अनु क्रमांक। अनुदेश प्रारूप
1 तथा और ऑपरेंड 1, ऑपरेंड 2
2 या OR ऑपरेंड 1, ऑपरेंड 2
3 XOR XOR operand1, operand2
4 परीक्षा परीक्षण ऑपरेंड 1, ऑपरेंड 2
5 नहीं नहीं ऑपरेंड १

सभी मामलों में पहला ऑपरेंड या तो रजिस्टर में या मेमोरी में हो सकता है। दूसरा ऑपरेंड या तो रजिस्टर / मेमोरी या एक तत्काल (स्थिर) मान में हो सकता है। हालाँकि, मेमोरी-टू-मेमोरी ऑपरेशन संभव नहीं हैं। ये निर्देश ऑपरेंड के बिट्स की तुलना या मिलान करते हैं और सीएफ, ओएफ, पीएफ, एसएफ और जेडएफ झंडे सेट करते हैं।

और निर्देश

AND निर्देश बिटवाइज़ और ऑपरेशन करके लॉजिकल एक्सप्रेशंस को सपोर्ट करने के लिए उपयोग किया जाता है। बिटवाइज और ऑपरेशन रिटर्न 1, यदि दोनों ऑपरेंड्स से मेल खाती बिट्स 1 हैं, अन्यथा यह 0. लौटाता है। उदाहरण के लिए -

Operand1: 	0101
             Operand2: 	0011
----------------------------
After AND -> Operand1:	0001

AND ऑपरेशन का उपयोग एक या अधिक बिट्स को साफ़ करने के लिए किया जा सकता है। उदाहरण के लिए, मान लें कि बीएल रजिस्टर में 0011 1010 शामिल हैं। यदि आपको उच्च-क्रम के बिट्स को शून्य करने की आवश्यकता है, तो आपको और एफएफएच के साथ।

AND	BL,   0FH   ; This sets BL to 0000 1010

चलिए एक और उदाहरण लेते हैं। यदि आप यह जांचना चाहते हैं कि क्या दी गई संख्या विषम है या यहां तक ​​कि, एक साधारण परीक्षण संख्या की कम से कम महत्वपूर्ण बिट की जांच करना होगा। यदि यह 1 है, तो संख्या विषम है, अन्यथा संख्या सम है।

मान लें कि संख्या AL रजिस्टर में है, हम लिख सकते हैं -

AND	AL, 01H     ; ANDing with 0000 0001
JZ    EVEN_NUMBER

निम्नलिखित कार्यक्रम यह दिखाता है -

उदाहरण

section .text
   global _start            ;must be declared for using gcc
	
_start:                     ;tell linker entry point
   mov   ax,   8h           ;getting 8 in the ax 
   and   ax, 1              ;and ax with 1
   jz    evnn
   mov   eax, 4             ;system call number (sys_write)
   mov   ebx, 1             ;file descriptor (stdout)
   mov   ecx, odd_msg       ;message to write
   mov   edx, len2          ;length of message
   int   0x80               ;call kernel
   jmp   outprog

evnn:   
  
   mov   ah,  09h
   mov   eax, 4             ;system call number (sys_write)
   mov   ebx, 1             ;file descriptor (stdout)
   mov   ecx, even_msg      ;message to write
   mov   edx, len1          ;length of message
   int   0x80               ;call kernel

outprog:

   mov   eax,1              ;system call number (sys_exit)
   int   0x80               ;call kernel

section   .data
even_msg  db  'Even Number!' ;message showing even number
len1  equ  $ - even_msg odd_msg db 'Odd Number!' ;message showing odd number len2 equ $ - odd_msg

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Even Number!

कुल्हाड़ी रजिस्टर में एक अजीब अंक के साथ मान बदलें, जैसे -

mov  ax, 9h                  ; getting 9 in the ax

कार्यक्रम प्रदर्शित होगा:

Odd Number!

इसी तरह आप पूरे रजिस्टर को क्लियर कर सकते हैं और 00H के साथ कर सकते हैं।

या निर्देश

OR निर्देश बिटवाइज़ या ऑपरेशन करके तार्किक अभिव्यक्ति का समर्थन करने के लिए उपयोग किया जाता है। बिटवाइज़ या ऑपरेटर 1 लौटाता है, यदि दोनों या दोनों ऑपरेंड से मिलान बिट्स एक हैं। यह 0 देता है, यदि दोनों बिट्स शून्य हैं।

उदाहरण के लिए,

Operand1:     0101
             Operand2:     0011
----------------------------
After OR -> Operand1:    0111

OR ऑपरेशन का उपयोग एक या अधिक बिट्स सेट करने के लिए किया जा सकता है। उदाहरण के लिए, मान लें कि AL रजिस्टर में 0011 1010 शामिल हैं, आपको चार निम्न-क्रम बिट्स सेट करने की आवश्यकता है, आप इसे 0000 1111 अर्थात, FH मान के साथ कर सकते हैं।

OR BL, 0FH                   ; This sets BL to  0011 1111

उदाहरण

निम्न उदाहरण OR निर्देश को प्रदर्शित करता है। मान 5 और 3 को एएल और बीएल रजिस्टरों में क्रमशः स्टोर करें, फिर निर्देश,

OR AL, BL

AL रजिस्टर में 7 स्टोर करना चाहिए -

section .text
   global _start            ;must be declared for using gcc
	
_start:                     ;tell linker entry point
   mov    al, 5             ;getting 5 in the al
   mov    bl, 3             ;getting 3 in the bl
   or     al, bl            ;or al and bl registers, result should be 7
   add    al, byte '0'      ;converting decimal to ascii
	
   mov    [result],  al
   mov    eax, 4
   mov    ebx, 1
   mov    ecx, result
   mov    edx, 1 
   int    0x80
    
outprog:
   mov    eax,1             ;system call number (sys_exit)
   int    0x80              ;call kernel
	
section    .bss
result resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

7

XOR निर्देश

XOR निर्देश बिटवाइज XOR ऑपरेशन को लागू करता है। XOR ऑपरेशन परिणामी बिट को 1 पर सेट करता है, यदि और केवल यदि ऑपरेंड से बिट्स भिन्न हैं। यदि ऑपरेंड से बिट्स समान हैं (दोनों 0 या दोनों 1), परिणामी बिट को 0 पर मंजूरी दे दी गई है।

उदाहरण के लिए,

Operand1:     0101
             Operand2:     0011
----------------------------
After XOR -> Operand1:    0110

XORing एक ऑपरेंड अपने आप में ऑपरेंड को बदल देता है 0। इसका उपयोग किसी रजिस्टर को साफ़ करने के लिए किया जाता है।

XOR     EAX, EAX

परीक्षण निर्देश

परीक्षण निर्देश AND और ऑपरेशन के समान ही काम करता है, लेकिन AND अनुदेश के विपरीत, यह पहले ऑपरेंड को नहीं बदलता है। इसलिए, यदि हमें यह जांचने की आवश्यकता है कि रजिस्टर में कोई संख्या सम या विषम है, तो हम मूल संख्या को बदले बिना TEST निर्देश का उपयोग करके भी ऐसा कर सकते हैं।

TEST    AL, 01H
JZ      EVEN_NUMBER

निर्देश नहीं

नहीं अनुदेश बिटवाइज़ नहीं ऑपरेशन को कार्यान्वित करता है। ऑपरेशन एक ऑपरेंड में बिट्स को उलट नहीं करता है। ऑपरेंड या तो एक रजिस्टर में या मेमोरी में हो सकता है।

उदाहरण के लिए,

Operand1:    0101 0011
After NOT -> Operand1:    1010 1100

विधानसभा भाषा में सशर्त निष्पादन कई लूपिंग और ब्रांचिंग निर्देशों द्वारा पूरा किया जाता है। ये निर्देश एक कार्यक्रम में नियंत्रण के प्रवाह को बदल सकते हैं। सशर्त निष्पादन दो परिदृश्यों में देखा जाता है -

अनु क्रमांक। सशर्त निर्देश
1

Unconditional jump

यह JMP अनुदेश द्वारा किया जाता है। सशर्त निष्पादन में अक्सर एक निर्देश के पते पर नियंत्रण का हस्तांतरण शामिल होता है जो वर्तमान में निष्पादित अनुदेश का पालन नहीं करता है। नियंत्रणों का स्थानांतरण आगे हो सकता है, निर्देशों का एक नया सेट निष्पादित करने के लिए या पीछे, उसी चरणों को फिर से निष्पादित करने के लिए।

2

Conditional jump

यह स्थिति के आधार पर जम्प निर्देश j <कंडीशन> के सेट द्वारा किया जाता है। सशर्त निर्देश अनुक्रमिक प्रवाह को तोड़कर नियंत्रण स्थानांतरित करते हैं और वे आईपी में ऑफसेट मूल्य को बदलकर ऐसा करते हैं।

सशर्त निर्देशों पर चर्चा करने से पहले CMP निर्देश पर चर्चा करें।

सीएमपी निर्देश

सीएमपी निर्देश दो ऑपरेंड की तुलना करता है। यह आमतौर पर सशर्त निष्पादन में उपयोग किया जाता है। यह निर्देश मूल रूप से एक ऑपरेंड को दूसरे से तुलना करने के लिए घटाता है कि ऑपरेटर बराबर हैं या नहीं। यह गंतव्य या स्रोत ऑपरेंड को परेशान नहीं करता है। इसका उपयोग निर्णय लेने के लिए सशर्त कूद अनुदेश के साथ किया जाता है।

वाक्य - विन्यास

CMP destination, source

सीएमपी दो संख्यात्मक डेटा फ़ील्ड की तुलना करता है। गंतव्य ऑपरेंड या तो रजिस्टर में या मेमोरी में हो सकता है। स्रोत ऑपरेंड एक निरंतर (तत्काल) डेटा, रजिस्टर या मेमोरी हो सकता है।

उदाहरण

CMP DX,	00  ; Compare the DX value with zero
JE  L7      ; If yes, then jump to label L7
.
.
L7: ...

सीएमपी का उपयोग अक्सर यह तुलना करने के लिए किया जाता है कि काउंटर वैल्यू कितनी बार लूप चलाने की आवश्यकता है। निम्नलिखित विशिष्ट स्थिति पर विचार करें -

INC	EDX
CMP	EDX, 10	; Compares whether the counter has reached 10
JLE	LP1     ; If it is less than or equal to 10, then jump to LP1

बिना शर्त कूदो

जैसा कि पहले उल्लेख किया गया है, यह JMP अनुदेश द्वारा किया जाता है। सशर्त निष्पादन में अक्सर एक निर्देश के पते पर नियंत्रण का हस्तांतरण शामिल होता है जो वर्तमान में निष्पादित अनुदेश का पालन नहीं करता है। नियंत्रणों का स्थानांतरण आगे हो सकता है, निर्देशों का एक नया सेट निष्पादित करने के लिए या पीछे, उसी चरणों को फिर से निष्पादित करने के लिए।

वाक्य - विन्यास

JMP निर्देश एक लेबल नाम प्रदान करता है जहां नियंत्रण का प्रवाह तुरंत स्थानांतरित हो जाता है। JMP निर्देश का सिंटैक्स है -

JMP	label

उदाहरण

निम्नलिखित कोड स्निपेट JMP निर्देश दिखाता है -

MOV  AX, 00    ; Initializing AX to 0
MOV  BX, 00    ; Initializing BX to 0
MOV  CX, 01    ; Initializing CX to 1
L20:
ADD  AX, 01    ; Increment AX
ADD  BX, AX    ; Add AX to BX
SHL  CX, 1     ; shift left CX, this in turn doubles the CX value
JMP  L20       ; repeats the statements

सशर्त कूदो

यदि कुछ निर्दिष्ट स्थिति सशर्त कूद में संतुष्ट हैं, तो नियंत्रण प्रवाह को एक लक्ष्य निर्देश में स्थानांतरित किया जाता है। स्थिति और डेटा के आधार पर कई सशर्त कूद निर्देश हैं।

अंकगणितीय संचालन के लिए उपयोग किए गए हस्ताक्षरित डेटा पर उपयोग किए जाने वाले सशर्त कूद निर्देश निम्नलिखित हैं -

अनुदेश विवरण झंडे का परीक्षण किया गया
जेई / JZ बराबर कूदो या शून्य कूदो जेडएफ
JNE / JNZ जंप नॉट इक्वेल या जंप नॉट जीरो जेडएफ
जेजी / JNLE जंप ग्रेटर या जंप नॉट कम / इक्वल ओएफ, एसएफ, जेडएफ
संयुक्त विशेषज्ञ समूह / JNL जंप ग्रेटर / इक्वल या जंप नॉट कम ओएफ, एसएफ
जीएल / JNGE जंप कम या जंप नॉट ग्रेटर / इक्वल ओएफ, एसएफ
JLE / JNG जंप कम / बराबर या जंप नॉट ग्रेटर ओएफ, एसएफ, जेडएफ

तार्किक संचालन के लिए उपयोग किए गए अहस्ताक्षरित डेटा पर उपयोग किए जाने वाले सशर्त कूद निर्देश निम्नलिखित हैं -

अनुदेश विवरण झंडे का परीक्षण किया गया
जेई / JZ बराबर कूदो या शून्य कूदो जेडएफ
JNE / JNZ जंप नॉट इक्वेल या जंप नॉट जीरो जेडएफ
जावेद / JNBE ऊपर या नीचे कूदो नहीं / बराबर कूद सीएफ, जेडएफ
जॅ / JNB कूदो ऊपर / बराबर या नीचे नहीं कूद सीएफ़
जेबी / JNAE नीचे कूदो या ऊपर नहीं कूदो / बराबर सीएफ़
JBE / JNA नीचे कूदो / बराबर या कूदो ऊपर नहीं वायुसेना, सीएफ

निम्नलिखित सशर्त कूद निर्देशों के विशेष उपयोग हैं और झंडे के मूल्य की जांच करें -

अनुदेश विवरण झंडे का परीक्षण किया गया
JXCZ अगर सीएक्स शून्य है तो कूदो कोई नहीं
जे.सी. कूदो अगर कैरी सीएफ़
JNC जम्प इज़ नो कैरी सीएफ़
JO ओवरफ्लो होने पर कूदें का
JNO यदि कोई अतिप्रवाह नहीं तो कूदो का
जेपी / JPE कूद समता या कूद समता भी पीएफ
JNP / JPO जंप नो पैरिटी या जंप पैरिटी ऑड पीएफ
जे एस जंप साइन (नकारात्मक मूल्य) एस एफ
JNS जम्प नो साइन (सकारात्मक मूल्य) एस एफ

J <कंडीशन> निर्देश के सेट के लिए सिंटैक्स -

उदाहरण,

CMP	AL, BL
JE	EQUAL
CMP	AL, BH
JE	EQUAL
CMP	AL, CL
JE	EQUAL
NON_EQUAL: ...
EQUAL: ...

उदाहरण

निम्नलिखित कार्यक्रम तीन चर का सबसे बड़ा प्रदर्शित करता है। चर दोहरे अंकों के चर हैं। तीन वैरिएबल num1, num2 और num3 में क्रमशः 47, 22 और 31 मान हैं -

section	.text
   global _start         ;must be declared for using gcc

_start:	                 ;tell linker entry point
   mov   ecx, [num1]
   cmp   ecx, [num2]
   jg    check_third_num
   mov   ecx, [num2]
   
	check_third_num:

   cmp   ecx, [num3]
   jg    _exit
   mov   ecx, [num3]
   
	_exit:
   
   mov   [largest], ecx
   mov   ecx,msg
   mov   edx, len
   mov   ebx,1	;file descriptor (stdout)
   mov   eax,4	;system call number (sys_write)
   int   0x80	;call kernel
	
   mov   ecx,largest
   mov   edx, 2
   mov   ebx,1	;file descriptor (stdout)
   mov   eax,4	;system call number (sys_write)
   int   0x80	;call kernel
    
   mov   eax, 1
   int   80h

section	.data
   
   msg db "The largest digit is: ", 0xA,0xD 
   len equ $- msg 
   num1 dd '47'
   num2 dd '22'
   num3 dd '31'

segment .bss
   largest resb 2

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The largest digit is: 
47

JMP निर्देश का उपयोग लूप को लागू करने के लिए किया जा सकता है। उदाहरण के लिए, लूप-बॉडी को 10 बार निष्पादित करने के लिए निम्न कोड स्निपेट का उपयोग किया जा सकता है।

MOV	CL, 10
L1:
<LOOP-BODY>
DEC	CL
JNZ	L1

हालाँकि, प्रोसेसर निर्देश सेट में पुनरावृत्ति को लागू करने के लिए लूप निर्देशों का एक समूह शामिल है। मूल LOOP निर्देश में निम्नलिखित सिंटैक्स है -

LOOP 	label

जहां, लेबल लक्ष्य लेबल होता है जो जंप निर्देशों में लक्ष्य निर्देश की पहचान करता है। LOOP निर्देश मानता है किECX register contains the loop count। जब लूप निर्देश को निष्पादित किया जाता है, तो ECX रजिस्टर को हटा दिया जाता है और नियंत्रण लेबल पर कूद जाता है, जब तक कि ECX रजिस्टर मूल्य, अर्थात, काउंटर मूल्य शून्य तक पहुंच जाता है।

उपरोक्त कोड स्निपेट के रूप में लिखा जा सकता है -

mov ECX,10
l1:
<loop body>
loop l1

उदाहरण

निम्नलिखित कार्यक्रम स्क्रीन पर नंबर 1 से 9 तक प्रिंट करता है -

section	.text
   global _start        ;must be declared for using gcc
	
_start:	                ;tell linker entry point
   mov ecx,10
   mov eax, '1'
	
l1:
   mov [num], eax
   mov eax, 4
   mov ebx, 1
   push ecx
	
   mov ecx, num        
   mov edx, 1        
   int 0x80
	
   mov eax, [num]
   sub eax, '0'
   inc eax
   add eax, '0'
   pop ecx
   loop l1
	
   mov eax,1             ;system call number (sys_exit)
   int 0x80              ;call kernel
section	.bss
num resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

123456789:

संख्यात्मक डेटा को आमतौर पर बाइनरी सिस्टम में दर्शाया जाता है। अंकगणितीय निर्देश बाइनरी डेटा पर काम करते हैं। जब नंबर स्क्रीन पर प्रदर्शित होते हैं या कीबोर्ड से दर्ज किए जाते हैं, तो वे ASCII फॉर्म में होते हैं।

अब तक, हमने इस इनपुट डेटा को अंकगणितीय गणनाओं के लिए ASCII फॉर्म में बाइनरी में बदल दिया है और परिणाम को वापस बाइनरी में बदल दिया है। निम्न कोड यह दिखाता है -

section	.text
   global _start        ;must be declared for using gcc
	
_start:	                ;tell linker entry point
   mov	eax,'3'
   sub     eax, '0'
	
   mov 	ebx, '4'
   sub     ebx, '0'
   add 	eax, ebx
   add	eax, '0'
	
   mov 	[sum], eax
   mov	ecx,msg	
   mov	edx, len
   mov	ebx,1	         ;file descriptor (stdout)
   mov	eax,4	         ;system call number (sys_write)
   int	0x80	         ;call kernel
	
   mov	ecx,sum
   mov	edx, 1
   mov	ebx,1	         ;file descriptor (stdout)
   mov	eax,4	         ;system call number (sys_write)
   int	0x80	         ;call kernel
	
   mov	eax,1	         ;system call number (sys_exit)
   int	0x80	         ;call kernel
	
section .data
msg db "The sum is:", 0xA,0xD 
len equ $ - msg   
segment .bss
sum resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The sum is:
7

इस तरह के रूपांतरण, हालांकि, एक ओवरहेड है, और असेंबली भाषा प्रोग्रामिंग बाइनरी रूप में, अधिक कुशल तरीके से संख्याओं को संसाधित करने की अनुमति देता है। दशमलव संख्या को दो रूपों में दर्शाया जा सकता है -

  • ASCII फॉर्म
  • बीसीडी या बाइनरी कोडेड दशमलव रूप

ASCII प्रतिनिधित्व

ASCII प्रतिनिधित्व में, दशमलव संख्या ASCII वर्णों के स्ट्रिंग के रूप में संग्रहीत की जाती है। उदाहरण के लिए, दशमलव 1234 का मान इस प्रकार है -

31	32	33	34H

जहां, 31H 1 के लिए ASCII मान है, 2 के लिए 32H ASCII मान है, और इसी तरह। ASCII प्रतिनिधित्व में प्रसंस्करण संख्या के लिए चार निर्देश हैं -

  • AAA - एडीसीआई एडिशन के बाद समायोजित करें

  • AAS - घटाव के बाद ASCII समायोजित करें

  • AAM - ASCII गुणा के बाद समायोजित करें

  • AAD - विभाजन से पहले ASCII समायोजित करें

ये निर्देश किसी भी ऑपरेंड को नहीं लेते हैं और एएल रजिस्टर में आवश्यक ऑपरेंड मान लेते हैं।

निम्नलिखित उदाहरण अवधारणा को प्रदर्शित करने के लिए AAS निर्देश का उपयोग करता है -

section	.text
   global _start        ;must be declared for using gcc
	
_start:	                ;tell linker entry point
   sub     ah, ah
   mov     al, '9'
   sub     al, '3'
   aas
   or      al, 30h
   mov     [res], ax
	
   mov	edx,len	        ;message length
   mov	ecx,msg	        ;message to write
   mov	ebx,1	        ;file descriptor (stdout)
   mov	eax,4	        ;system call number (sys_write)
   int	0x80	        ;call kernel
	
   mov	edx,1	        ;message length
   mov	ecx,res	        ;message to write
   mov	ebx,1	        ;file descriptor (stdout)
   mov	eax,4	        ;system call number (sys_write)
   int	0x80	        ;call kernel
	
   mov	eax,1	        ;system call number (sys_exit)
   int	0x80	        ;call kernel
	
section	.data
msg db 'The Result is:',0xa	
len equ $ - msg			
section .bss
res resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The Result is:
6

बीसीडी प्रतिनिधित्व

बीसीडी प्रतिनिधित्व दो प्रकार के होते हैं -

  • अनपैक्ड बीसीडी प्रतिनिधित्व
  • पैक्ड बीसीडी प्रतिनिधित्व

अनपैक्ड बीसीडी प्रतिनिधित्व में, प्रत्येक बाइट एक दशमलव अंक के द्विआधारी समकक्ष को संग्रहीत करता है। उदाहरण के लिए, संख्या 1234 इस रूप में संग्रहीत है -

01	02	03	04H

इन नंबरों के प्रसंस्करण के लिए दो निर्देश हैं -

  • AAM - ASCII गुणा के बाद समायोजित करें

  • AAD - विभाजन से पहले ASCII समायोजित करें

चार ASCII समायोजित निर्देश, एएए, एएएस, एएएम, और एएडी का उपयोग अनपैक किए गए बीसीडी प्रतिनिधित्व के साथ भी किया जा सकता है। पैक्ड बीसीडी प्रतिनिधित्व में, प्रत्येक अंक को चार बिट्स का उपयोग करके संग्रहीत किया जाता है। दो दशमलव अंकों को एक बाइट में पैक किया जाता है। उदाहरण के लिए, संख्या 1234 इस रूप में संग्रहीत है -

12	34H

इन नंबरों के प्रसंस्करण के लिए दो निर्देश हैं -

  • DAA - जोड़ के बाद दशमलव समायोजित

  • DAS - दशमलव घटाव के बाद समायोजित करें

पैक्ड बीसीडी प्रतिनिधित्व में गुणन और विभाजन के लिए कोई समर्थन नहीं है।

उदाहरण

निम्न प्रोग्राम दो 5-अंकीय दशमलव संख्याओं को जोड़ता है और योग प्रदर्शित करता है। यह उपरोक्त अवधारणाओं का उपयोग करता है -

section	.text
   global _start        ;must be declared for using gcc

_start:	                ;tell linker entry point

   mov     esi, 4       ;pointing to the rightmost digit
   mov     ecx, 5       ;num of digits
   clc
add_loop:  
   mov 	al, [num1 + esi]
   adc 	al, [num2 + esi]
   aaa
   pushf
   or 	al, 30h
   popf
	
   mov	[sum + esi], al
   dec	esi
   loop	add_loop
	
   mov	edx,len	        ;message length
   mov	ecx,msg	        ;message to write
   mov	ebx,1	        ;file descriptor (stdout)
   mov	eax,4	        ;system call number (sys_write)
   int	0x80	        ;call kernel
	
   mov	edx,5	        ;message length
   mov	ecx,sum	        ;message to write
   mov	ebx,1	        ;file descriptor (stdout)
   mov	eax,4	        ;system call number (sys_write)
   int	0x80	        ;call kernel
	
   mov	eax,1	        ;system call number (sys_exit)
   int	0x80	        ;call kernel

section	.data
msg db 'The Sum is:',0xa	
len equ $ - msg			
num1 db '12345'
num2 db '23456'
sum db '     '

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The Sum is:
35801

हमने पहले ही अपने पिछले उदाहरणों में चर लंबाई के तार का उपयोग किया है। वैरिएबल की लंबाई के तार जितने आवश्यक हो सकते हैं। आम तौर पर, हम दो तरीकों से स्ट्रिंग की लंबाई निर्दिष्ट करते हैं -

  • स्पष्ट रूप से स्ट्रिंग की लंबाई भंडारण
  • एक प्रहरी वर्ण का उपयोग करना

हम $ लंबाई काउंटर प्रतीक का उपयोग करके स्पष्ट रूप से स्ट्रिंग लंबाई स्टोर कर सकते हैं जो स्थान काउंटर के वर्तमान मूल्य का प्रतिनिधित्व करता है। निम्नलिखित उदाहरण में -

msg  db  'Hello, world!',0xa ;our dear string
len  equ  $ - msg            ;length of our dear string

स्ट्रिंग चर संदेश के अंतिम चरित्र के बाद बाइट को $ अंक । इसलिए,$-msgस्ट्रिंग की लंबाई देता है। हम भी लिख सकते हैं

msg db 'Hello, world!',0xa ;our dear string
len equ 13                 ;length of our dear string

वैकल्पिक रूप से, आप स्ट्रिंग की लंबाई स्पष्ट रूप से संग्रहीत करने के बजाय स्ट्रिंग को परिसीमित करने के लिए एक अनुगामी प्रहरी वर्ण के साथ तारों को संग्रहीत कर सकते हैं। प्रहरी वर्ण एक विशेष वर्ण होना चाहिए जो एक स्ट्रिंग के भीतर प्रकट नहीं होता है।

उदाहरण के लिए -

message DB 'I am loving it!', 0

स्ट्रिंग निर्देश

प्रत्येक स्ट्रिंग निर्देश के लिए स्रोत ऑपरेंड, गंतव्य ऑपरेटर या दोनों की आवश्यकता हो सकती है। 32-बिट सेगमेंट के लिए, स्ट्रिंग निर्देश क्रमशः स्रोत और गंतव्य ऑपरेंड को इंगित करने के लिए ESI और EDI रजिस्टरों का उपयोग करते हैं।

16-बिट सेगमेंट के लिए, हालांकि, SI और DI रजिस्टरों का उपयोग क्रमशः स्रोत और गंतव्य को इंगित करने के लिए किया जाता है।

प्रसंस्करण स्ट्रिंग के लिए पांच बुनियादी निर्देश हैं। वे हैं -

  • MOVS - यह निर्देश मेमोरी स्थान से दूसरे स्थान पर 1 बाइट, वर्ड या डेटा का दोहराव चलता है।

  • LODS- यह निर्देश मेमोरी से लोड होता है। यदि ऑपरेंड एक बाइट का होता है, तो इसे AL रजिस्टर में लोड किया जाता है, यदि ऑपरेंड एक शब्द है, तो इसे AX रजिस्टर में लोड किया जाता है और EAX रजिस्टर में एक डबलवर्ड लोड किया जाता है।

  • STOS - यह निर्देश मेमोरी (AL, AX, या EAX) से डेटा को मेमोरी में स्टोर करता है।

  • CMPS- यह निर्देश मेमोरी में दो डेटा आइटम की तुलना करता है। डेटा एक बाइट आकार, शब्द या डबलवर्ड का हो सकता है।

  • SCAS - यह निर्देश एक रजिस्टर की सामग्री (AL, AX या EAX) की तुलना मेमोरी में किसी आइटम की सामग्री से करता है।

उपरोक्त निर्देश में से प्रत्येक में एक बाइट, शब्द और दोहराव संस्करण है, और दोहराव के उपसर्ग का उपयोग करके स्ट्रिंग निर्देशों को दोहराया जा सकता है।

ये निर्देश ES: DI और DS: SI जोड़ी का उपयोग करते हैं, जहां DI और SI रजिस्टर में मान्य ऑफसेट पते होते हैं जो मेमोरी में संग्रहीत बाइट्स को संदर्भित करते हैं। SI सामान्यतः DS (डेटा सेगमेंट) से जुड़ा होता है और DI हमेशा ES (अतिरिक्त सेगमेंट) से जुड़ा होता है।

डीएस: एसआई (या ईएसआई) और ईएस: डीआई (या ईडीआई) रजिस्टर क्रमशः स्रोत और गंतव्य ऑपरेंड को इंगित करते हैं। स्रोत ऑपरेंड को DS: SI (या ESI) और गंतव्य संचालक को ES: DI (या EDI) की स्मृति में माना जाता है।

16-बिट पतों के लिए, SI और DI रजिस्टरों का उपयोग किया जाता है, और 32-बिट पतों के लिए, ESI और EDI रजिस्टरों का उपयोग किया जाता है।

निम्न तालिका स्ट्रिंग निर्देशों और ऑपरेंड्स के ग्रहण किए गए स्थान के विभिन्न संस्करण प्रदान करती है।

मूल निर्देश पर संचालन करता है बाइट ऑपरेशन वर्ड ऑपरेशन डबल शब्द ऑपरेशन
mOVs ईएस: डीआई, डीएस: एसआई MOVSB MOVSW MOVSD
LODS कुल्हाड़ी, डीएस: एसआई LODSB LODSW LODSD
STOS ES: DI, AX STOSB STOSW STOSD
CMPS डीएस: एसआई, ईएस: डीआई CMPSB CMPSW CMPSD
एससीए ES: DI, AX SCASB SCASW SCASD

पुनरावृत्ति उपसर्ग

REP उपसर्ग, जब एक स्ट्रिंग निर्देश से पहले सेट किया जाता है, उदाहरण के लिए - REP MOVSB, सीएक्स रजिस्टर में रखे एक काउंटर के आधार पर निर्देश की पुनरावृत्ति का कारण बनता है। आरईपी निर्देश को निष्पादित करता है, सीएक्स को 1 से घटाता है, और जांचता है कि क्या सीएक्स शून्य है। सीएक्स शून्य होने तक यह अनुदेश प्रसंस्करण को दोहराता है।

दिशा ध्वज (DF) ऑपरेशन की दिशा निर्धारित करता है।

  • ऑपरेशन को दाएं से बाएं करने के लिए CLD (स्पष्ट दिशा ध्वज, DF = 0) का उपयोग करें।
  • ऑपरेशन को बाएं से दाएं बनाने के लिए STD (सेट डायरेक्शन फ्लैग, DF = 1) का उपयोग करें।

आरईपी उपसर्ग में भी निम्नलिखित विविधताएं हैं:

  • REP: यह बिना शर्त दोहराना है। यह सीएक्स के शून्य होने तक ऑपरेशन को दोहराता है।

  • REPE या REPZ: यह सशर्त दोहराव है। यह ऑपरेशन को दोहराता है जबकि शून्य ध्वज बराबर / शून्य इंगित करता है। यह तब रुकता है जब ZF न के बराबर / शून्य या जब CX शून्य होता है इंगित करता है।

  • REPNE या REPNZ: यह सशर्त दोहराव भी है। यह ऑपरेशन को दोहराता है जबकि शून्य ध्वज बराबर / शून्य इंगित नहीं करता है। यह तब रुकता है जब ZF बराबर / शून्य इंगित करता है या जब CX शून्य पर शून्य हो जाता है।

हम पहले ही चर्चा कर चुके हैं कि कोडांतरक के लिए डेटा परिभाषा निर्देश का उपयोग चर के लिए भंडारण आवंटित करने के लिए किया जाता है। चर को कुछ विशिष्ट मूल्य के साथ आरंभ किया जा सकता है। प्रारंभिक मूल्य को हेक्साडेसिमल, दशमलव या बाइनरी फॉर्म में निर्दिष्ट किया जा सकता है।

उदाहरण के लिए, हम निम्नलिखित में से किसी भी एक शब्द चर 'महीने' को परिभाषित कर सकते हैं -

MONTHS	DW	12
MONTHS	DW	0CH
MONTHS	DW	0110B

डेटा परिभाषा निर्देशों का उपयोग एक आयामी सरणी को परिभाषित करने के लिए भी किया जा सकता है। आइए हम संख्याओं के एक आयामी आयाम को परिभाषित करते हैं।

NUMBERS	DW  34,  45,  56,  67,  75, 89

उपरोक्त परिभाषा 34, 45, 56, 67, 75, 89 संख्याओं के साथ प्रत्येक छह शब्दों की एक सरणी घोषित करती है। यह लगातार मेमोरी स्पेस के 2x6 = 12 बाइट्स आवंटित करता है। पहली संख्या का प्रतीकात्मक पता NUMBERS होगा और दूसरी संख्या NUMBERS + 2 और इसी तरह होगी।

एक और उदाहरण लेते हैं। आप आकार 8 की सूची नामक एक सरणी को परिभाषित कर सकते हैं, और शून्य के साथ सभी मानों को आरंभ कर सकते हैं, जैसे -

INVENTORY   DW  0
            DW  0
            DW  0
            DW  0
            DW  0
            DW  0
            DW  0
            DW  0

जिसे संक्षिप्त रूप में दिया जा सकता है -

INVENTORY   DW  0, 0 , 0 , 0 , 0 , 0 , 0 , 0

TIMES निर्देश का उपयोग एक ही मूल्य के लिए कई आरंभीकरण के लिए भी किया जा सकता है। समय का उपयोग करते हुए, इन्वेंट्री सरणी को इस प्रकार परिभाषित किया जा सकता है:

INVENTORY TIMES 8 DW 0

उदाहरण

निम्न उदाहरण 3-तत्व सरणी x को परिभाषित करके उपरोक्त अवधारणाओं को प्रदर्शित करता है, जो तीन मूल्यों को संग्रहीत करता है: 2, 3 और 4. यह सरणी में मान जोड़ता है और योग 9 प्रदर्शित करता है -

section	.text
   global _start   ;must be declared for linker (ld)
	
_start:	
 		
   mov  eax,3      ;number bytes to be summed 
   mov  ebx,0      ;EBX will store the sum
   mov  ecx, x     ;ECX will point to the current element to be summed

top:  add  ebx, [ecx]

   add  ecx,1      ;move pointer to next element
   dec  eax        ;decrement counter
   jnz  top        ;if counter not 0, then loop again

done: 

   add   ebx, '0'
   mov  [sum], ebx ;done, store result in "sum"

display:

   mov  edx,1      ;message length
   mov  ecx, sum   ;message to write
   mov  ebx, 1     ;file descriptor (stdout)
   mov  eax, 4     ;system call number (sys_write)
   int  0x80       ;call kernel
	
   mov  eax, 1     ;system call number (sys_exit)
   int  0x80       ;call kernel

section	.data
global x
x:    
   db  2
   db  4
   db  3

sum: 
   db  0

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

9

असेंबली लैंग्वेज में प्रक्रियाएँ या सबरूटीन बहुत महत्वपूर्ण होते हैं, क्योंकि असेंबली लैंग्वेज प्रोग्राम आकार में बड़े होते हैं। प्रक्रियाओं को एक नाम से पहचाना जाता है। इस नाम के बाद, प्रक्रिया का शरीर वर्णित है जो एक अच्छी तरह से परिभाषित कार्य करता है। प्रक्रिया का अंत एक वापसी विवरण द्वारा इंगित किया गया है।

वाक्य - विन्यास

एक प्रक्रिया को परिभाषित करने के लिए वाक्यविन्यास निम्नलिखित है -

proc_name:
   procedure body
   ...
   ret

कॉल निर्देश का उपयोग करके प्रक्रिया को दूसरे फ़ंक्शन से कहा जाता है। कॉल निर्देश में एक तर्क के रूप में तथाकथित प्रक्रिया का नाम होना चाहिए जैसा कि नीचे दिखाया गया है -

CALL proc_name

आरटीई अनुदेश का उपयोग करके कॉलिंग प्रक्रिया को नियंत्रण प्रक्रिया कहा जाता है।

उदाहरण

आइए हम एक बहुत ही सरल प्रक्रिया लिखते हैं जिसका नाम योग है जो ECX और EDX में संग्रहीत चर को जोड़ता है और EAX सहयोगी में राशि लौटाता है -

section	.text
   global _start        ;must be declared for using gcc
	
_start:	                ;tell linker entry point
   mov	ecx,'4'
   sub     ecx, '0'
	
   mov 	edx, '5'
   sub     edx, '0'
	
   call    sum          ;call sum procedure
   mov 	[res], eax
   mov	ecx, msg	
   mov	edx, len
   mov	ebx,1	        ;file descriptor (stdout)
   mov	eax,4	        ;system call number (sys_write)
   int	0x80	        ;call kernel
	
   mov	ecx, res
   mov	edx, 1
   mov	ebx, 1	        ;file descriptor (stdout)
   mov	eax, 4	        ;system call number (sys_write)
   int	0x80	        ;call kernel
	
   mov	eax,1	        ;system call number (sys_exit)
   int	0x80	        ;call kernel
sum:
   mov     eax, ecx
   add     eax, edx
   add     eax, '0'
   ret
	
section .data
msg db "The sum is:", 0xA,0xD 
len equ $- msg   

segment .bss
res resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The sum is:
9

ढेर डेटा संरचना

स्टैक मेमोरी में एक सरणी जैसी डेटा संरचना होती है जिसमें डेटा को स्टोर किया जा सकता है और स्टैक के 'शीर्ष' नामक स्थान से हटाया जा सकता है। संग्रहीत किए जाने वाले डेटा को स्टैक में 'पुश' किया जाता है और पुनर्प्राप्त किए जाने वाले डेटा को स्टैक से 'पॉपअप' किया जाता है। स्टैक एक LIFO डेटा संरचना है, अर्थात, पहले संग्रहीत डेटा अंतिम पुनर्प्राप्त किया जाता है।

असेंबली भाषा स्टैक ऑपरेशन के लिए दो निर्देश प्रदान करती है: PUSH और POP। इन निर्देशों में सिंटैक्स होते हैं -

PUSH    operand
POP     address/register

स्टैक सेगमेंट में आरक्षित मेमोरी स्पेस का उपयोग स्टैक को लागू करने के लिए किया जाता है। स्टैक को लागू करने के लिए रजिस्टरों एसएस और ईएसपी (या एसपी) का उपयोग किया जाता है। स्टैक के शीर्ष, जो स्टैक में सम्मिलित अंतिम डेटा आइटम को इंगित करता है, एसएस द्वारा इंगित किया जाता है: ईएसपी रजिस्टर, जहां एसएस स्टैक सेगमेंट की शुरुआत को इंगित करता है और एसपी (या ईएसपी) ऑफसेट देता है। ढेर खंड।

स्टैक कार्यान्वयन की निम्नलिखित विशेषताएं हैं -

  • केवल words या doublewords स्टैक में बचाया जा सकता है, बाइट नहीं।

  • स्टैक रिवर्स दिशा में बढ़ता है, अर्थात, निचले मेमोरी एड्रेस की ओर

  • स्टैक में डाले गए अंतिम आइटम के लिए स्टैक के शीर्ष बिंदु; यह पिछले शब्द के निचले बाइट को इंगित करता है।

जैसा कि हमने कुछ उपयोग करने से पहले स्टैक में रजिस्टरों के मूल्यों को संग्रहीत करने के बारे में चर्चा की; इसे निम्नलिखित तरीके से किया जा सकता है -

; Save the AX and BX registers in the stack
PUSH    AX
PUSH    BX

; Use the registers for other purpose
MOV	AX, VALUE1
MOV 	BX, VALUE2
...
MOV 	VALUE1, AX
MOV	VALUE2, BX

; Restore the original values
POP	BX
POP	AX

उदाहरण

निम्न प्रोग्राम पूरे ASCII वर्ण सेट को प्रदर्शित करता है। मुख्य कार्यक्रम डिस्प्ले नामक एक प्रक्रिया को कॉल करता है, जो ASCII वर्ण सेट प्रदर्शित करता है।

section	.text
   global _start        ;must be declared for using gcc
	
_start:	                ;tell linker entry point
   call    display
   mov	eax,1	        ;system call number (sys_exit)
   int	0x80	        ;call kernel
	
display:
   mov    ecx, 256
	
next:
   push    ecx
   mov     eax, 4
   mov     ebx, 1
   mov     ecx, achar
   mov     edx, 1
   int     80h
	
   pop     ecx	
   mov	dx, [achar]
   cmp	byte [achar], 0dh
   inc	byte [achar]
   loop    next
   ret
	
section .data
achar db '0'

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz{|}
...
...

एक पुनरावर्ती प्रक्रिया वह है जो स्वयं को कॉल करती है। दो तरह की पुनरावृत्ति होती है: प्रत्यक्ष और अप्रत्यक्ष। प्रत्यक्ष पुनरावर्तन में, प्रक्रिया स्वयं को बुलाती है और अप्रत्यक्ष पुनरावर्तन में, पहली प्रक्रिया दूसरी प्रक्रिया कहती है, जो बदले में पहली प्रक्रिया कहती है।

कई गणितीय एल्गोरिदम में पुनरावृत्ति देखी जा सकती है। उदाहरण के लिए, किसी संख्या के भाज्य की गणना के मामले पर विचार करें। किसी संख्या का गुणन समीकरण द्वारा दिया जाता है -

Fact (n) = n * fact (n-1) for n > 0

उदाहरण के लिए: 5 का भाज्य 1 x 2 x 3 x 4 x 5 = 5 x 4 का भाज्य है और यह पुनरावर्ती प्रक्रिया दिखाने का एक अच्छा उदाहरण हो सकता है। प्रत्येक पुनरावर्ती एल्गोरिथ्म में एक समाप्ति स्थिति होनी चाहिए, अर्थात, किसी शर्त के पूरा होने पर कार्यक्रम के पुनरावर्ती कॉल को रोक दिया जाना चाहिए। तथ्यात्मक एल्गोरिथ्म के मामले में, अंतिम स्थिति तब पहुंच जाती है जब n 0 होता है।

निम्नलिखित कार्यक्रम से पता चलता है कि विधानसभा भाषा में फैक्टरियल एन को कैसे लागू किया जाता है। कार्यक्रम को सरल रखने के लिए, हम गुट 3 की गणना करेंगे।

section	.text
   global _start         ;must be declared for using gcc
	
_start:                  ;tell linker entry point

   mov bx, 3             ;for calculating factorial 3
   call  proc_fact
   add   ax, 30h
   mov  [fact], ax
    
   mov	  edx,len        ;message length
   mov	  ecx,msg        ;message to write
   mov	  ebx,1          ;file descriptor (stdout)
   mov	  eax,4          ;system call number (sys_write)
   int	  0x80           ;call kernel

   mov   edx,1            ;message length
   mov	  ecx,fact       ;message to write
   mov	  ebx,1          ;file descriptor (stdout)
   mov	  eax,4          ;system call number (sys_write)
   int	  0x80           ;call kernel
    
   mov	  eax,1          ;system call number (sys_exit)
   int	  0x80           ;call kernel
	
proc_fact:
   cmp   bl, 1
   jg    do_calculation
   mov   ax, 1
   ret
	
do_calculation:
   dec   bl
   call  proc_fact
   inc   bl
   mul   bl        ;ax = al * bl
   ret

section	.data
msg db 'Factorial 3 is:',0xa	
len equ $ - msg			

section .bss
fact resb 1

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Factorial 3 is:
6

मैक्रो लिखना विधानसभा भाषा में मॉड्यूलर प्रोग्रामिंग सुनिश्चित करने का एक और तरीका है।

  • मैक्रो निर्देशों का एक अनुक्रम है, जिसे एक नाम से सौंपा गया है और कार्यक्रम में कहीं भी इस्तेमाल किया जा सकता है।

  • NASM में, मैक्रोज़ के साथ परिभाषित किया गया है %macro तथा %endmacro निर्देशों।

  • मैक्रो% मैक्रो निर्देश के साथ शुरू होता है और% एंडमैक्रो निर्देश के साथ समाप्त होता है।

स्थूल परिभाषा के लिए सिंटैक्स -

%macro macro_name  number_of_params
<macro body>
%endmacro

जहाँ, number_of_params संख्या मापदंडों को निर्दिष्ट करता है , मैक्रो_ name मैक्रो का नाम निर्दिष्ट करता है।

मैक्रो को आवश्यक मापदंडों के साथ मैक्रो नाम का उपयोग करके लागू किया जाता है। जब आपको किसी प्रोग्राम में कई बार निर्देशों का उपयोग करने की आवश्यकता होती है, तो आप उन निर्देशों को एक मैक्रो में डाल सकते हैं और हर समय निर्देशों को लिखने के बजाय इसका उपयोग कर सकते हैं।

उदाहरण के लिए, कार्यक्रमों के लिए एक बहुत ही सामान्य आवश्यकता स्क्रीन में पात्रों की एक स्ट्रिंग लिखना है। वर्णों की एक स्ट्रिंग प्रदर्शित करने के लिए, आपको निम्नलिखित निर्देशों की आवश्यकता है -

mov	edx,len	    ;message length
mov	ecx,msg	    ;message to write
mov	ebx,1       ;file descriptor (stdout)
mov	eax,4       ;system call number (sys_write)
int	0x80        ;call kernel

चरित्र स्ट्रिंग प्रदर्शित करने के उपरोक्त उदाहरण में, INT 80H फ़ंक्शन कॉल द्वारा EAX, EBX, ECX और EDX को पंजीकृत किया गया है। इसलिए, हर बार जब आपको स्क्रीन पर प्रदर्शित करने की आवश्यकता होती है, तो आपको इन रजिस्टरों को स्टैक पर सहेजने की आवश्यकता है, INT 80H को आह्वान करें और फिर स्टैक से रजिस्टरों के मूल मूल्य को पुनर्स्थापित करें। इसलिए, डेटा को बचाने और पुनर्स्थापित करने के लिए दो मैक्रोज़ लिखना उपयोगी हो सकता है।

हमने देखा है कि, IMUL, IDIV, INT आदि जैसे कुछ निर्देशों को कुछ विशिष्ट रजिस्टरों में संग्रहीत करने के लिए कुछ जानकारी की आवश्यकता होती है और कुछ विशिष्ट रजिस्टर (एस) में मान भी लौटाते हैं। यदि प्रोग्राम पहले से ही महत्वपूर्ण डेटा रखने के लिए उन रजिस्टरों का उपयोग कर रहा था, तो इन रजिस्टरों के मौजूदा डेटा को स्टैक में सहेजा जाना चाहिए और निर्देश के निष्पादित होने के बाद पुनर्स्थापित किया जाना चाहिए।

उदाहरण

निम्नलिखित उदाहरण मैक्रोज़ को परिभाषित और उपयोग करते हुए दिखाता है -

; A macro with two parameters
; Implements the write system call
   %macro write_string 2 
      mov   eax, 4
      mov   ebx, 1
      mov   ecx, %1
      mov   edx, %2
      int   80h
   %endmacro
 
section	.text
   global _start            ;must be declared for using gcc
	
_start:                     ;tell linker entry point
   write_string msg1, len1               
   write_string msg2, len2    
   write_string msg3, len3  
	
   mov eax,1                ;system call number (sys_exit)
   int 0x80                 ;call kernel

section	.data
msg1 db	'Hello, programmers!',0xA,0xD 	
len1 equ $ - msg1 msg2 db 'Welcome to the world of,', 0xA,0xD len2 equ $- msg2 

msg3 db 'Linux assembly programming! '
len3 equ $- msg3

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello, programmers!
Welcome to the world of,
Linux assembly programming!

सिस्टम किसी भी इनपुट या आउटपुट डेटा को बाइट्स की धारा मानता है। तीन मानक फ़ाइल धाराएँ हैं -

  • मानक इनपुट (स्टडिन),
  • मानक आउटपुट (स्टडआउट), और
  • मानक त्रुटि (stderr)।

फ़ाइल डिस्क्रिप्टर

file descriptorफ़ाइल आईडी के रूप में एक फ़ाइल को असाइन किया गया 16-बिट पूर्णांक है। जब कोई नई फ़ाइल बनाई जाती है या कोई मौजूदा फ़ाइल खोली जाती है, तो फ़ाइल को एक्सेस करने के लिए फ़ाइल डिस्क्रिप्टर का उपयोग किया जाता है।

मानक फ़ाइल धाराओं का फ़ाइल विवरणक - stdin, stdout तथा stderr क्रमशः 0, 1 और 2 हैं।

फ़ाइल सूचक

file pointerबाइट के संदर्भ में फ़ाइल में बाद में पढ़ने / लिखने के संचालन के लिए स्थान निर्दिष्ट करता है। प्रत्येक फ़ाइल को बाइट्स के अनुक्रम के रूप में माना जाता है। प्रत्येक खुली फ़ाइल एक फ़ाइल पॉइंटर से जुड़ी होती है जो फ़ाइल की शुरुआत के सापेक्ष बाइट्स में एक ऑफसेट निर्दिष्ट करती है। जब कोई फ़ाइल खोली जाती है, तो फ़ाइल पॉइंटर को शून्य पर सेट किया जाता है।

फ़ाइल हैंडलिंग सिस्टम कॉल

निम्न तालिका फ़ाइल हैंडलिंग से संबंधित सिस्टम कॉल का संक्षेप में वर्णन करती है -

% eax नाम % EBX % ECX % EDX
2 sys_fork संरचना pt_regs - -
3 sys_read अहस्ताक्षरित int चार * size_t
4 sys_write अहस्ताक्षरित int कास्ट चार * size_t
5 sys_open कास्ट चार * पूर्णांक पूर्णांक
6 sys_close अहस्ताक्षरित int - -
8 sys_creat कास्ट चार * पूर्णांक -
19 sys_lseek अहस्ताक्षरित int off_t अहस्ताक्षरित int

सिस्टम कॉल का उपयोग करने के लिए आवश्यक चरण वही हैं, जैसा कि हमने पहले चर्चा की थी -

  • EAX रजिस्टर में सिस्टम कॉल नंबर डालें।
  • रजिस्टर EBB, ECX, आदि में सिस्टम कॉल के तर्कों को स्टोर करें।
  • प्रासंगिक व्यवधान (80h) को बुलाओ।
  • परिणाम आमतौर पर EAX रजिस्टर में वापस आ जाता है।

फ़ाइल बनाना और खोलना

फ़ाइल बनाने और खोलने के लिए, निम्न कार्य करें -

  • EAX रजिस्टर में सिस्टम कॉल sys_creat () नंबर 8 डालें।
  • फ़ाइल नाम को EBX रजिस्टर में रखें।
  • ECX रजिस्टर में फाइल परमिशन डालें।

सिस्टम कॉल EAX रजिस्टर में बनाई गई फ़ाइल के फ़ाइल डिस्क्रिप्टर को लौटाता है, त्रुटि के मामले में, त्रुटि कोड EAX रजिस्टर में होता है।

एक मौजूदा फ़ाइल खोलना

मौजूदा फ़ाइल खोलने के लिए, निम्न कार्य करें -

  • EAX रजिस्टर में सिस्टम कॉल sys_open () नंबर 5 डालें।
  • फ़ाइल नाम को EBX रजिस्टर में रखें।
  • फ़ाइल एक्सेस मोड को ECX रजिस्टर में रखें।
  • EDX रजिस्टर में फाइल परमिशन डालें।

सिस्टम कॉल EAX रजिस्टर में बनाई गई फ़ाइल के फ़ाइल डिस्क्रिप्टर को लौटाता है, त्रुटि के मामले में, त्रुटि कोड EAX रजिस्टर में होता है।

फ़ाइल एक्सेस मोड के बीच, सबसे अधिक उपयोग किया जाता है: रीड-ओनली (0), राइट-ओनली (1), और रीड-राइट (2)।

एक फ़ाइल से पढ़ना

किसी फ़ाइल से पढ़ने के लिए, निम्न कार्य करें -

  • EAX रजिस्टर में सिस्टम कॉल sys_read () नंबर 3 डालें।

  • फ़ाइल डिस्क्रिप्टर को EBX रजिस्टर में डालें।

  • पॉइंटर को ECX रजिस्टर में इनपुट बफर में डालें।

  • EDX रजिस्टर में बफर साइज, यानी पढ़ने के लिए बाइट्स की संख्या डालें।

EAX रजिस्टर में पढ़ी गई बाइट की संख्या को सिस्टम कॉल रिटर्न देता है, त्रुटि के मामले में, त्रुटि कोड EAX रजिस्टर में है।

एक फ़ाइल के लिए लेखन

फ़ाइल में लिखने के लिए, निम्नलिखित कार्य करें -

  • EAX रजिस्टर में सिस्टम कॉल sys_write () नंबर 4 डालें।

  • फ़ाइल डिस्क्रिप्टर को EBX रजिस्टर में डालें।

  • ईसीएक्स रजिस्टर में आउटपुट बफर को पॉइंटर डालें।

  • EDX रजिस्टर में बफर साइज, यानी लिखने के लिए बाइट्स की संख्या डालें।

EAX रजिस्टर में लिखी गई बाइट की वास्तविक संख्या को सिस्टम कॉल रिटर्न करता है, त्रुटि के मामले में, त्रुटि कोड EAX रजिस्टर में होता है।

फ़ाइल बंद करना

फ़ाइल बंद करने के लिए, निम्न कार्य करें -

  • EAX रजिस्टर में सिस्टम कॉल sys_close () नंबर 6 डालें।
  • फ़ाइल डिस्क्रिप्टर को EBX रजिस्टर में डालें।

EAX रजिस्टर में त्रुटि, त्रुटि कोड होने पर सिस्टम कॉल रिटर्न।

एक फ़ाइल को अद्यतन करना

फ़ाइल अपडेट करने के लिए, निम्न कार्य करें -

  • EAX रजिस्टर में सिस्टम कॉल sys_lseek () नंबर 19 डालें।
  • फ़ाइल डिस्क्रिप्टर को EBX रजिस्टर में डालें।
  • ऑफसेट मान को ECX रजिस्टर में डालें।
  • EDX रजिस्टर में ऑफसेट के लिए संदर्भ स्थिति रखें।

संदर्भ स्थिति हो सकती है:

  • फ़ाइल की शुरुआत - मान 0
  • वर्तमान स्थिति - मूल्य 1
  • फ़ाइल का अंत - मान 2

EAX रजिस्टर में त्रुटि, त्रुटि कोड होने पर सिस्टम कॉल रिटर्न।

उदाहरण

निम्न प्रोग्राम myfile.txt नामक एक फाइल बनाता है और खोलता है , और इस फाइल में एक पाठ 'वेलकम टू ट्युटोरियल पॉइंट' लिखता है। अगला, प्रोग्राम फ़ाइल से पढ़ता है और डेटा को बफर नाम की जानकारी में संग्रहीत करता है । अंत में, यह जानकारी में संग्रहीत पाठ को प्रदर्शित करता है ।

section	.text
   global _start         ;must be declared for using gcc
	
_start:                  ;tell linker entry point
   ;create the file
   mov  eax, 8
   mov  ebx, file_name
   mov  ecx, 0777        ;read, write and execute by all
   int  0x80             ;call kernel
	
   mov [fd_out], eax
    
   ; write into the file
   mov	edx,len          ;number of bytes
   mov	ecx, msg         ;message to write
   mov	ebx, [fd_out]    ;file descriptor 
   mov	eax,4            ;system call number (sys_write)
   int	0x80             ;call kernel
	
   ; close the file
   mov eax, 6
   mov ebx, [fd_out]
    
   ; write the message indicating end of file write
   mov eax, 4
   mov ebx, 1
   mov ecx, msg_done
   mov edx, len_done
   int  0x80
    
   ;open the file for reading
   mov eax, 5
   mov ebx, file_name
   mov ecx, 0             ;for read only access
   mov edx, 0777          ;read, write and execute by all
   int  0x80
	
   mov  [fd_in], eax
    
   ;read from file
   mov eax, 3
   mov ebx, [fd_in]
   mov ecx, info
   mov edx, 26
   int 0x80
    
   ; close the file
   mov eax, 6
   mov ebx, [fd_in]
   int  0x80 
	
   ; print the info 
   mov eax, 4
   mov ebx, 1
   mov ecx, info
   mov edx, 26
   int 0x80
       
   mov	eax,1             ;system call number (sys_exit)
   int	0x80              ;call kernel

section	.data
file_name db 'myfile.txt'
msg db 'Welcome to Tutorials Point'
len equ  $-msg

msg_done db 'Written to file', 0xa
len_done equ $-msg_done

section .bss
fd_out resb 1
fd_in  resb 1
info resb  26

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Written to file
Welcome to Tutorials Point

sys_brk()सिस्टम कॉल को कर्नेल द्वारा प्रदान किया जाता है, बाद में इसे स्थानांतरित करने की आवश्यकता के बिना मेमोरी आवंटित करने के लिए। यह कॉल मेमोरी में एप्लिकेशन इमेज के ठीक पीछे मैमोरी आवंटित करता है। यह सिस्टम फ़ंक्शन आपको डेटा अनुभाग में उच्चतम उपलब्ध पता सेट करने की अनुमति देता है।

यह सिस्टम कॉल एक पैरामीटर लेता है, जिसे सेट करने के लिए आवश्यक उच्चतम मेमोरी एड्रेस है। यह मान EBX रजिस्टर में संग्रहीत है।

किसी भी त्रुटि के मामले में, sys_brk () रिटर्न -1 या नकारात्मक त्रुटि कोड को ही लौटाता है। निम्न उदाहरण गतिशील मेमोरी आवंटन को दर्शाता है।

उदाहरण

निम्नलिखित प्रोग्राम sys_brk () सिस्टम कॉल का उपयोग करके 16kb मेमोरी आवंटित करता है -

section	.text
   global _start         ;must be declared for using gcc
	
_start:	                 ;tell linker entry point

   mov	eax, 45		 ;sys_brk
   xor	ebx, ebx
   int	80h

   add	eax, 16384	 ;number of bytes to be reserved
   mov	ebx, eax
   mov	eax, 45		 ;sys_brk
   int	80h
	
   cmp	eax, 0
   jl	exit	;exit, if error 
   mov	edi, eax	 ;EDI = highest available address
   sub	edi, 4		 ;pointing to the last DWORD  
   mov	ecx, 4096	 ;number of DWORDs allocated
   xor	eax, eax	 ;clear eax
   std			 ;backward
   rep	stosd            ;repete for entire allocated area
   cld			 ;put DF flag to normal state
	
   mov	eax, 4
   mov	ebx, 1
   mov	ecx, msg
   mov	edx, len
   int	80h		 ;print a message

exit:
   mov	eax, 1
   xor	ebx, ebx
   int	80h
	
section	.data
msg    	db	"Allocated 16 kb of memory!", 10
len     equ	$ - msg

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Allocated 16 kb of memory!