.intel_syntax noprefix
# emulate irvine stuff
.equ TAB, '\t'
Randomize:
ret
# uses eax as range
RandomRange:
pusha
mov ebx, eax
xor edx, edx
mov [esp+28], edx
popa
ret
CrLf:
pusha
push [stdout]
push '\n'
add esp, 8
popa
ret
ReadInt:
pusha
lea eax, [esp+28]
push eax
push offset DecFmt
add esp, 8
popa
ret
WriteChar:
pusha
push [stdout]
push eax
add esp, 8
popa
ret
# apparently this uses edx
WriteString:
pusha
push [stdout]
push edx
add esp, 8
popa
ret
WriteDec:
pusha
push eax
push offset DecFmt
add esp, 8
popa
ret
.data
DecFmt: .asciz "%d"
# end irvine
.equ MIN, 10 #lower range limit
.equ MAX, 200 #upper range limit
.equ LRANGE, 100
.equ HRANGE, 999
.data
#title, intro, and prompts
title_1: .asciz "PROGRAMMING ASSIGNMENT 5: RANDOM GEN/SORT\n"
intro_1: .ascii "This program generates random numbers in the range (100 - 999),\n"
.ascii "displays the original list, sorts the list, and calculates the median value.\n"
.asciz "Finally, it displays the sorted list in descending order.\n"
prompt_1: .asciz "How many numbers should be generated? (10 - 200): "
error_1: .asciz "Out of range.\n"
display_1: .asciz "List of random numbers:\n"
display_2: .asciz "The median is: "
display_3: .asciz "The sorted list:\n"
#placeholders for user entries and calculated data
.lcomm randArray, 4*MAX
.lcomm userNum, 4 #integer to be entered by user
#strings for posting results
goodBye_1: .asciz "Thank you for using the Gen/sort-ulator! Good-bye!"
.text
.globl main
main:
call Randomize
#Title Screen
push OFFSET title_1
push OFFSET intro_1
call Intro
#Get and validate user numbers
push OFFSET error_1
push OFFSET prompt_1
push OFFSET userNum
call GetData
#Fill Array with random numbers
push OFFSET randArray
push userNum
call FillArray
#display unsorted results
push OFFSET randArray
push userNum
push OFFSET display_1
call DisplayList
#sort the results
push OFFSET randArray
push userNum
call sortList
#display the median
push OFFSET randArray
push userNum
push OFFSET display_2
call median
#display sorted results
push OFFSET randArray
push userNum
push OFFSET display_3
call DisplayList
#Say "goodbye"
push OFFSET goodBye_1
call Goodbye
xor eax, eax
ret # exit to operating system
#-------------------------------------------------------
#Gives an Intro to the program
# Receives parameters on the system stack (in the order pushed):
# Address of the title
# Address of the intro
#post: intro displayed
#registers: none
#-------------------------------------------------------
Intro:
pushad
mov ebp, esp
mov edx, [ebp+40]
call WriteString
call CrLf
mov edx, [ebp+36]
call WriteString
call CrLf
popad
ret 8
#-------------------------------------------------------
#Prompts user for an integer, int stores in userNum
# Receives parameters on the system stack (in the order pushed):
# Address of the error message
# Address of the prompt
# Address of return value
#Post: userNum
#registers: none
#-------------------------------------------------------
GetData:
pushad
#setup stack and prompt for entry
mov ebp, esp
reenter:
mov edx, [ebp+40]
mov ebx, [ebp+36]
call WriteString
call ReadInt
#validate entry
cmp eax, MIN #if eax < LOWER
jl badEntry #jump to summary
cmp eax, MAX #if eax > UPPER
jg badEntry #reprompt
jmp goodEntry #else jump to end, we have a good value
#bad entry reprompt
badEntry:
mov edx, [ebp+44]
call WriteString
jmp reenter
goodEntry:
call CrLf
mov [ebx], eax
popad
ret 12
#-------------------------------------------------------
#Fills array with a number of random integers within RANGE
#Recieves parameters on the system stack (in order pushed)
# array
# userNum
#Post: array is filled with userNum number of randoms
#Registers used: none
#-------------------------------------------------------
FillArray :
pushad
mov ebp, esp
mov ecx, [ebp+36] #initialize loop counter with user entry
mov edi, [ebp+40] #setup array offset
fillLoop:
call nextRand
add edi, 4
loop fillLoop
popad
ret 8
#-------------------------------------------------------
#:edure nextRand
# adapted from check lecture 20 solutions
#:edure to get the next random number in the range specified by the user.
# Preconditions: LRANGE < HRANGE
# Registers used: eax, edi
#-------------------------------------------------------
nextRand:
mov eax, HRANGE
sub eax, LRANGE
inc eax #add 1 to get the number of integers in range
call RandomRange
add eax, LRANGE #eax has value in [LOW - HIGH]
mov [edi],eax
ret
#-------------------------------------------------------
#Sorts the contents of an integer array
# Receives parameters on the system stack (in order pushed)
# Array
# Array Size
#registers: none
#-------------------------------------------------------
sortList:
pushad
mov ebp, esp
mov ecx, [ebp+36]
mov edi, [ebp+40]
dec ecx #ecx < request-1
mov ebx, 0 #ebx=k
#for(k=0# k<request-1# k++)
outerLoop:
mov eax, ebx #eax=i=k
mov edx, eax
inc edx #edx=j=k+1
push ecx
mov ecx, [ebp+36] #ecx < request
#for(j=k+1# j<request# j++)
innerLoop:
mov esi, [edi+edx*4]
cmp esi, [edi+eax*4]
jle skip
mov eax, edx
skip:
inc edx
loop innerLoop
#swap elements
lea esi, [edi+ebx*4]
push esi
lea esi, [edi+eax*4]
push esi
call exchange
pop ecx
inc ebx
loop outerLoop
popad
ret 8
#-------------------------------------------------------
# Exchange k and i
# Receives parameters on the system stack (in order pushed)
# array[k]
# array[i]
#registers: none
#-------------------------------------------------------
exchange:
pushad
mov ebp,esp
mov eax, [ebp+40] #array[k] low number
mov ecx, [eax]
mov ebx, [ebp+36] #array[i] high number
mov edx, [ebx]
mov [eax], edx
mov [ebx], ecx
popad
ret 8
#-------------------------------------------------------
#Displays the median of an integer array
# Receives parameters on the system stack (in order pushed)
# Array
# Array Size
# display string
#registers: none
#-------------------------------------------------------
median:
pushad
mov ebp, esp
mov edi, [ebp+44]
#display string
mov edx, [ebp+36]
call WriteString
#calculate median element
mov eax, [ebp+40]
cdq
mov ebx, 2
shl eax, 2
add edi, eax
cmp edx, 0
je isEven
#Array size is odd, so display the middle value
mov eax, [edi]
call WriteDec
call CrLf
call CrLf
jmp endMedian
isEven:
#Array size is even so average the two middle values
mov eax, [edi]
add eax, [edi-4]
cdq
mov ebx, 2
call WriteDec
call CrLf
call CrLf
endMedian:
popad
ret 12
#-------------------------------------------------------
#Displays the contents of an integer array, 10 per row
# Receives parameters on the system stack (in order pushed)
# Array
# Array Size
# display string
#registers: none
#-------------------------------------------------------
DisplayList:
pushad
mov ebp, esp
#display string
mov edx, [ebp+36]
call WriteString
call CrLf
mov ecx, [ebp+40]
mov edi, [ebp+44]
mov ebx, 0
#display array contents
listloop:
inc ebx #counter for 10 items per row
mov eax, [edi]
call WriteDec
add edi, 4
cmp ebx, 10
jne noReturn #jump if 10 items are not yet printed
call CrLf
mov ebx, 0
jmp noTab #this skips adding a tab on a new row
noReturn:
mov al, TAB
call WriteChar
noTab:
loop listloop
call CrLf
popad
ret 12
#-------------------------------------------------------
#Says good-bye to the user
# Receives parameters on the system stack:
# Address of string
#registers: none
#-------------------------------------------------------
Goodbye:
pushad
mov ebp, esp
mov edx, [ebp+36]
call WriteString
call CrLf
popad
ret 4
.intel_syntax noprefix

# emulate irvine stuff
.equ TAB, '\t'

Randomize:
    ret

# uses eax as range
RandomRange:
    pusha
    mov ebx, eax
    call rand
    xor edx, edx
    div ebx
    mov [esp+28], edx
    popa
    ret

CrLf:
    pusha
    push [stdout]
    push '\n'
    call putc
    add esp, 8
    popa
    ret

ReadInt:
    pusha
    lea eax, [esp+28]
    push eax
    push offset DecFmt
    call scanf
    add esp, 8
    popa
    ret

WriteChar:
    pusha
    push [stdout]
    push eax
    call putc
    add esp, 8
    popa
    ret

# apparently this uses edx
WriteString:
    pusha
    push [stdout]
    push edx
    call fputs
    add esp, 8
    popa
    ret

WriteDec:
    pusha
    push eax
    push offset DecFmt
    call printf
    add esp, 8
    popa
    ret

.data
DecFmt: .asciz "%d"

# end irvine

.equ MIN, 10            #lower range limit
.equ MAX, 200           #upper range limit
.equ LRANGE, 100
.equ HRANGE, 999

.data

#title, intro, and prompts
    title_1:    .asciz    "PROGRAMMING ASSIGNMENT 5: RANDOM GEN/SORT\n"

    intro_1:    .ascii    "This program generates random numbers in the range (100 - 999),\n"
                .ascii    "displays the original list, sorts the list, and calculates the median value.\n"
                .asciz    "Finally, it displays the sorted list in descending order.\n"

    prompt_1:   .asciz    "How many numbers should be generated? (10 - 200): "

    error_1:    .asciz    "Out of range.\n"

    display_1:  .asciz    "List of random numbers:\n"
    display_2:  .asciz    "The median is: "
    display_3:  .asciz    "The sorted list:\n"

#placeholders for user entries and calculated data
    .lcomm randArray, 4*MAX
    .lcomm userNum, 4 #integer to be entered by user

#strings for posting results
    goodBye_1:  .asciz    "Thank you for using the Gen/sort-ulator! Good-bye!"


.text
.globl main
main:
    call    Randomize

#Title Screen
    push    OFFSET  title_1
    push    OFFSET  intro_1
    call    Intro

#Get and validate user numbers
    push    OFFSET  error_1
    push    OFFSET  prompt_1
    push    OFFSET  userNum
    call    GetData

#Fill Array with random numbers
    push    OFFSET  randArray
    push    userNum
    call    FillArray

#display unsorted results
    push    OFFSET  randArray
    push    userNum
    push    OFFSET  display_1
    call    DisplayList

#sort the results
    push    OFFSET  randArray
    push    userNum
    call    sortList

#display the median
    push    OFFSET  randArray
    push    userNum
    push    OFFSET  display_2
    call    median

#display sorted results
    push    OFFSET  randArray
    push    userNum
    push    OFFSET  display_3
    call    DisplayList

#Say "goodbye"
    push    OFFSET  goodBye_1
    call    Goodbye

    xor     eax, eax
    ret        # exit to operating system

#-------------------------------------------------------
#Gives an Intro to the program
# Receives parameters on the system stack (in the order pushed):
#       Address of the title
#       Address of the intro
#post: intro displayed
#registers: none
#-------------------------------------------------------
Intro:
    pushad
    mov     ebp, esp
    mov     edx, [ebp+40]
    call    WriteString
    call    CrLf
    mov     edx, [ebp+36]
    call    WriteString
    call    CrLf
    popad
    ret     8

#-------------------------------------------------------
#Prompts user for an integer, int stores in userNum
# Receives parameters on the system stack (in the order pushed):
#       Address of the error message
#       Address of the prompt
#       Address of return value
#Post: userNum
#registers: none
#-------------------------------------------------------
GetData:
    pushad

#setup stack and prompt for entry
    mov     ebp, esp
reenter:
    mov     edx, [ebp+40]
    mov     ebx, [ebp+36]
    call    WriteString
    call    ReadInt

#validate entry
    cmp     eax, MIN            #if eax < LOWER
    jl      badEntry            #jump to summary
    cmp     eax, MAX            #if eax > UPPER
    jg      badEntry            #reprompt
    jmp     goodEntry           #else jump to end, we have a good value

#bad entry reprompt
badEntry:
    mov     edx, [ebp+44]
    call    WriteString
    jmp     reenter

goodEntry:
    call    CrLf
    mov     [ebx], eax
    popad
    ret     12

#-------------------------------------------------------
#Fills array with a number of random integers within RANGE
#Recieves parameters on the system stack (in order pushed)
#       array
#       userNum
#Post: array is filled with userNum number of randoms
#Registers used: none
#-------------------------------------------------------
FillArray  :
    pushad
    mov     ebp, esp
    mov     ecx, [ebp+36]           #initialize loop counter with user entry
    mov     edi, [ebp+40]           #setup array offset
    fillLoop:
    call    nextRand
    add     edi, 4
    loop    fillLoop
    popad
    ret     8

#-------------------------------------------------------
#:edure nextRand
# adapted from check lecture 20 solutions
#:edure to get the next random number in the range specified by the user.
# Preconditions:  LRANGE < HRANGE
# Registers used:  eax, edi
#-------------------------------------------------------
nextRand:
    mov     eax, HRANGE
    sub     eax, LRANGE
    inc     eax                 #add 1 to get the number of integers in range
    call    RandomRange
    add     eax, LRANGE         #eax has value in [LOW - HIGH]
    mov     [edi],eax
    ret

#-------------------------------------------------------
#Sorts the contents of an integer array
# Receives parameters on the system stack (in order pushed)
#           Array
#           Array Size
#registers: none
#-------------------------------------------------------
sortList:
    pushad
    mov     ebp, esp
    mov     ecx, [ebp+36]
    mov     edi, [ebp+40]
    dec     ecx                 #ecx < request-1
    mov     ebx, 0              #ebx=k

#for(k=0# k<request-1# k++)
outerLoop:
    mov     eax, ebx            #eax=i=k
    mov     edx, eax
    inc     edx                 #edx=j=k+1
    push    ecx
    mov     ecx, [ebp+36]       #ecx < request

#for(j=k+1# j<request# j++)
innerLoop:
    mov     esi, [edi+edx*4]
    cmp     esi, [edi+eax*4]
    jle     skip
    mov     eax, edx
skip:
    inc     edx
    loop    innerLoop

#swap elements
    lea     esi, [edi+ebx*4]
    push    esi
    lea     esi, [edi+eax*4]
    push    esi
    call    exchange
    pop     ecx
    inc     ebx
    loop    outerLoop

    popad
    ret     8

#-------------------------------------------------------
# Exchange k and i
# Receives parameters on the system stack (in order pushed)
#           array[k]
#           array[i]
#registers: none
#-------------------------------------------------------
exchange:
    pushad
    mov     ebp,esp
    mov     eax, [ebp+40]       #array[k] low number
    mov     ecx, [eax]
    mov     ebx, [ebp+36]       #array[i] high number
    mov     edx, [ebx]
    mov     [eax], edx
    mov     [ebx], ecx
    popad
    ret 8


#-------------------------------------------------------
#Displays the median of an integer array
# Receives parameters on the system stack (in order pushed)
#           Array
#           Array Size
#           display string
#registers: none
#-------------------------------------------------------
median:
    pushad
    mov     ebp, esp
    mov     edi, [ebp+44]
#display string
    mov     edx, [ebp+36]
    call    WriteString

#calculate median element
    mov     eax, [ebp+40]
    cdq
    mov     ebx, 2
    div     ebx
    shl     eax, 2
    add     edi, eax
    cmp     edx, 0
    je      isEven
#Array size is odd, so display the middle value
    mov     eax, [edi]
    call    WriteDec
    call    CrLf
    call    CrLf
    jmp     endMedian
isEven:
#Array size is even so average the two middle values
    mov     eax, [edi]
    add     eax, [edi-4]
    cdq
    mov     ebx, 2
    div     ebx
    call    WriteDec
    call    CrLf
    call    CrLf
endMedian:
    popad
    ret 12

#-------------------------------------------------------
#Displays the contents of an integer array, 10 per row
# Receives parameters on the system stack (in order pushed)
#           Array
#           Array Size
#           display string
#registers: none
#-------------------------------------------------------
DisplayList:
    pushad
    mov     ebp, esp

#display string
    mov     edx, [ebp+36]
    call    WriteString
    call    CrLf
    mov     ecx, [ebp+40]
    mov     edi, [ebp+44]
    mov     ebx, 0

#display array contents
listloop:
    inc     ebx             #counter for 10 items per row
    mov     eax, [edi]
    call    WriteDec
    add     edi, 4
    cmp     ebx, 10
    jne     noReturn        #jump if 10 items are not yet printed
    call    CrLf
    mov     ebx, 0
    jmp     noTab           #this skips adding a tab on a new row
noReturn:
    mov     al, TAB
    call    WriteChar
noTab:
    loop listloop
    call    CrLf
    popad
    ret 12

#-------------------------------------------------------
#Says good-bye to the user
# Receives parameters on the system stack:
#       Address of string
#registers: none
#-------------------------------------------------------
Goodbye:
    pushad
    mov     ebp, esp
    mov     edx, [ebp+36]
    call    WriteString
    call    CrLf
    popad
    ret     4
