Montaż - procedury

Procedury lub podprogramy są bardzo ważne w języku asemblera, ponieważ programy w asemblerze mają zwykle duży rozmiar. Procedury są identyfikowane nazwą. Po tej nazwie opisana jest treść procedury, która wykonuje dobrze zdefiniowaną pracę. Zakończenie procedury sygnalizowane jest zwrotem.

Składnia

Poniżej znajduje się składnia definiująca procedurę -

proc_name:
   procedure body
   ...
   ret

Procedura jest wywoływana z innej funkcji za pomocą instrukcji CALL. Instrukcja CALL powinna mieć nazwę wywoływanej procedury jako argument, jak pokazano poniżej -

CALL proc_name

Wywołana procedura zwraca sterowanie do procedury wywołującej przy użyciu instrukcji RET.

Przykład

Napiszmy bardzo prostą procedurę o nazwie suma, która dodaje zmienne przechowywane w rejestrze ECX i EDX i zwraca sumę w rejestrze 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

Kiedy powyższy kod jest kompilowany i wykonywany, daje następujący wynik -

The sum is:
9

Struktura danych stosów

Stos to macierzowa struktura danych w pamięci, w której dane mogą być przechowywane i usuwane z miejsca zwanego „górą” stosu. Dane, które mają być przechowywane, są „wypychane” do stosu, a dane do pobrania są „wyrzucane” ze stosu. Stos to struktura danych LIFO, tj. Dane zapisane jako pierwsze są pobierane jako ostatnie.

Język asemblera udostępnia dwie instrukcje dotyczące operacji na stosie: PUSH i POP. Te instrukcje mają składnię taką jak -

PUSH    operand
POP     address/register

Przestrzeń pamięci zarezerwowana w segmencie stosu jest używana do implementacji stosu. Rejestry SS i ESP (lub SP) są używane do implementacji stosu. Szczyt stosu, który wskazuje na ostatni element danych wstawiony do stosu, jest wskazywany przez rejestr SS: ESP, gdzie rejestr SS wskazuje początek segmentu stosu, a SP (lub ESP) podaje przesunięcie do segment stosu.

Implementacja stosu ma następujące cechy -

  • Tylko words lub doublewords można zapisać na stosie, a nie bajt.

  • Stos rośnie w odwrotnym kierunku, tj. W kierunku niższego adresu pamięci

  • Góra stosu wskazuje na ostatni element włożony do stosu; wskazuje na niższy bajt ostatniego wstawionego słowa.

Jak rozmawialiśmy o przechowywaniu wartości rejestrów w stosie przed użyciem ich do jakiegoś użytku; można to zrobić w następujący sposób -

; 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

Przykład

Poniższy program wyświetla cały zestaw znaków ASCII. Program główny wywołuje procedurę o nazwie display , która wyświetla zestaw znaków 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'

Kiedy powyższy kod jest kompilowany i wykonywany, daje następujący wynik -

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