Procedures and Recursion used in Assembly Language:
By nature in Assembly there is an inclination towards to be big size of Assembly program. Also there is some way to keep remain in smaller in code size by subroutines or procedures. The article is concerned with the procedures and its various kind of implementation with example.
Procedures in Assembly
Assembly language programs intend to be large in size. Procedures or subroutines are very vital in assembly language which are identified by a name. Following this name, the body of the procedure is defined which accomplishes a well-defined job. End of the procedure is indicated by a return statement.
Syntax proc_name: procedure body ... ret
The procedure is called from another function by using the CALL instruction. The CALL instruction should have the name of the called procedure as an argument as shown below:
CALL proc_name
The called procedure returns the control to the calling procedure by using the RET instruction.
Example: Write a very simple procedure named sum that adds the variables stored in the ECX and EDX register and returns the sum in the EAX register.
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
OUTPUT:
The sum is:
9
Stacks Data Structure
A stack is an array-like data structure in the memory in which data can be stored and removed from a location called the ‘top’ of the stack. The data desires to be stored is ‘pushed’ into the stack and data to be retrieved is ‘popped’ out from the stack in LIFO structure. So, the data stored first is retrieved last. Assembly language provides two instructions for stack operations: PUSH and POP. These instructions have syntaxes like:
PUSH operand POP address/register
The memory space used in the stack segment is used for implementing stack. The registers SS and ESP (or SP) are used for implementing the stack. The top of the stack, which points to the last data item injected into the stack is pointed to by the SS: ESP register, where the SS register points to the opening of the stack segment and the SP (or ESP) gives the offset into the stack segment. The stack implementation has the following characteristics:
Only words or doublewords could be saved into the stack, not a byte. The stack grows in the reverse direction. For example, toward the lower memory address. The top of the stack points to the last item inserted in the stack; it points to the lower byte of the last word inserted. For storing the values of the registers in the stack before using them for some use; it can be done in following way:
; 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
Example
The following program displays the entire ASCII character set. The main program calls a procedure named display, which displays the ASCII character set.
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'
OUTPUT:
0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz{|}
…
…
Recursion in Assembly
A recursive procedure is one that calls itself. There are two types of recursion:
- direct and
- indirect
In direct recursion, the procedure calls itself and in indirect recursion, the first procedure calls a second procedure, which in turn calls the first procedure. Recursion could be observed in numerous mathematical algorithms. For instance, suppose the case of calculating the factorial of a number. Factorial of a number is given by the equation:
Fact (n) = n * fact (n-1) for n > 0
For instance: factorial of 5 is 1 x 2 x 3 x 4 x 5 = 5 x factorial of 4 and this can be a good instance of showing a recursive procedure. Every recursive algorithm must have an ending condition. In the case of factorial algorithm, the end condition is reached when n is 0.
Example: The following program shows how factorial n is implemented in assembly language. To keep the program simple, we will calculate factorial 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
OUTPUT:
Factorial 3 is:
6